日期:2014-05-20  浏览次数:20915 次

java中有无引用传递
讨论下,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, 那么在传递参数的时候:

//函数fun
void fun(Foo foo){
      ...
}
//传递参数foo给函数fun
fun(foo);

这时候直接这么写,C++编译器就直接是把对象foo按值拷贝传递给了函数fun, 如果要使用按引用传递,则必须写成:

fun(&foo);

在Java里,由于在语法级别没有引用和指针的概念了,Java虚拟机强制规定,对于Primitive类型(int,float,double,boolean等基础数据类型),在传递参数时按值传递;对继承于Object的对象类型,在传递参数是“必须”使用引用传递,也就是说程序员没有选择的余地,即使想在Java中按值传递对象也是不可能的
------解决方案--------------------
你就这样理解:
值传递就是值的拷贝
引用传递就是引用的拷贝
比如
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、对于参数为对象的情况,在传递之前,是复制该对象的引用,在方法内,如果复制的该引用重新赋值了,传参数的那个对象不会被改变;

------解决方案--------------------
这是我以前写的一个测试类,大家看看,不包无错

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;