最近写了一个程序,在主函数中调用一个子函数,传的参数是整形数组,子函数无返回值类型,结果在主函数中输出数组,发现数组被改变,很是纳闷,所以就谈久了一下这个问题。
public static void main(String args[]){
int A[]=new int[1],b;
A[0]=1;
b=8;
change1(A,b);
System.out.println(A[0]);
System.out.println(b);
}
public static void change1(int A[],int b){
A[0]=2;
b=9;
}
输出的结果为 2 换行 8
从网上找原因最终发现以下结论:
首先,Java的数据类型分为基本数据类型(int,short,long,double,float,char,byte,boolean)和对象数据类型(比如类,类可以实例化对象)。
1、对于基本数据类型传参,传的是值。在子函数中对此值操作,完全不会影响主函数的值。
2、对于对象数据类型传参,传的是引用。(其实引用我也不理解,但是用C语言的指针来理解完全可以。所以我把它看成指针)
暂且理解为传的是指针,也即是地址,把地址传过去了,也就是在主函数与子函数中操作的数组A是同一个数组,所以在主函数中输出时值发生了改变。
这点切记小心,不然你觉着没问题,到最后发现值改变了,这的确很尴尬。我就是要在主函数中调用子函数下面继续用数组A,结果发现他的值改变了。
然后我有测试了一下赋值运算,发现结果与传参类似。也满足上面的规律。这个赋值也是讲数组A的首地址传给了B,AB指向同一块内从
public class test {
public static void main(String args[]){
int A[]=new int[1],b;
A[0]=1;
b=8;
change1(A,b);
System.out.println(A[0]);
System.out.println(b);
}
public static void change1(int A[],int b){
int B[]=new int[1];
int c;
B=A;
c=b;
System.out.println(A[0]);
System.out.println(b);
B[0]=2;
c=9;
System.out.println(A[0]);
System.out.println(b);
}
}
输出的结果为1,8,2,8,2,8
此例说明对象数据类型赋值是指两个变量同时指向一个内存地址。任何一个变量改变都会导致此地址值得改变。
思考:以后写程序,用到数组等对象数据类型时(假设为对象A)你只要对A操作,或把A赋值给(或传递给)对象B,然后对B操作,A值同样改。所以A的原始值不可能保存下来。谨慎,谨慎。但是还是有解决办法的:
java中有个函数叫”克隆“可以解决这个问题,他可以创建一个新的内存空间,原来的副本数据不会改变。