讨论下,java中有无引用传递,
为什么很多人说java中没有引用传递呢?
------解决方案--------------------------------------------------------
我个人理解翻来覆去就两句话:
1、Java函数调用时,参数传递方式为值传递;
2、当该参数类型为对象(Object)时,则所传递的值为对象的引用(也可理解为地址)。
如果你认为Java函数调用有指针传递或引用传递的话,你可以尝试编写这样一个函数来验证:
public void swap(Object a, Object b);
效果就是调用后,主调函数的这两个对象互换即可。
比如主调函数是:
String a = "A";
String b = "B";
swap(a, b);
System.out.println(a); // 这里请输出 B
System.out.println(b); // 这里请输出 A
当然,你别用反射之类的方式直接去修改常量池啥的招数。。。
------解决方案--------------------------------------------------------
如果传递的是primities : int, double, short etc, 是值传递。
如果传递的是对其他object的引用,就是传递的那个object reference
int cube (int a) {<-- pass by value
return a*a*a
}
Box enLargbox ( Box aBox){ <-- pass by reference
aBox.setLength (aBox.length * 2);
aBox.setWidth (aBox.width *2);
aBox.setHight (aBox.hight *2);
return aBox;
}
------解决方案--------------------------------------------------------
其实不能说Java "无引用传递", 而是Java针对非Primitive的数据类型"自动”进行了引用传递。
在C++里,假设有一个对象叫做foo, 那么在传递参数的时候:
- C/C++ code
//函数funvoid fun(Foo foo){ ...}//传递参数foo给函数funfun(foo);
------解决方案--------------------------------------------------------
你就这样理解:
值传递就是值的拷贝
引用传递就是引用的拷贝
比如
public class Demo{
public void testInt(int i){
++i;
}
public void testStrintBuilder(StringBuilder sb){
sb.append("123");
}
public static void main(String[] args){
Demo demo = new Demo();
int i = 1;
demo.testInt(i);
StringBuilder sb = new StringBuilder();
demo.testStrintBuilder(sb);
//打印1,表示参数拷贝
System.out.println(i);
//打印123,表示参数没有拷贝,还是之前的引用
System.out.println(sb.toString);
}
}
------解决方案--------------------------------------------------------
1、java无引用传递,方法的参数(基础类型)在传进去之后,对参数的操作不会修改原值;
2、对于参数为对象的情况,在传递之前,是复制该对象的引用,在方法内,如果复制的该引用重新赋值了,传参数的那个对象不会被改变;
------解决方案--------------------------------------------------------
这是我以前写的一个测试类,大家看看,不包无错
- Java code
import java.util.ArrayList;import java.util.List;/** * 结论:在JAVA中 * 赋值或传递基本类型时,是通过复制基本类型的值来实现的; * 赋值或传递对象时,是通过传递引用(指针)来实现的。 */public class PassParameter { public int count = 0; public String content = "Hello world"; public PassParameter pp; public static void main(String[] args) { PassParameter p = new PassParameter(); int i1 = 0; int i2 = i1; Integer integer = 0; String str = "Hello world"; List<Integer> intListA = new ArrayList<Integer>(); List<Integer> intListB = new ArrayList<Integer>(); List<String> strListA = new ArrayList<String>(); List<String> strListB = strListA; ///////////////////////////////////////////////////////// // 基本类型是通过复制值传递的 System.out.println("Before: i = " + i1); p.addInt(i1); System.out.println("After: i = " + i1); // 对象是通过传递引用(指针)传递的 System.out.println("Before: str = " + str); p.appendString(str); System.out.println("After: str = " + str); // 对象是通过传递引用(指针)传递的 System.out.println("Before: Integer = " + integer); p.addInteger(integer); System.out.println("After: Integer = " + integer); // 成员变量基本类型与普通基本类型无异 System.out.println("Before: count = " + p.count); p.addInt(p.count); System.out.println("After: count = " + p.count); // 成员String类型变量与普通String类型变量无异 System.out.println("Before: content = " + p.content); p.appendString(p.content); System.out.println("After: content = " + p.content); // 对象是通过传递引用(指针)传递的 p.pp = new PassParameter(); System.out.println("Before: Class = " + p.pp.count); p.changeClass(p.pp); System.out.println("After: Class = " + p.pp.count); // 容器(对象)是通过传递引用(指针)传递的 System.out.println("Before: List size = " + intListA.size()); p.addInteger(intListA); System.out.println("After: List size = " + intListA.size()); intListB = intListA; p.addInteger(intListB); System.out.println("After2: List size = " + intListA.size()); // 基本类型是通过复制值传递的 i1 = 10000; i2 = i1; i1++; System.out.println("i1 = " + i1); System.out.println("i2 = " + i2); // 容器(对象)是通过传递引用(指针)传递的 strListA.add("a"); strListB.add("b"); System.out.println("strListA.size = " + strListA.size()); System.out.println("strListB.size = " + strListB.size()); ///////////////////////////////////////////////////////// } public void addInt(int i) { i++; } // 由于传递后对象进行重赋值使引用对象(指针)发生了改变,所以对原对象不会有影响 public void addInteger(Integer i) { i = 10; } // 由于传递后对象进行重赋值使引用对象(指针)发生了改变,所以对原对象不会有影响 public void appendString(String str) { str = str + " !"; } public void addInteger(List<Integer> intList) { intList.add(1); intList.add(2); intList.add(3); intList.add(4); intList.add(5); } public void changeClass(PassParameter pp) { pp.count++; }}