傳值調用(Call by Value)
傳值調用是指方法在調用參數時,不是對原參數進行操作,而是創建參數的拷貝並對其進行操作,這種調用有利於保護數據.其在JAVA中的實現如下:
public class Test {
public static void main(String[] args) {
int a = 0;
int b = 1;
System.out.println("a : " + a + " b : " + b);
swap(a, b);
System.out.println("a : " + a + " b : " + b);
}
//交換i j的值
public static void swap(int i, int j) {
int temp = i;
i = j;
j = temp;
}
}
程式輸出結果:
a : 0 b : 1
a : 0 b : 1
也就是說,其實方法的運行並沒有改變a,b的值,讓我們深入JVM的記憶體中看看是怎么回事:
先看看JVM的存儲區(右圖)
對於swap(a, b)的方法調用,可以這樣理解,a,b兩個引用存儲在Heap內,他們的int值0和1存儲在Stack內,在調用swap方法時,新建了i,j兩個引用
i,j也指向int值0和1,在方法內互換的是i,j所指向的值,所以a,b所指向的並未發生改變
JAVA中的方法,只能對封裝在類內部的變數的值進行操作,例子:
1.MyClass類封裝variable
public class MyClass {
public int variable = 0;
public void setVariable(int i) {
variable = i;
}
}
2.測試類Test
public class Test {
public static void main(String[] args) {
MyClass mc1 = new MyClass();
MyClass mc2 = new MyClass();
mc2.setVariable(1); // mc2.variable的值設為1
swapOne(mc1, mc2); // 這行代碼等於無效
System.out.println("mc1's variable is " + mc1.variable);
System.out.println("mc2's variable is " + mc2.variable);
swapTwo(mc1, mc2); // 這行代碼交換了mc1和mc2的variable的值
System.out.println("mc1's variable is " + mc1.variable);
System.out.println("mc2's variable is " + mc2.variable);
}
public static void swapOne(MyClass mc1k, MyClass mc2k) {
MyClass temp = mc1k;
mc1k = mc2k;
mc2k = temp;
}
public static void swapTwo(MyClass mc1k, MyClass mc2k) {
int temp = mc1k.variable;
mc1k.variable = mc2k.variable;
mc2k.variable = temp;
}
}
程式輸出結果 :
mc1's variable is 0
mc2's variable is 1
mc1's variable is 1
mc2's variable is 0
解析:方法的拷貝出mc1和mc2的引用mc1k和mc2k.
mc1與mc1k指向同一個實例MyClass(其中實例內的variable == 0);
mc2與mc2k指向同一個實例MyClass(其中實例內的variable == 1);
swapOne方法交換了mc1k和mc2k所指向的對象,但是mc1k和mc2k在方法結束後被JVM回收了,mc1和mc2沒有改變
swapTwo方法通過mc1k和mc2k互換了指向的對象MyClass裡面的變數variable的值,mc1和mc2的指向的實例並未改變,但是指向的實例內的變數的值改變了,所以看起來好像是mc1和mc2所指向的實例互換了.
在C語言中,被調用函式不能直接修改主調用函式中變數的值,而只能修改其私有的臨時副本的值。
傳值調用,即傳遞給被調用函式的參數值存放在臨時變數中,而不是存放在原來的變數中。
例如求冪:
#include <stdio.h>
調用函式:
int power(int,int );//函式聲明,即函式原型,其中的參數類型不可省略,但參數名是可選的
int main()
{
int i;
for(i=0;i<10;++i)
printf("%d %3d %6d\n",i,power(2,i),power(-3,i));
getchar();
}
以下是函式定義;n>=0
每次調用時,main函式向power函式傳遞2個參數,在調用執行完成時,power函式向main函式返回一個格式化的整數並列印。
被調用函式①
int power(int base,int n)
{
int i,p;
p=1;
for(i=1;i<=n;++i)
p=p*base;
return p;
}
函式也可以定義成以下模式、
被調用函式②:
int power(int base,int n)
{
int p;
for(p=1;n>0;--n)
p=p*base;
return p;
}
函式②中n是調用函式main傳遞給被調用函式power的,在for循環中n被當成臨時變數直到自減為不大於0而退出循環;這種方法稱為參數的‘傳值調用’。顯然函式②比函式①少用了一個變數i,這樣程式也更緊湊簡潔。