您的位置:首页 > 编程语言 > Java开发

老生常谈--java中到底是按值传递还是按引用传递

2017-10-18 13:53 411 查看
    java到底是按值传递还是按引用传递?这是一个老生常谈的问题...而且在网上总是能找到不同的理解和答案,我在知乎上做了一些整理。

    在讨论这个问题之前,先看一些容易被我们忽略的一个知识,java中的=,虽然你可能认为这很简单,但是还是请耐心看完:

= 是赋值操作(任何包含=的如+=、-=、 /=等等,都内含了赋值操作)。不再是你以前理解的数学含义了,而+ - * /和 = 在java中更不是一个级别,换句话说, = 是一个动作,一个可以改变内存状态的操作,一个可以改变变量的符号,而+ - * /却不会。这里的赋值操作其实是包含了两个意思:1、放弃了原有的值或引用;2、得到了 = 右侧变量的值或引用。Java中对 = 的理解很重要啊!!可惜好多人忽略了,或者理解了却没深思过。对于基本数据类型变量,= 操作是完整地复制了变量的值。换句话说,“=之后,你我已无关联”;至于基本数据类型,就不在这科普了。对于非基本数据类型变量,= 操作是复制了变量的引用。换句话说,“嘿,= 左侧的变量,你丫别给我瞎动!咱俩现在是一根绳上的蚂蚱,除非你再被 = 一次放弃现有的引用!!上面说了 = 是一个动作,所以我把 = 当作动词用啦!!”。而非基本数据类型变量你基本上可以参数本身是变量,参数传递本质就是一种 = 操作参数是变量,所有我们对变量的操作、变量能有的行为,参数都有。所以把C语言里参数是传值啊、传指针啊的那套理论全忘掉,参数传递就是 = 操作。  

    看完了=的含义,也看到了最后一句话:参数传递就是=操作
    综上,再深入理解一下,首先java中的参数传递实际上是=操作,对于基本数据类型来说,形参通过值传递传递了实参的数值,而对于非基本数据类型来说,形参通过值传递传递了实参那个对象的地址。也就是说,实际上可以理解为java中的传递是值传递,只不过对于基本数据类型来说,值传递传递的是基本类型字面值的拷贝,而当参数是一个对象的引用类型的时候,值传递传递的是对象引用的地址值的拷贝。
    说到了这里可以说对许多参数传递的问题都有了更深刻的理解:
    为什么当参数是基本数据类型的时候,在函数内修改参数,调用函数之后这个基本数据类型数值不会变?因为他只不过是一个拷贝而已。
public static void test1(int i){
i=2;
}
public static void main(String [] args){
int i=1;
test(i);//传递的不过是一个字面值的复制
System.out.println(i);//1
}


    为什么当参数是对象引用的时候,在函数内部修改对象,调用函数之后对象也改变了?因为传参穿的是地址的复制,在函数里改变这个地址对应的数据,函数执行完之后这个地址对应的数据当然改变了。
public static void test(Entity e){//测试方法 传递一个对象的地址的复制
e.num=2;//修改内部属性
}
public static void main(String[] args){
Entity e = new Entity(3);//把e.num设置为3
test(e);//传递的是e的地址的复制
System.out.println(e.num)//2;
}


    为什么当参数是对象引用的时候,在函数内部修改参数的指向,调用函数之后指向依然不变?因为传参的时候穿的是地址的复制,相当于参数传递之后形参就指向了被传递的地址,之后修改参数指向其他地址对函数之外没有任何影响。
public static void test(String e){//测试类
e = e.toLowerCase();//尝试修改形参指向
}
public static void main(String[] args){
String e = new String("ABC");
test(e);//传递的是e地址的复制 也就是函数内部的e指向了堆中的new String("ABC") 函数内部还想让e指向一个其他东西,那就和new String("ABC")没有一点关系了。
System.out.println(e);//ABC
}
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: