java新手入門面臨的一個經典的話題,本文意在終結這個話題,java中有說法:Java裏面參數傳遞都是按值傳遞
,怎麼理解這句話?用文字說明恐怕不容易說明白,說明白恐怕也難以想明白。
前提
先明確一下,按值還是按引用的概念,它是來自c++
語言,引用不是漢語詞典中的一個詞,而是c++的概念——“&”這個符號還記得吧?
爲什麼有這個話題呢?其一,是對按引用傳遞理解不透徹;其二,諸多java書籍及討論論點並沒有切中要害。��
一句話概括,按值傳參還是按引用傳參,既然是參數傳遞方式,那麼只針對形參和實參,這裏說的是參數本身,不是參數對象的子對象或孫子對象。
有了前提,上c++代碼:
#include <iostream>
using namespace std;
class User
{
private:
int m_id;
public:
User(int id=0){m_id = id;}
void setId(int id){m_id = id;}
int getId(){return m_id;}
};
void test0(User t){//按值傳參
User s;
t = s;
t.setId(1002);
cout << "test1:" << t.getId() << endl;
}
void test1(User *t){//按值傳參
t = new User();//指針指向了一個新對象,外面實參沒變
t->setId(1002);
cout << "test1:" << t->getId() << endl;
}
void test2(User* & t){//按引用傳參
t = new User();//指針指向了一個新對象,外面實參也跟着變了
t->setId(1002);
cout << "test2:" << t->getId() << endl;
}
int main(int argc, char const *argv[]) {
cout<< "\npass by ref:"<<endl;
User* t = new User();
t->setId(1001);
cout << t->getId() << endl;
test2(t);
cout << t->getId() << endl;
cout<< "\npass by value:"<<endl;
t = new User();
t->setId(1001);
cout << t->getId() << endl;
test1(t);
cout << t->getId() << endl;
return 0;
}
輸出結果:
pass by ref:
1001
test2:1002
1002
pass by value:
1001
test1:1002
1001
c++小結:
按值傳遞,那麼在函數內修改了形參指向一個新對象,外面的實參不受影響。
按引用傳遞,那麼在函數內修改了形參指向一個新對象,外面的實參也變了。
旨在說明問題,代碼可能有內存泄漏。
上java:
package com.pollyduan.bean;
@Data
public class User {
private Integer id;
public static void testObject(User t){
t=new User();//指向了一個新對象,外面實參沒變
t.setId(1002);
System.out.println("testObject="+t);
}
@Test
public void testObject(){
User user=new User();
user.setId(1001);
System.out.println("user="+user);
testObject(user);
System.out.println("user="+user);
}
}
輸出結果:
user=User(id=1001)
testObject=User(id=1002)
user=User(id=1001)
java小結:
跟c++的邏輯比較一下,請自行對號入座。
萬事無絕對,你可能發現jdk中有引用傳參的例子,如:
char[] gg={'a','b','c'};
char[] newGG=new char[gg.length];
System.arraycopy(gg,0,newGG,0,gg.length);
看下源碼,它不是jdk本身的方法,是JNI,也就是說是jvm的JNI庫封裝的方法,知道即可。