栏目分类:
子分类:
返回
名师互学网用户登录
快速导航关闭
当前搜索
当前分类
子分类
实用工具
热门搜索
名师互学网 > IT > 软件开发 > 后端开发 > Java

java8编程思想-赋值(对象传递和返回)(值传递和引用传递)

Java 更新时间: 发布时间: IT归档 最新发布 模块sitemap 名妆网 法律咨询 聚返吧 英语巴士网 伯小乐 网商动力

java8编程思想-赋值(对象传递和返回)(值传递和引用传递)

文章目录
  • 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.java

public 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);
  } 
}


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

本地拷贝 控制克隆 不可变类 本章小结
转载请注明:文章转载自 www.mshxw.com
本文地址:https://www.mshxw.com/it/325089.html
我们一直用心在做
关于我们 文章归档 网站地图 联系我们

版权所有 (c)2021-2022 MSHXW.COM

ICP备案号:晋ICP备2021003244-6号