跨平台开发公交站

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

2022-03-06  本文已影响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 了吗? 从这个角度理解看起来好像真的就是引用传递!但是可惜这并不是,这是一种误解。

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

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

image

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

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

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

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

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

上一篇下一篇

猜你喜欢

热点阅读