64

我听说在某些语言中类型被称为“盒装”。

在 Java 中,我听说过“自动装箱”。这是什么?它是否具有类型的包装类?如果我使用盒装或未盒装类型,我的代码将如何变化?

4

6 回答 6

79

某些数据类型被认为是“原始的”,这意味着它们不被视为对象并且不具有对象的属性。

在大多数平台上,整数和字符都是原始类型但可以装箱的示例。

装箱意味着将它们包装在一个对象中,以便它们具有对象的行为。

确切的含义和行为取决于您使用的语言。某些语言(例如 Smalltalk ......至少在我做这件事的时候已经过去了......)不允许任何原始类型并将所有内容都视为对象,但是这会带来性能损失,因为最后当天,处理器需要使用原始数字和原始内存来完成有用的工作。如果要添加两个已装箱的整数,例如,在幕后将它们“拆箱”为原始类型,则将数字相加,然后将它们装箱回一个新的整数。

于 2009-09-13T17:26:00.703 回答
29

Java 的更多具体信息:

在大多数情况下,自动装箱允许 java 自动将之类boolean的东西转换int为它们的 Object 版本。它还允许相反的情况发生。BooleanInteger

例如:

int a = 3; // no boxing is happening
Integer b = 3;  // newer versions of java automatically convert the int 3 to Integer 3
int c = b;  // these same versions also automatically convert Integer 3 to int 3

没有自动装箱功能的较早版本的 java 将需要此代码来执行相同的操作:

int a = 3;  // works the same
Integer b = new Integer(3);  //must set up a Integer object manually
int c = b.intValue(); //must change Integer object to a primitive

但是,在某些情况下您仍然必须手动执行操作。例如,假设您有一个具有两种方法的类,如下所示:

assertEquals(int a, int b);
assertEquals(Object a, Object b)

现在,如果您尝试这样做:

Integer a = 3;
int b = 3;
assertEquals(a, b);  // this will not compile

这不起作用的原因是因为它无法确定是否应该取消装箱a到一个int或装箱b到一个Integer. 因此,应该调用哪个方法签名是不明确的。要解决此问题,您可以执行以下操作之一:

assertEquals((int) a, b);
assertEquals(a, (Integer) b);
于 2009-09-13T17:37:15.280 回答
12

是的,装箱意味着采用值类型并将其包装在引用类型中。由于 java 引入了自动装箱,您可以执行以下操作:

void foo(Object bar) {}
//...
    foo(1);

java会自动把int 1变成Integer。在以前的版本中,您必须这样做:

foo(new Integer(1));

使用泛型时,自动装箱在 java 中最有用,因为您不能将基元与泛型一起使用,因此要将整数存储在列表中,您必须制作 aList<Integer>并将整数放入装箱的列表中。

于 2009-09-13T17:30:29.853 回答
4

装箱类型意味着值在堆上的块中分配并通过指针引用。这有利于运行时实现的统一性(它使具有通用函数等变得更容易),但代价是额外的间接。

于 2009-09-13T17:28:51.803 回答
1

通常,当您使用集合时,您正在处理对象数组。在像 Java 这样的语言中,原语和对象之间是有区别的。当一个基元被“装箱”时,它本质上只是一个基元的包装器,因此它与期望一个对象的框架的其余部分配合得很好。

自动装箱只是将基元放入对象或透明地从对象中拉出基元的行为,因此您不必担心自己做这件事的额外步骤。

于 2009-09-13T17:29:59.620 回答
1

装箱意味着他们采用常规值类型并围绕它创建一个对象。有点像把它放在一个盒子里。这通常是要避免的,因为构造对象的开销。

于 2009-09-13T18:52:51.527 回答