文章目录
On Java 8
《On Java 8》中的内容
赋值
对象传递和返回
理解:值传递和引用传递
第四章 运算符-赋值
赋值
运算符的赋值是由符号 = 完成的。它代表着获取 = 右边的值并赋给左边的变量。右边可以是任何常量、变量或者可产生一个返回值的表达式。但左边必须是一个明确的、已命名的变量。也就是说,必须要有一个物理的空间来存放右边的值。举个例子来说,可将一个常数赋给一个变量(A = 4),但不可将任何东西赋给一个常数(比如不能 4 =A)。
(值传递和引用传递:)基本类型的赋值都是直接的,而不像对象,赋予的只是其内存的引用。举个例子,a = b ,如果 b 是基本类型,那么赋值操作会将 b 的值复制一份给变量 a,此后若 a 的值发生改变是不会影响到 b 的。作为一名程序员,这应该成为我们的常识。
如果是为对象赋值,那么结果就不一样了。对一个对象进行操作时,我们实际上操作的是它的引用。所以我们将右边的对象赋予给左边时,赋予的只是该对象的引用。此时,两者指向的堆中的对象还是同一个。代码示例:
输出结果:
1: t1.level: 9, t2.level: 47
2: t1.level: 47, t2.level: 47
3: t1.level: 27, t2.level: 27
这是一个简单的 Tank 类,在 main() 方法创建了两个实例对象。两个对象的 level属性分别被赋予不同的值。然后,t2 的值被赋予给 t1。在许多编程语言里,预期的结果是 t1 和 t2 的值会一直相对独立。但是,在 Java 中,由于赋予的只是对象的引用,改变 t1 也就改变了 t2。这是因为 t1 和 t2 此时指向的是堆中同一个对象(引用传递)。(t1 原始对象的引用在 t2 赋值给其时丢失,它引用的对象会在垃圾回收时被清理)。
这种现象通常称为别名(aliasing),这是 Java 处理对象的一种基本方式。但是假若你不想出现这里的别名引起混淆的话,你可以这么做。代码示例:
较之前的做法,这样做保留了两个单独的对象,而不是丢弃一个并将 t1 和 t2 绑定到同一个对象。但是这样的操作有点违背 Java 的设计原则。对象的赋值是个需要重视的环节,否则你可能收获意外的 “惊喜”。
输出结果:
1: x.c: a
2: x.c: z
在许多编程语言中,方法 f() 似乎会在内部复制其参数 Letter y。但是一旦传递了一个引用,那么实际上 y.c =‘z’; 是在方法 f() 之外改变对象。别名现象以及其解决方案是个复杂的问题,在附录中有包含:对象传递和返回。意识到这一点,我们可以警惕类似的陷阱。
附录: 对象传递和返回
到现在为止,你已经对 “传递” 对象实际上是传递引用这一想法想法感到满意。
在许多编程语言中,你可以使用该语言的 “常规” 方式来传递对象,并且大多数情况下一切正常。但是通常会出现这种情况,你必须做一些不平常的事情,突然事情变得更加复杂。Java 也不例外,当您传递对象并对其进行操作时,准确了解正在发生的事情很重要。本附录提供了这种见解。
提出本附录问题的另一种方法是,如果你之前使用类似 C++ 的编程语言,则是 “Java 是否有指针?” Java 中的每个对象标识符(除原语外)都是这些指针之一,但它们的用法是不仅受编译器的约束,而且受运行时系统的约束。换一种说法,Java 有指针,但没有指针算法。这些就是我一直所说的 “引用”,您可以将它们视为 “安全指针”,与小学的安全剪刀不同-它们不敏锐,因此您不费吹灰之力就无法伤害自己,但是它们有时可能很乏味。
传递引用
当你将引用传递给方法时,它仍指向同一对象。一个简单的实验演示了这一点:
方法 toString() 在打印语句中自动调用,并且 PassReferences 直接从 Object继承而无需重新定义 toString()。因此,使用的是 Object 的 toString()版本,它打印出对象的类,然后打印出该对象所在的地址(不是引用,而是实际的对象存储)。
本地拷贝
控制克隆
不可变类
本章小结