《On Java 8》-赋值(对象传递和返回)(值传递和引用传递)

网友投稿 256 2022-09-22

《On Java 8》-赋值(对象传递和返回)(值传递和引用传递)

文章目录

​​On Java 8​​

​​第四章 运算符-赋值​​​​附录: 对象传递和返回​​

​​传递引用​​​​本地拷贝​​​​控制克隆​​​​不可变类​​​​本章小结​​

On Java 8

《On Java 8》中的内容

赋值

对象传递和返回

理解:值传递和引用传递

第四章 运算符-赋值

赋值

运算符的赋值是由符号 = 完成的。它代表着获取 = 右边的值并赋给左边的变量。右边可以是任何常量、变量或者可产生一个返回值的表达式。但左边必须是一个明确的、已命名的变量。也就是说,必须要有一个物理的空间来存放右边的值。举个例子来说,可将一个常数赋给一个变量(A = 4),但不可将任何东西赋给一个常数(比如不能 4 =A)。

(值传递和引用传递:)基本类型的赋值都是直接的,而不像对象,赋予的只是其内存的引用。举个例子,a = b ,如果 b 是基本类型,那么赋值操作会将 b 的值复制一份给变量 a,此后若 a 的值发生改变是不会影响到 b 的。作为一名程序员,这应该成为我们的常识。

如果是为对象赋值,那么结果就不一样了。对一个对象进行操作时,我们实际上操作的是它的引用。所以我们将右边的对象赋予给左边时,赋予的只是该对象的引用。此时,两者指向的堆中的对象还是同一个。代码示例:

// operators/Assignment.java// Assignment with objects is a bit tricky*class Tank { int level;}public class Assignment { public static void main(String[] args) { Tank t1 = **new** Tank(); Tank t2 = **new** Tank(); t1.level = 9; t2.level = 47; System.out.println("1: t1.level: " + t1.level +", t2.level: " + t2.level); t1 = t2; System.out.println("2: t1.level: " + t1.level +", t2.level: " + t2.level); t1.level = 27; System.out.println("3: t1.level: " + t1.level +", t2.level: " + t2.level); } }

输出结果:

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.level = t2.level;

较之前的做法,这样做保留了两个单独的对象,而不是丢弃一个并将 t1 和 t2 绑定到同一个对象。但是这样的操作有点违背 Java 的设计原则。对象的赋值是个需要重视的环节,否则你可能收获意外的 “惊喜”。

// 方法调用中的别名现象//当我们把对象传递给方法时,会发生别名现象。// operators/PassObject.java// 正在传递的对象可能不是你之前使用的class Letter { char c;}public class PassObject { static void f(Letter y) { y.c = 'z'; } public static void main(String[] args) { Letter x = new Letter(); x.c = 'a'; System.out.println("1: x.c: " + x.c); f(x); System.out.println("2: x.c: " + x.c); } }

输出结果:

1: x.c: a

2: x.c: z

在许多编程语言中,方法 f() 似乎会在内部复制其参数 Letter y。但是一旦传递了一个引用,那么实际上 y.c =‘z’; 是在方法 f() 之外改变对象。别名现象以及其解决方案是个复杂的问题,在附录中有包含:对象传递和返回。意识到这一点,我们可以警惕类似的陷阱。

附录: 对象传递和返回

到现在为止,你已经对 “传递” 对象实际上是传递引用这一想法想法感到满意。

在许多编程语言中,你可以使用该语言的 “常规” 方式来传递对象,并且大多数情况下一切正常。但是通常会出现这种情况,你必须做一些不平常的事情,突然事情变得更加复杂。Java 也不例外,当您传递对象并对其进行操作时,准确了解正在发生的事情很重要。本附录提供了这种见解。

提出本附录问题的另一种方法是,如果你之前使用类似 C++ 的编程语言,则是 “Java 是否有指针?” Java 中的每个对象标识符(除原语外)都是这些指针之一,但它们的用法是不仅受编译器的约束,而且受运行时系统的约束。换一种说法,Java 有指针,但没有指针算法。这些就是我一直所说的 “引用”,您可以将它们视为 “安全指针”,与小学的安全剪刀不同-它们不敏锐,因此您不费吹灰之力就无法伤害自己,但是它们有时可能很乏味。

传递引用

当你将引用传递给方法时,它仍指向同一对象。一个简单的实验演示了这一点:

// references/PassReferences.javapublic class PassReferences { public static void f(PassReferences h) { System.out.println("h inside f(): " + h); } public static void main(String[] args) { PassReferences p = new PassReferences(); System.out.println("p inside main(): " + p); f(p); } }/* Output:p inside main(): PassReferences@15db9742h inside f(): PassReferences@15db9742*/

方法 toString() 在打印语句中自动调用,并且 PassReferences 直接从 Object继承而无需重新定义 toString()。因此,使用的是 Object 的 toString()版本,它打印出对象的类,然后打印出该对象所在的地址(不是引用,而是实际的对象存储)。

本地拷贝

控制克隆

不可变类

本章小结

版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:湖人接连续约两大球星,“浓眉”获5年1.9亿美元大单!
下一篇:Java 16 新特性
相关文章

 发表评论

暂时没有评论,来抢沙发吧~