1. 简单类型是按值传递的

  Java 方法的参数是简单类型的时候,是按值传递的 (pass by value)。这一点我们可以通过一个简单的例子来说明:

  1. public class Test {
  2. public static void test(boolean test) {
  3. test = ! test;
  4. System.out.println("In test(boolean) : test = " + test);
  5. }
  6. public static void main(String[] args) {
  7. boolean test = true;
  8. System.out.println("Before test(boolean) : test = " + test);
  9. test(test);
  10. System.out.println("After test(boolean) : test = " + test);
  11. }
  12. }

  运行结果:

Before test(boolean) : test = true
In test(boolean) : test = false
After test(boolean) : test = true

  不难看出,虽然在 test(boolean) 方法中改变了传进来的参数的值,但对这个参数源变量本身并没有影响,即对 main(String[]) 方法里的 test 变量没有影响。那说明,参数类型是简单类型的时候,是按值传递的。以参数形式传递简单类型的变量时,实际上是将参数的值作了一个拷贝传进方法函数的,那么 在方法函数里再怎么改变其值,其结果都是只改变了拷贝的值,而不是源值。

  2. 什么是引用

  Java 是传值还是传引用,问题主要出在对象的传递上,因为 Java 中简单类型没有引用。既然争论中提到了引用这个东西,为了搞清楚这个问题,我们必须要知道引用是什么。

  简单的说,引用其实就像是一个对象的名字或者别名 (alias),一个对象在内存中会请求一块空间来保存数据,根据对象的大小,它可能需要占用的空间大小也不等。访问对象的时候,我们不会直接是访问对象 在内存中的数据,而是通过引用去访问。引用也是一种数据类型,我们可以把它想象为类似 C 语言中指针的东西,它指示了对象在内存中的地址——只不过我们不能够观察到这个地址究竟是什么。

  如果我们定义了不止一个引用指向同一个对象,那么这些引用是不相同的,因为引用也是一种数据类型,需要一定的内存空间来保存。但是它们的值是相同的,都指示同一个对象在内存的中位置。比如:

  1. String a = "Hello";
  2. String b = a;

  这里,a 和 b 是不同的两个引用,我们使用了两个定义语句来定义它们。但它们的值是一样的,都指向同一个对象 "Hello"。也许你还觉得不够直观,因为 String 对象的值本身是不可更改的 (像 b = "World"; b = a; 这种情况不是改变了 "World" 这一对象的值,而是改变了它的引用 b 的值使之指向了另一个 String 对象 a)。那么我们用 StringBuffer 来举一个例子:

  1. public class Test {
  2. public static void main(String[] args) {
  3. StringBuffer a = new StringBuffer("Hello");
  4. StringBuffer b = a;
  5. b.append(", World");
  6. System.out.println("a is " + a);
  7. }
  8. }

  运行结果:

a is Hello, World

  这个例子中 a 和 b 都是引用,当改变了 b 指示的对象的值的时候,从输出结果来看,a 所指示的对象的值也改变了。所以,a 和 b 都指向同一个对象即包含 "Hello" 的一个 StringBuffer 对象。

  这里我描述了两个要点:

  1. 引用是一种数据类型,保存了对象在内存中的地址,这种类型即不是我们平时所说的简单数据类型也不是类实例(对象);
  2. 不同的引用可能指向同一个对象,换句话说,一个对象可以有多个引用,即该类类型的变量。

  3. 对象是如何传递的呢

  关于对象的传递,有两种说法,即“它是按值传递的”和“它是按引用传递的”。这两种说法各有各的道理,但是它们都没有从本质上去分析,即致于产生了争论。

  既然现在我们已经知道了引用是什么东西,那么现在不妨来分析一下对象作是参数是如何传递的。还是先以一个程序为例:

  1. public class Test {
  2. public static void test(StringBuffer str) {
  3. str.append(", World!");
  4. }
  5. public static void main(String[] args) {
  6. StringBuffer string = new StringBuffer("Hello");
  7. test(string);
  8. System.out.println(string);
  9. }
  10. }

  运行结果:

  Hello, World!

  test(string) 调用了 test(StringBuffer) 方法,并将 string 作为参数传递了进去。这里 string 是一个引用,这一点是勿庸置疑的。前面提到,引用是一种数据类型,而且不是对象,所以它不可能按引用传递,所以它是按值传递的,它么它的值究竟是什么呢? 是对象的地址。

  由此可见,对象作为参数的时候是按值传递的,对吗?错!为什么错,让我们看另一个例子:

  1. public class Test {
  2. public static void test(String str) {
  3. str = "World";
  4. }
  5. public static void main(String[] args) {
  6. String string = "Hello";
  7. test(string);
  8. System.out.println(string);
  9. }
  10. }

  运行结果:

Hello

  为什么会这样呢?因为参数 str 是一个引用,而且它与 string 是不同的引用,虽然它们都是同一个对象的引用。str = "World" 则改变了 str 的值,使之指向了另一个对象,然而 str 指向的对象改变了,但它并没有对 "Hello" 造成任何影响,而且由于 string 和 str 是不同的引用,str 的改变也没有对 string 造成任何影响,结果就如例中所示。

  其结果是推翻了参数按值传递的说法。那么,对象作为参数的时候是按引用传递的了?也错!因为上一个例子的确能够说明它是按值传递的。

  结果,就像光到底是波还是粒子的问题一样,Java 方法的参数是按什么传递的问题,其答案就只能是:即是按值传递也是按引用传递,只是参照物不同,结果也就不同。

  4. 正确看待传值还是传引用的问题

  要正确的看待这个问题必须要搞清楚为什么会有这样一个问题。

  实际上,问题来源于 C,而不是 Java。

  C 语言中有一种数据类型叫做指针,于是将一个数据作为参数传递给某个函数的时候,就有两种方式:传值,或是传指针,它们的区别,可以用一个简单的例子说明:

  1. void SwapValue(int a, int b) {
  2. int t = a;
  3. a = b;
  4. b = t;
  5. }
  6. void SwapPointer(int * a, int * b) {
  7. int t = * a;
  8. * a = * b;
  9. * b = t;
  10. }
  11. void main() {
  12. int a = 0, b = 1;
  13. printf("1 : a = %d, b = %d\n", a, b);
  14. SwapValue(a, b);
  15. printf("2 : a = %d, b = %d\n", a, b);
  16. SwapPointer(&a, &b);
  17. printf("3 : a = %d, b = %d\n", a, b);
  18. }

  运行结果:

1 : a = 0, b = 1
2 : a = 0, b = 1
3 : a = 1, b = 0

  大家可以明显的看到,按指针传递参数可以方便的修改通过参数传递进来的值,而按值传递就不行。

  当 Java 成长起来的时候,许多的 C 程序员开始转向学习 Java,他们发现,使用类似 SwapValue 的方法仍然不能改变通过参数传递进来的简单数据类型的值,但是如果是一个对象,则可能将其成员随意更改。于是他们觉得这很像是 C 语言中传值/传指针的问题。但是 Java 中没有指针,那么这个问题就演变成了传值/传引用的问题。可惜将这个问题放在 Java 中进行讨论并不恰当。

  讨论这样一个问题的最终目的只是为了搞清楚何种情况才能在方法函数中方便的更改参数的值并使之长期有效。

  Java 中,改变参数的值有两种情况,第一种,使用赋值号“=”直接进行赋值使其改变,如例 1 和例 4;第二种,对于某些对象的引用,通过一定途径对其成员数据进行改变,如例 3。对于第一种情况,其改变不会影响到方法该方法以外的数据,或者直接说源数据。而第二种方法,则相反,会影响到源数据——因为引用指示的对象没有变,对 其成员数据进行改变则实质上是改变的该对象。

  5. 如何实现类似 swap 的方法

  传值还是传引用的问题,到此已经算是解决了,但是我们仍然不能解决这样一个问题:如果我有两个 int 型的变量 a 和 b,我想写一个方法来交换它们的值,应该怎么办?

  结论很让人失望——没有办法!因此,我们只能具体情况具体讨论,以经常使用交换方法的排序为例:

  1. public class Test {
  2. public static void swap(int[] data, int a, int b) {
  3. int t = data[a];
  4. data[a] = data[b];
  5. data[b] = t;
  6. }
  7. public static void main(String[] args) {
  8. int[] data = new int[10];
  9. for (int i = 0; i < 10; i++) {
  10. data[i] = (int) (Math.random() * 100);
  11. System.out.print(" " + data[i]);
  12. }
  13. System.out.println();
  14. for (int i = 0; i < 9; i++) {
  15. for (int j = i; j < 10; j++) {
  16. if (data[i] > data[j]) {
  17. swap(data, i, j);
  18. }
  19. }
  20. }
  21. for (int i = 0; i < 10; i++) {
  22. System.out.print(" " + data[i]);
  23. }
  24. System.out.println();
  25. }
  26. }

  运行结果(情况之一):

78 69 94 38 95 31 50 97 84 1
1 31 38 50 69 78 84 94 95 97

  swap(int[] data, int a, int b) 方法在内部实际上是改变了 data 所指示的对象的成员数据,即上述讨论的第二种改变参数值的方法。希望大家能够举一反三,使用类似的方法来解决相关问题。

转载于:https://blog.51cto.com/justlpf/1108463

