弄懂Java为何只有值传递
最近有同学问我关于Java中值传递与引用传递的问题,在此小结一下
值传递是指在函数调用时将实参内容复制一份传递给形参,这样在函数中改变该参数不会对原参数产生影响。
引用传递是指将对象地址的引用传递给该函数,这样在函数中改变该参数会对原参数产生影响。
两者的主要区别在于:是否将实参内容复制一份传递给了形参!
举个最简单的例子:
- public class test {
- public static void main(String[] args) {
- int num = 10;
- changeNum(num);
- System.out.println(num);
- }
- public static void changeNum(int num){
- System.out.println(num);
- num = 20;
- System.out.println(num);
- }
- }
运行结果如下: 都知道这是值传递,将实参内容10复制一份传递给了该函数,对原参数不产生影响。
这个例子开始犯迷糊,以为是传递该地址过去
- public class test {
- public static void main(String[] args) {
- Person p1 = new Person();
- p1.setName("张三");
- p1.setSex("男");
- changePerson(p1);
- System.out.println(p1);
- }
- public static void changePerson(Person person){
- person.setName("小花");
- person.setSex("女");
- }
- }
- public class Person {
- private String name;
- private String sex;
- //省略了getter/setter/toString
- }
运行结果如下:我同学在这里犯的迷糊,跟我说是引用传递
将test类的代码稍加修改加两行代码:
- public class test {
- public static void main(String[] args) {
- Person p1 = new Person();
- p1.setName("张三");
- p1.setSex("男");
- changePerson(p1);
- System.out.println(p1);
- }
- public static void changePerson(Person person){
- person = new Person();
- person.setName("小花");
- person.setSex("女");
- System.out.println(person);
- }
- }
运行结果如下:可见并不是引用传递,如果是引用传递则两次输出结果都改为name=小花,sex=女了。
还是前面所说:两者的主要区别在于:是否将实参内容复制一份传递给了形参!
此处还是值传递,复制了实参内容,只不过实参内容是该对象的地址值,注意地址值也是值!
举个例子:你把你家的钥匙刻了一把给B,B拿着刻印的钥匙去你家打扫卫生,等你回来时发现家里变干净了,你家受到了影响,但你的钥匙并没有受到影响。
另一个容易对Java值/引用传递犯迷糊的例子:运行结果为:“abc”
- public class test {
- public static void main(String[] args) {
- String s1 = "a"+"b"+"c";
- changeStr(s1);
- System.out.println(s1);
- }
- public static void changeStr(String str){
- str+="d";
- }
- }
关于String类有一点特殊,它是不可变类,建议了解下我之前小结的搞定Java String面试题
关于字符串中的”+”号:Java有这么一串话:
java语言提供了字符串串联运算符特殊支持( + ),和其他对象转换为字符串。字符串连接是通过
StringBuilder
实施(或StringBuffer
)类及其append
方法。字符串的转换是通过方法toString
实施,由Object
和继承的所有类的java。
所以该str+=”d”等于String str = new StringBuffer(str).append(“d”).toString();
所以此处相当于新建了一个String对象。