关于Java是值传递还是引用传递

先说结论,Java是值传递

很多小伙伴认为:值传递和引用传递区分的条件就是,如果方法的实参传递的是一个值,则是值传递,如果传递的是一个引用,则是引用传递,或者说,如果传递的是一个基本数据类型,则是值传递,如果传递的是一个对象,则是引用传递。
但是这种观点是错误的,首先让我们来看一下值传递和引用传递的定义:

值传递是指在调用函数时将实际参数复制一份传递到函数中,这样在函数中如果对这个参数进行修改,将不会影响到实际参数
引用传递是指在调用函数时将实际参数的地址直接传递到函数中,那么在函数中对该参数所进行的修改,将影响到实际参数

有了定义,我们可以做实验了:

public static void main(String[] args) {
        Test test = new Test();
        int value = 100;
        test.alterAndPrint(value);
        System.out.println("main函数中value的值为: " + value);
    }

    public void alterAndPrint(int i) {
        i = 500;
        System.out.println("alterAndPrint函数中value的值为: " + i);
    }

这段代码的输出结果为:

alterAndPrint函数中value的值为: 500
main函数中value的值为: 100

有了这个结果,很多小伙伴就觉得可以下结论了:Java是值传递,但是可能会有人说不对,然后摆出另一个实验结果:

public static void main(String[] args) {
        Test test = new Test();
        Person person = new Person();
        person.setName("zhangsan");
        person.setAge(18);
        test.alterAndPrint(person);
        System.out.println("main函数中的Person为: " + person);
    }

    public void alterAndPrint(Person p) {
        p.setName("lisi");
        p.setAge(24);
        System.out.println("alterAndPrint函数中的Person为: " + p);
    }

结果为:

alterAndPrint函数中的Person为: Person{name='lisi', age=24}
main函数中的Person为: Person{name='lisi', age=24}

然后发现,实参的内容发生了改变,于是就得出结论,Java在传递基本数据类型的时候是值传递,在传递对象的时候是引用传递,但是还有一种特殊情况:

public static void main(String[] args) {
        Test test = new Test();
        String s = "Java";
        test.alterAndPrint(s);
        System.out.println("main函数中的String为: " + str);
    }

    public void alterAndPrint(String str) {
        str = "Golang";
        System.out.println("alterAndPrint函数中的String为: " + str);
    }

这段代码结果为:

alterAndPrint函数中的String为: Golang
main函数中的String为: Java

好,走到这一步,已经有很多小伙伴开始疑惑了,不是传递对象的时候是引用传递吗,为什么又变成值传递了。
其实,概念上是没有问题的,但是我们的实验方法出了些许问题,我们再来回看定义:

值传递是指在调用函数时将实际参数复制一份传递到函数中,这样在函数中如果对这个参数进行修改,将不会影响到实际参数
引用传递是指在调用函数时将实际参数的地址直接传递到函数中,那么在函数中对该参数所进行的修改,将影响到实际参数

重点区分一下这两个的区别就是:

值传递需要复制一份参数,且函数中对参数的改变不会影响到原参数
引用传递不需要复制,且函数中对该参数的改变会影响到原参数

在上面的Person那个例子中,我们改变的是参数的内容,而不是参数本身,所以说本身方法就错了,正常的实验方法,也即是真正地去改变参数应该是这样:

public static void main(String[] args) {
       Test test = new Test();
       Person person = new Person();
       person.setName("zhangsan");
       person.setAge(18);
       test.alterAndPrint(person);
       System.out.println("main函数中的Person为: " + person);
   }

   public void alterAndPrint(Person p) {
       p = new Person();
       p.setName("lisi");
       p.setAge(24);
       System.out.println("alterAndPrint函数中的Person为: " + p);
   }

输出结果为:

alterAndPrint函数中的Person为: Person{name='lisi', age=24}
main函数中的Person为: Person{name='zhangsan', age=18}

配合图来理解:



在main函数中我们new了一个person对象,并且给它的name和age属性赋值,person就拥有了这个内存的地址,也即是0x777999,当调用alterAndPrint方法时,将实参person传给形参p,p也指向了这一块地址,执行alterAndPrint方法的内容时,对参数进行修改,也即p = new Person(),这行代码会在堆区开辟一段新的内存,后面对p的更改都不会影响到0x777999这块内存的数据,
上面这种方式是什么传递,反正肯定不是引用传递,因为如果是引用传递的话,在p = new Person()这一串代码执行完后,实际的参数的引用也应该指向0x777555这一块内存,但是我们发现并没有这样

通过上面的定义我们可以知道,这是把实际参数的引用地址复制了一份,传递给了形式参数,所以,上面的参数其实是值传递,把实参对象引用的地址当做值传递给了形式参数。

所以,要区分值传递和引用传递并不是靠辨别传递的内容,而是有没有将实参复制一份副本传递给形参,如果传递的是地址,那么就要看这个地址是否发生改变,而不是看地址对应的对象的内容是否发生改变
那为什么String的那个例子会出现另一种情况呢?
因为str = "Golang"这一句代码改变了引用的地址,new了一个新的String,也即等价于str=new String("Golang'),而并没有改变原实参的引用地址


所以说,Java方法之间其实还是值传递的,对于基本数据类型,值的内容是数值,对于对象参数,值的内容是对象的引用。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,951评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,606评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,601评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,478评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,565评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,587评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,590评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,337评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,785评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,096评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,273评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,935评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,578评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,199评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,440评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,163评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,133评论 2 352

推荐阅读更多精彩内容