Java是传值还是传引用相关推荐

  1. 传值类型_Java内存管理:Stackoverflow问答-Java是传值还是传引用(十一)

    勿在流沙筑高台,出来混迟早要还的. 做一个积极的人 编码.改bug.提升自己 我有一个乐园,面向编程,春暖花开! 本文导图: 一.由一个提问引发的思考 在Stack Overflow 看到这样一个问题 ...

  2. Java内存管理-Stackoverflow问答-Java是传值还是传引用?(十一)

    做一个积极的人 编码.改bug.提升自己 我有一个乐园,面向编程,春暖花开! 推荐阅读 第一季 0.Java的线程安全.单例模式.JVM内存结构等知识梳理 1.Java内存管理-程序运行过程(一) 2 ...

  3. 几句话弄清楚Java参数传值还是传引用

    最近刷题用递归解决关于树的问题的时候,在递归函数调用的时候什么时候传入值,什么时候传入引用有疑问,因为这关系到什么时候我们需要恢复现场,什么时候不需要.在网上搜索了一下,感觉这篇总结的非常简单明了,简 ...

  4. JAVA中传值与“传引用“辨析

    一.概念简单辨析 Java常用的数据类型有 (1).八大基本数据类型: 1.boolean:数据值只有true或false,适用于逻辑计算. 2.char:char型(字符型)数据在内存中占用2个字节 ...

  5. Java传值与传引用

    基本参数类型:传值 对象类型:传引用 值传递(passl-by-value)过程中,被调函数的形式参数作为被调函数的局部变量处理,即在堆栈中开辟了内存空间以存放由主调函数放进来的实参的值,从而成为了实 ...

  6. Java中的传值与传引用

    曾经对java中传值还是传引用的问题十分困惑,而且也被问到过许多次,无论是面试,还是平时我们平时的讨论中. 在stackoverflow上面浏览时,无意中发现了跟这个问题有关的论题,于是深入地查看了, ...

  7. JAVA中是传值还是传引用?回答KEZHANG问题

    首先,推荐对Java有一定理解的同仁一本书<Practical Java>.在<Practical Java>中也有一个章节介绍Java中关于传值和传引用的问题,堪称经典. & ...

  8. java的函数传值_java 函数形参传值和传引用的区别[转]

    java方法中传值和传引用的问题是个基本问题,但是也有很多人一时弄不清. (一)基本数据类型:传值,方法不会改变实参的值. publicclassTestFun { publicstaticvoidt ...

  9. 语言深入:java中究竟是传值还是传引用

    http://hi.baidu.com/hugoxian/item/5212a65bb1546aded48bace1 首先,推荐对Java有一定理解的同仁一本书<Practical Java&g ...

  10. [转载] java中的经典问题:传值与传引用

    参考链接: 有关Java中数组分配的有趣事实 参数传递的秘密 知道方法参数如何传递吗? 记得刚开始学编程那会儿,老师教导,所谓参数,有形式参数和实际参数之分,参数列表中写的那些东西都叫形式参数,在实际 ...

最新文章

  1. Oracle 10GR2+ASM在OEL上的安装(VMware Server 1.0)
  2. 中国「人造太阳」刚刚再破纪录:运行时间突破千秒,负责人:为稳态的聚变工程堆奠定基础...
  3. 驱动WDK7600在VC6.0下的编译开发配置环境
  4. canvas高度问题_Android的drawText的坐标问题
  5. 查看网关物理地址命令
  6. 自定义线程池-线程池工作流程介绍
  7. setnx是原子操作吗_谈谈Volatile关键字?为什么不能保证原子性?用什么可以替代?为什么?...
  8. Java GC系列(2):Java垃圾回收是如何工作的
  9. python小数补0,python用零填充小数
  10. 用栈实现队列(Leetcode第232题)+用队列实现栈(Leetcode第225题)
  11. STS热部署,springboot项目中修改代码不用重新启动服务
  12. 数据分析岗位需求分析报告
  13. 一文读懂人脸识别技术
  14. hadoop测试环境完全分布式安装配置
  15. 《开店有讲究》读书笔记
  16. Unity 基于图像处理的图像显示特效制作过场特效
  17. Ubuntu关闭开机自启动服务
  18. 英超前瞻乐.fun|体育 中秋利物浦主场对战狼队 历史交战能否延续全胜
  19. 无法卸载vue2.x提示up to date
  20. Loadrunner11.00破解方法

热门文章

  1. DotText源码阅读(3)-框架配置体系和反序列化
  2. Spatial Pyramid Pooling in Deep Convolutional Networks for Visual Recognition(SPP-net)
  3. 描述cookie,sessionstroage,localstrage的区别
  4. 《ASP网络编程从入门到精通》
  5. 118 Python程序中的线程操作-守护线程
  6. MySQL数据库企业级应用实践(主从复制)
  7. 吴裕雄--天生自然C语言开发:作用域规则
  8. 基于百度通用翻译API的一个翻译小工具
  9. python系统学习:第二周之字典应用
  10. python之绘制图形库turtle