在c/c++中对于传参类型,无外乎就是传值、传引用、传指针这几种。但在java中,由于没有指针类型,其传参的方式也发生了相应的变化。之前有搜过相关的知识点一直理解的是:Java的传参方式中主要有两种:传值和传引用。(基本数据类型相当于传值,对象相当于传参)
但是,在做需求的时候遇到了,将从上一个方法中获取到的对象,当做参数传进下一个方法中的时候,如果传进来的是空对象,那么在给这个“对象”直接赋值的时候,会报空指针异常!
探究与理解:
Java中传值也就是传的是原变量的一个副本,函数内部无论怎么操作都不会影响到原变量本身,影响到的只是原变量的那个副本。
经典的一个test:
public class Test { public static void main(String[] args){ int a = 1, b = 2; fun(a,b); System.out.println("a="+a);} static void fun(int m , int n){ m = m + n; } }
运行结果:a=1.
另一个例子:
public class Test { public static void main(String[] args){ List<String> str_1 = new LinkedList<String>(); str_1.add("111"); fun(str_1); System.out.println("str_1 ="+str_1);} static void fun(List s){ s.add("222");} }
运行结果:str_1 = [111,222] 从这个例子来看似乎,Java传对象的时候就是传引用。
那么,我们再看另一个例子:
public class Test { public static void main(String[] args){ List<String> str_1 = new LinkedList<String>(); List<String> str_2 = new LinkedList<String>(); str_1.add("111"); str_2.add("333"); fun(str_1,str_2); System.out.println("str_1 ="+str_1); System.out.println("str_2 ="+str_2);} static void fun(List s,List m){ List<String> str = new LinkedList<String>(); str = s; s = m; m = str;} }
运行结果为:str_1 =[111] , str_2 =[333] .我们发现str_1与str_2的值并没有发生交换。 如果该对象传的是引用的话,那它们两个的值应该发生了变化才对。 然而,java中对象作为实参,传递的实际上还是值,即java中的传参传递的都是值!
下面从jvm的构成角度来解释一下自己对这个问题的理解:
JVM的结构图:
从Jvm的结构图上可以看出来,Jvm在执行Java程序的时候将属于它的内存分为五部分。其中程序代码(字节码)和基本的数据类型是放在java栈的栈帧中,而对象是从堆中分配的。堆,我们可以形象的理解成“对象池”。程序和程序中需要用到的对象放在两个相对独立的区域中,那么程序怎么使用对象呢?
实际上是程序中真正使用对象的地方其实只是声明了一个对象的引用,也就是把堆中分配了的相应对象的地址放到引用中,当然引用也是放在栈内存中,栈和堆之间就是通过一个一个的引用来联系的。至于引用,我们可以把它看做是一个指针常量,这个指针指向引用对象的地址。所以,不管是基本类型变量(int,float,double等)还是对象,相应的内存地址中存放的都是一个数(无符号整数,整数,浮点数等)。当方法调用时候,方法的参数会占内存中开辟一块新的区域,同时把要传递的基本类型,或者引用类型名称复制到这块内存中,结果是,基本类型(存放在占内存中的)复制之后连同其数值也复制到了这块内存中,而对象只是复制了引用名,实际还要联系到原对象所在的堆区域中。之所以说java中传递的都是值,是因为传递得是内存地址中的数,当然,这个值对于基本类型和对象类型来说意义是不一样的,对于基本类型这个数就是其值本身,传递值的结果就是,改变新的变量的值不影响旧的变量的值;而对于对象来说这个数是它的地址,传递这个值就相当于传递了真实对象的引用,传递了引用或者说是地址的结果就是变化会全局可见。所以说java中的方法都是按值调用的!只不过基本类型和对象类型的"值"的在具体的机制上作用不同。
所以在第二个例子中,fun方法中,两个list s和m交换,实际只是str_1与str_2在栈中对应的引用值进行了交换,而引用与相应堆区域的联系并没有改变。因此,str_1与str_2的值并没有发生变化。
结论:
java中不论是基本类型还是对象,方法调用时,传递的都是值!
对于基本类型的变量,相当于操作其副本。
对于对象,相当于复制了一个引用,该引用指向的还是原变量的内存区域。