{{ ... }}
Java中的双大括号初始化语法 ( ) 是什么?
13 回答
双大括号初始化创建一个从指定类(外大括号)派生的匿名类,并在该类(内大括号)内提供一个初始化块。例如
new ArrayList<Integer>() {{
add(1);
add(2);
}};
请注意,使用这种双括号初始化的效果是您正在创建匿名内部类。this
创建的类有一个指向周围外部类的隐式指针。虽然通常不是问题,但在某些情况下可能会导致悲伤,例如在序列化或垃圾收集时,值得了解这一点。
每次有人使用双括号初始化时,都会杀死一只小猫。
除了语法相当不寻常且不是真正地道(当然,品味是有争议的)之外,您还不必要地在您的应用程序中创建了两个重大问题,我最近刚刚在此处更详细地介绍了这些问题。
1. 你创建了太多的匿名类
每次使用双括号初始化时,都会创建一个新类。例如这个例子:
Map source = new HashMap(){{
put("firstName", "John");
put("lastName", "Smith");
put("organizations", new HashMap(){{
put("0", new HashMap(){{
put("id", "1234");
}});
put("abc", new HashMap(){{
put("id", "5678");
}});
}});
}};
...将产生这些类:
Test$1$1$1.class
Test$1$1$2.class
Test$1$1.class
Test$1.class
Test.class
这对您的类加载器来说是相当多的开销 - 没有!当然,如果您执行一次,它不会花费太多初始化时间。但是,如果您在整个企业应用程序中执行此操作 20'000 次......所有的堆内存只是为了一点“语法糖”?
2. 您可能会造成内存泄漏!
如果您使用上面的代码并从方法返回该映射,则该方法的调用者可能会毫无疑问地持有无法被垃圾收集的非常重的资源。考虑以下示例:
public class ReallyHeavyObject {
// Just to illustrate...
private int[] tonsOfValues;
private Resource[] tonsOfResources;
// This method almost does nothing
public Map quickHarmlessMethod() {
Map source = new HashMap(){{
put("firstName", "John");
put("lastName", "Smith");
put("organizations", new HashMap(){{
put("0", new HashMap(){{
put("id", "1234");
}});
put("abc", new HashMap(){{
put("id", "5678");
}});
}});
}};
return source;
}
}
返回的Map
现在将包含对封闭实例的引用ReallyHeavyObject
。您可能不想冒险:
图片来自http://blog.jooq.org/2014/12/08/dont-be-clever-the-double-curly-braces-anti-pattern/
3.你可以假装Java有地图文字
为了回答您的实际问题,人们一直在使用这种语法来假装 Java 有类似映射文字的东西,类似于现有的数组文字:
String[] array = { "John", "Doe" };
Map map = new HashMap() {{ put("John", "Doe"); }};
有些人可能会觉得这在句法上很刺激。
- 第一个大括号创建了一个新的匿名内部类。
- 第二组大括号在 Class 中创建一个实例初始化器,如静态块。
例如:
public class TestHashMap {
public static void main(String[] args) {
HashMap<String,String> map = new HashMap<String,String>(){
{
put("1", "ONE");
}{
put("2", "TWO");
}{
put("3", "THREE");
}
};
Set<String> keySet = map.keySet();
for (String string : keySet) {
System.out.println(string+" ->"+map.get(string));
}
}
}
这个怎么运作
第一个大括号创建一个新的匿名内部类。这些内部类能够访问其父类的行为。所以,在我们的例子中,我们实际上是在创建一个 HashSet 类的子类,所以这个内部类能够使用 put() 方法。
第二组大括号只不过是实例初始化器。如果您还记得核心 Java 概念,那么您可以轻松地将实例初始化程序块与静态初始化程序关联起来,因为它与 struct 类似。唯一的区别是静态初始化器添加了静态关键字,并且只运行一次;无论您创建多少对象。
有关双括号初始化的有趣应用,请参见此处的 Dwemthy's Array in Java。
摘录
private static class IndustrialRaverMonkey
extends Creature.Base {{
life = 46;
strength = 35;
charisma = 91;
weapon = 2;
}}
private static class DwarvenAngel
extends Creature.Base {{
life = 540;
strength = 6;
charisma = 144;
weapon = 50;
}}
现在,准备好BattleOfGrottoOfSausageSmells
……大块培根!
我认为重要的是要强调Java 中没有“双括号初始化”之类的东西。Oracle 网站没有这个词。在这个例子中,有两个特性一起使用:匿名类和初始化块。似乎旧的初始化程序块已被开发人员遗忘,并在该主题中引起了一些混乱。来自Oracle 文档的引用:
实例变量的初始化块看起来就像静态初始化块,但没有 static 关键字:
{
// whatever code is needed for initialization goes here
}
1- 没有双括号这样的东西:
我想指出,没有双括号初始化这样的东西。只有普通的传统大括号初始化块。第二个大括号块与初始化无关。答案说这两个大括号初始化了一些东西,但事实并非如此。
2-这不仅仅是关于匿名类,而是关于所有类:
几乎所有答案都说它是创建匿名内部类时使用的东西。我认为阅读这些答案的人会得到这样的印象,即这只在创建匿名内部类时使用。但它用于所有类。阅读这些答案看起来像是一些专门针对匿名课程的全新特殊功能,我认为这是一种误导。
3-目的只是将括号彼此放在一起,而不是新概念:
更进一步,这个问题讨论了第二个开括号就在第一个开括号之后的情况。在普通类中使用时,通常在两个大括号之间有一些代码,但它是完全一样的。所以这是一个放置括号的问题。所以我认为我们不应该说这是一些新的令人兴奋的事情,因为这是我们都知道的事情,只是在括号之间写了一些代码。我们不应该创建称为“双括号初始化”的新概念。
4-创建嵌套匿名类与两个大括号无关:
我不同意您创建太多匿名类的论点。您创建它们不是因为初始化块,而只是因为您创建了它们。即使您没有使用两个大括号初始化它们也会被创建,因此即使没有初始化也会出现这些问题......初始化不是创建初始化对象的因素。
另外,我们不应该谈论使用这种不存在的东西“双括号初始化”甚至是正常的单括号初始化所产生的问题,因为所描述的问题只是因为创建了匿名类而存在,因此与原始问题无关。但是所有答案都给读者留下了这样的印象,即创建匿名类不是错,而是这种邪恶的(不存在的)东西称为“双括号初始化”。
为了避免双括号初始化的所有负面影响,例如:
- 破坏“等于”兼容性。
- 使用直接分配时不执行检查。
- 可能的内存泄漏。
做接下来的事情:
- 制作单独的“Builder”类,特别是用于双括号初始化。
- 用默认值声明字段。
- 将对象创建方法放在该类中。
例子:
public class MyClass {
public static class Builder {
public int first = -1 ;
public double second = Double.NaN;
public String third = null ;
public MyClass create() {
return new MyClass(first, second, third);
}
}
protected final int first ;
protected final double second;
protected final String third ;
protected MyClass(
int first ,
double second,
String third
) {
this.first = first ;
this.second= second;
this.third = third ;
}
public int first () { return first ; }
public double second() { return second; }
public String third () { return third ; }
}
用法:
MyClass my = new MyClass.Builder(){{ first = 1; third = "3"; }}.create();
好处:
- 简单地使用。
- 不要破坏“等于”兼容性。
- 您可以在创建方法中执行检查。
- 没有内存泄漏。
缺点:
- 没有任何。
因此,我们拥有了最简单的 java builder 模式。
查看 github 上的所有示例:java-sf-builder-simple-example
正如@Lukas Eder 所指出的,必须避免集合的双括号初始化。
它创建了一个匿名内部类,并且由于所有内部类都保留对父实例的引用,因此如果这些收集对象被更多对象引用而不仅仅是声明对象,它可以 - 并且 99% 可能会 - 阻止垃圾收集。
Java 9 引入了便利方法List.of
、Set.of
和Map.of
,应该使用它们。它们比双大括号初始值设定项更快、更高效。
它是 - 除其他用途外 - 用于初始化集合的快捷方式。学到更多 ...
你的意思是这样的?
List<String> blah = new ArrayList<String>(){{add("asdfa");add("bbb");}};
这是创建时的数组列表初始化(hack)
您可以将一些 Java 语句作为循环来初始化集合:
List<Character> characters = new ArrayList<Character>() {
{
for (char c = 'A'; c <= 'E'; c++) add(c);
}
};
Random rnd = new Random();
List<Integer> integers = new ArrayList<Integer>() {
{
while (size() < 10) add(rnd.nextInt(1_000_000));
}
};
但是这种情况会影响性能,请查看此讨论
第一个大括号创建一个新的匿名类,第二个大括号创建一个实例初始化器,如静态块。
就像其他人指出的那样,使用它是不安全的。
但是,您始终可以使用此替代方法来初始化集合。
- 爪哇 8
List<String> list = new ArrayList<>(Arrays.asList("A", "B", "C"));
- 爪哇 9
List<String> list = List.of("A", "B", "C");
这似乎与在 flash 和 vbscript 中如此流行的 with 关键字相同。这是一种改变现状的方法,仅此this
而已。