Java中的双大括号初始化

什么是Java双大括号初始化?

通常情况下,初始化Java集合并向其中添加几个元素的步骤如下:

Set<Integer> set = new HashSet<>();
set.add(1);
set.add(2);
set.add(3);

或者我们可以在静态初始化块中向作为静态变量的集合添加元素:

private static final Set<Integer> set = new HashSet<>();
static {
    set.add(1);
    set.add(2);
    set.add(3);
}

从语法上来看,这样的初始化方法虽然格式清晰明了,但语法上略显冗余。事实上,Java允许一种精简的双大括号初始化方法:

Set<Integer> set = new HashSet<Integer>() {{
    add(1);
    add(2);
    add(3);
}};

或是在接收集合作为输入的函数中直接初始化:

someFunction(new HashSet<Integer>() {{
    add(1);
    add(2);
    add(3); 
}}
);

语法解读

事实上,如下双大括号初始化Set的代码

Set<Integer> set = new HashSet<Integer>() {{
    add(1);
    add(2);
    add(3);
}};

是以下这段代码的改写:

Set<Integer> set = new HashSet<Integer>() {
    {
        add(1);
        add(2);
        add(3);
    }
};

改写之后的代码就非常容易理解了。显然这是在HashSet的构造器中写了一个匿名内部类,这个匿名内部类含有一个实例初始化块,初始化块的内容是三个add()函数,向被初始化的this指向的HashSet中添加了三个元素。

效率问题和产生的.class文件结构

利用双大括号初始化集合从效率上来说可能不如标准的集合初始化步骤。原因在于使用双大括号初始化会导致内部类文件的产生,而这个过程就会影响代码的执行效率。

例如如下代码:

// Double brace initialization
class Test1 {
    public static void main(String[] args) {
        long st = System.currentTimeMillis();

    Set<Integer> set0 = new HashSet<Integer>() {{
        add(1);
        add(2);
    }};

    Set<Integer> set1 = new HashSet<Integer>() {{
        add(1);
        add(2);
    }};
    
    /* snip */

    Set<Integer> set999 = new HashSet<Integer>() {{
        add(1);
        add(2);
    }};

      System.out.println(System.currentTimeMillis() - st);
  }
}
// Normal initialization
class Test2 {
    public static void main(String[] s) {
        long st = System.currentTimeMillis();

        Set<Integer> set0 = new HashSet<Integer>();
        set0.add(1);
        set0.add(2);

        Set<Integer> set1 = new HashSet<Integer>();
        set1.add(1);
        set1.add(2);
        
        /* snip */

        Set<Integer> set999 = new HashSet<Integer>();
        set999.add(1);
        set999.add(2);  
        System.out.println(System.currentTimeMillis() - st);
  }
}

javac命令compile第一段代码会产生1000个.class文件,如下所示:

Test1$1.class
Test1$2.class
...
Test1$1000.class

同时从程序输出的运行时间来看, 双大括号初始化也显著慢于普通初始化的代码。差值大约在100ms左右。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,765评论 18 399
  • (一)Java部分 1、列举出JAVA中6个比较常用的包【天威诚信面试题】 【参考答案】 java.lang;ja...
    独云阅读 7,142评论 0 62
  • 公元二零一六年八月十八号,你被天使安排来到了我们的身边,当时你带着3.2kg的小身板,伴着哭声,一丝不挂! ...
    挥霍5211阅读 370评论 2 3
  • 摄影师才才阅读 307评论 1 0
  • 两个星期的理财训练营即将结束,我心中感慨良多。首先感谢长投网给了我们一个这样的平台,其次感谢班班和学姐们的悉悉...
    胖朱瘦瓜阅读 270评论 1 2