美文网首页跨平台开发公交站
引用还是传值——被打脸后才发现多年的理解是错的

引用还是传值——被打脸后才发现多年的理解是错的

作者: 恋猫月亮 | 来源:发表于2022-03-06 20:38 被阅读0次

这是一个很基础的问题,如果你已经理解透彻了,其实可以不需要往下看(如果理解没错的话),因为相信你已经知道了答案,本篇主要是解释给和我一样一直以来有这样误解的人,事实上这是一个简单的问题,之所以会陷入这个误区,主要还是因为习惯了高级语言后,特别是屏蔽了指针感知后,多年来“口口相传”导致的误解

起因是:

关于 dart 在函数里究竟是引用还是传值,到 java 在方法里是引用还是传值?

其实结论也很简单,不管是 dart 和 java ,在正统意义上理解,都是值传递。 等下,是不是这时候有些人就开始质疑了:“就这”?

不急,有兴趣的可以往下看,先说正统意义上的理解,如下示例1代码所示,这就是正统意义上传递还是引用的最直观示例:

///示例1

public static class People {
    public String name;
    People(String name) {
        this.name = name;
    }
}

public static void main(String[] args) {
    People a = new People("111");
    changePeople(a);
    System.out.println("print " + a.name);
}

public static void changePeople(People p) {
    p = new People("222");
}

如上代码,如果是真正意义上的引用传递,那么打印出来的应该是 "print 222" ,但是事实上运行后打印出来的是 "print 111"

如果你觉得这样不对,那就是和我以前一样理解错误的话,那肯定会举这样的例子,如下示例2所示:

///示例2

public static class People {
    public String name;
    People(String name) {
        this.name = name;
    }
}

public static void main(String[] args) {
    People a = new People("111");
    System.out.println("print a hash " + a.hashCode());
    changePeople(a);
    System.out.println("print " + a.name);
}

public static void changePeople(People p) {
    System.out.println("print p hash " + p.hashCode());
    p.name  = "222";
}

运行之后的结果是:

I/System.out: print a hash 240863055
I/System.out: print p hash 240863055
I/System.out: print 222

分明 ap 不就是一个地址吗?打印之后 aname 不也变成了 222 了吗? 从这个角度理解看起来好像真的就是引用传递!但是可惜这并不是,这是一种误解。

其实这里的问题主要出在讨论的角度出现了问题:

  • 示例 1 正统上大家说的引用传递是对于变量对象的角度;
  • 示例 2 讨论的引用还是传递是以值的角度;

知乎的这个例子举的就特别有意思,以它的例子为模板:

  • 你有一把钥匙,当你的朋友想要去你家的时候,如果你直接把你的钥匙给他了,这就是引用传递。这种情况下,如果他对这把钥匙做了什么事情,比如他在钥匙上刻下了自己名字,那么这把钥匙还给你的时候,你自己的钥匙上也会多出他刻的名字。、

  • 你有一把钥匙,当你的朋友想要去你家的时候,如果你复制了你的钥匙给他,这就是值传递。这种情况下,如果他对他钥匙做了什么事情,都和你的钥匙无关。

  • 最后按照示例2的角度代入这个故事,你的朋友拿着你给他的钥匙,进到你的家里,把你家的电视砸了,你再用你的钥匙开门进去,看到的也是被砸了的家,这就是示例2中的 p.name 赋值的类比。

image

所以示例2其实就是如上图的一个状态,其实 a 传递进去 changePeople 之后,在 changePeople 里的 p 已经是另外一个地址,而不是传递的 a 的地址 ,所以并不是传统意义上的引用传递,而我们打印出来的一致的 hashCode ,其实就是值 People 的地址和引用。

这个结论在 java 和 dart 里都是一致的,而我也是被 js 的同学所打脸,所以在函数上 java、dart、js 这些高级语言的设计都是如此。

我思考了下,从值的角度导致误解出现的原因,其实应该归结于高级语言里屏蔽了指针等的底层概念:

首先在 java、 dart 函数里讨论对象的传递引用意义不大,因为不能被操作的引用对象没意义,如果引用对象不被赋值给变量,它就会被GC,所以最终都关注到“值”本身。

所以作为操作不了对象引用的语言,讨论引用传递确实没有意义,从而导致大家把值和对象关系搞混了~当然,如果你也有什么想法或者理解,欢迎评论交流。

相关文章

  • 引用还是传值——被打脸后才发现多年的理解是错的

    这是一个很基础的问题,如果你已经理解透彻了,其实可以不需要往下看(如果理解没错的话),因为相信你已经知道了答案,本...

  • java中参数传递,是值传递

    错误理解 错误理解一:值传递和引用传递,区分的条件是传递的内容,如果是个值,就是值传递。 如果是个引用,就是引用传...

  • Python 函数中,参数是传值,还是传引用?

    Python 函数中,参数是传值,还是传引用? 代码示例 在 C/C++ 中,传值和传引用是函数参数传递的两种方式...

  • python语言那些...

    值传递还是引用传递引用传递,只是在遇到传递不可变对象时像值传递。对于不可变对象的函数传参,依然是传的引用(地址)。...

  • 深拷贝

    传值还是传引用 传值还是传引用?很基础的问题,对于刚刚接触编程的同学来说,却是一个大坑。在js中,所有的对象都是通...

  • Python中函数调用是传值还是传引用?

    Python在调用函数的时候,究竟是传值还是传引用呢?这个问题的答案无外乎这几种说法:传值,传引用,对于可变对象是...

  • java调用方法传递机制

    java是传值,不是传引用,有人说有的人说:基础类型传值,对象类型传引用。其实对象类也是传值,只不过对象的值恰好是...

  • 一起来了解一下 c++的三种函数形参传递

    c++ 的形参 大致可以分三种 ,传值 ,传指针 传 引用, 传值 ,不改变原来函数外变量的值,原来是几 还是...

  • java传参数是传值还是引用

    1、基参数是本数据类型 传递是值(或者说值得副本)。 2、参数是对象的话传递的是对象引用的副本。(1.修改引用副本...

  • 参数按值传递

    参数如果是基本类型是按值传递,如果是引用类型按共享传递。 1、按值传值 2、共享传值 共享传值: 按引用传递是传...

网友评论

    本文标题:引用还是传值——被打脸后才发现多年的理解是错的

    本文链接:https://www.haomeiwen.com/subject/lyvwrrtx.html