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