Java中的双大括号初始化

Java中双大括号初始化是一种简洁的集合初始化方式,它等价于创建匿名内部类并包含实例初始化块来添加元素。然而,这种方式在效率上低于标准初始化,因为会生成额外的内部类文件,影响执行速度。对于大量元素初始化,这种差异可能导致明显的性能下降。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

什么是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左右。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

耀凯考前突击大师

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值