再談Java雙括弧技巧:不規(guī)范的語(yǔ)法?
Java雙括弧技巧回顧
初看 cgaolei 翻譯的 Java技巧之雙括弧初始化 一文,走馬觀花,只知用法,未細(xì)看后面的解釋。蔚為驚艷,心里想 Java 竟然有這么神奇的語(yǔ)法而一直未得知。因?yàn)樵诔跏蓟蠒r(shí)確實(shí)方便不少。原來(lái)做某些測(cè)試要初始化集合時(shí)會(huì)用到 commons-lang 包和 JDK 的 Arrays 工具類,現(xiàn)在知道可以這么用了:
- Map map = new HashMap() {{
- put("Name", "Unmi");
- put("QQ", "1125535");
- }};
- List stooges = new ArrayList() {{
- add("Larry");
- add("Moe");
- add("Curly");
- }};
看起來(lái)都是在一條語(yǔ)句里完成,而不需要分步驟寫成:
- Map map = new HashMap();
- map.put("Name","Unmi");
- map.put("QQ","1125535");
一不小心沒(méi)好好理解的人可能以為它是什么特別的語(yǔ)法,關(guān)鍵是大括號(hào)連一塊了,原作者也是在故作姿態(tài),美其名曰:雙括弧語(yǔ)法(double-brace syntax)。真是亂花漸欲迷人眼,其實(shí)就是匿名類加初始?jí)K。該文有解釋:***層括弧 實(shí)際是定義了一個(gè)內(nèi)部匿名類 (Anonymous Inner Class),第二層括弧 實(shí)際上是一個(gè)實(shí)例初始化塊 (instance initializer block),這個(gè)塊在內(nèi)部匿名類構(gòu)造時(shí)被執(zhí)行。
那怎么去更好理解它呢?如果我們寫成如下的方式應(yīng)該會(huì)更好理解吧,提個(gè)技巧,在 Eclipse 中對(duì)***段代碼按下 Ctrl + Shift + F 就如下了:
- Map map = new HashMap() {
- {
- put("Name", "Unmi");
- put("QQ", "1125535");
- }
- };
其實(shí)就是匿名類啊,會(huì)創(chuàng)建出一個(gè) HashMap 的子類來(lái),匿名類中一個(gè) {} 括起來(lái)的初始化塊,里面自然可放置初始化代碼。{} 塊中的代碼編譯后會(huì)放到 <init>(),也就是構(gòu)造方法中去,所以可用來(lái)初始化實(shí)例。如果是寫在 TestDoubleBrace 類中,編譯后你會(huì)看到會(huì)生成 TestDoubleBrace$1.class 文件,反編譯該文件內(nèi)容是:
- final class com.unmi.TestDoubleBrace$1 extends java.util.HashMap{ //創(chuàng)建了一個(gè) HashMap 的子類 TestDoubleBracke$1
- com.unmi.TestDoubleBrace$1();
- Code:
- 0: aload_0
- 1: invokespecial #8; //Method java/util/HashMap."<init>":()V //{} 中的代碼放到了構(gòu)造方法中去了
- 4: aload_0
- 5: ldc #10; //String Name
- 7: ldc #12; //String Unmi
- 9: invokevirtual #14; //Method put:(Ljava/lang/Object;Ljava/lang/Object;)Ljava/lang/Object;
- 12: pop
- 13: aload_0
- 14: ldc #18; //String QQ
- 16: ldc #20; //String 1125535
- 18: invokevirtual #14; //Method put:(Ljava/lang/Object;Ljava/lang/Object;)Ljava/lang/Object;
- 21: pop
- 22: return
- }
所以說(shuō)白了,什么雙括弧語(yǔ)法啊,就是代碼寫得不規(guī)范,才使得那么的令人費(fèi)解。如果還不能理解,再列兩個(gè)慣用代碼來(lái):
- JFrame frame = new JFrame();
- frame.addMouseListener(new MouseAdapter() {
- public void mouseClicked(MouseEvent e) {
- // do womething here.
- }
- });
- Thread thread = new Thread() {{ // 也學(xué)著樣把大括號(hào)也連一塊寫了
- this.setName("作業(yè)處理線程");
- }// 如果不重新定義 run() 方法,那么后面那個(gè)大括號(hào)也能與這個(gè)并一塊
- public void run() {
- // do something here.
- }
- };
- thread.start();
應(yīng)該沒(méi)問(wèn)題了吧,上面是事件監(jiān)聽器和多線程常用的寫法,如果他不把大括號(hào)連在一起,而是規(guī)范的寫代碼,相信您一開始也不會(huì)對(duì)所謂的 Double Brace Syntax 有太多的困惑。要說(shuō)這種初始化方法運(yùn)用到集合中還挺方便的,只是無(wú)端的多了些匿名類。
剛開始我看到這種Java雙括弧寫法也是把它奉若圣經(jīng),對(duì)它只一知半解,昨天在用 XStream 把一個(gè)對(duì)象生成 XML 文件時(shí),其中有一個(gè) List 屬性,我就借用了這種雙括符法來(lái)初始化元素,結(jié)果生成的 XML 文件走了樣,原因是 XStream 的 Converter 能處理 ArrayList,但無(wú)法很好的處理生成的 ArrayList 的匿名子類。因此才回頭認(rèn)真的重新審視了一番這個(gè)所謂的雙括符初始化語(yǔ)法。
以上就是對(duì)Java雙括弧技巧的一些思考。本文來(lái)自隔葉黃鶯 The Blog of Unmi博客,原文名:《也說(shuō) Java 的雙括符初始化, 其實(shí)就是令人費(fèi)解的不規(guī)范代碼》
【編輯推薦】