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库封装的方法,知道即可。
时间: 2024-12-01 00:58:47