再谈Java方法传参那些事

时间:2021-12-19 23:27:42

把一个变量带进一个方法,该方法执行结束后,它的值有时会改变,有时不会改变。一开始会觉得--“好神奇呀”。当我们了解java内存分析的知识后,一切都是那么简单明了了--“哦,这么回事呀”。但是今天的上机课,对于引用变量,我犯了一个错误,下面是代码的简化--

void method(int a[])
{
    int temp[]={1,2,3,4};
    a[0]=5;
    a=temp;
}

在这里,我试图把临时数组变量的值赋给a,从而改变实参的值。

方法传参,分传值和传址,基本数据类型变量的传参传的是值;引用变量传参传的是地址。传入的参数,实际是实参的一份拷贝。因为,当我们调用一个方法时,逻辑上,jvm会开辟一个我们称之为栈区的内存,当方法运行结束后,这段内存也就被释放了。我们传入的实参的拷贝的生命也仅限于这段栈区,这便是局部变量的宿命。对于基本数据变量,由于它的值是存在栈区的,且与调用该方法的函数的栈内存是独立的,所以影响不到该函数。引用变量的值是地址,它指向堆区的某个地址,这是一个被共享的内存。我们对其的操作就是操作堆区的数据(继续看下文,这句话是不严谨的)。

这是之前我对传参的理解。用它来理解今天的问题,也是没有问题的。但却忽略了一个问题。

既然传入引用变量是传址的,那么上面的代码为什么不能如愿改变它的值呢?

当我们注意到上文说,传参实际是实参的一份拷贝,引用变量也是如此,而且,引用变量也是存在栈区的。那么,我的“a=temp”是在栈区上改变数据(他们的值仅是地址,不能影响堆内存的对象),而且是拷贝而来的数据。

于是:被调用的方法,在堆内存的操作在方法执行结束后才是有效的。变量名永远是存在栈区的(独立,不被共享),引用变量也是如此。

同样的:

void alg1(int a[])//引用变量指向堆内存对象上的操作,是可以改变传入参数变量的
{
    int t[]=a;//把a的值,即指向对象的地址赋值给t
    for(int i=0; i<a.length; i++)
        t[i]=i;
}
void alg2(int a[])//虽然看似对该变量的操作,却不是同一个对象的引用了
{
    int t[]=a;//把a的值,即指向对象的地址赋值给t
    a=new int[5];
    for(int i=0; i<a.length; i++)
        a[i]=t[i]*t[i];
}

《算法设计与分析》中对于方法调用的描述:

通常,在一份算法中调用另一个算法时,系统需要在运行被调用算法之前完成3件事:

  (1)将所有实参指针,返回地址等信息传递给调用算法;

  (2)为被调用算法的局部变量分配储存区;

  (3)将控制转换到被调用算法的入口。

在从被调用算法返回调用算法时,系统也相应地要完成3件事:

  (1)保存被调用算法的计算结果;

  (2)释放分配给被调用算法的数据区;

  (3)依照被调用算法保存的返回地址将控制转移到调用算法。