java值传递和引用传递

来源:互联网 发布:360度全景软件 编辑:程序博客网 时间:2024/06/01 08:58
值传递
public class MethodAndParam {static void swap(int a,int b){int c = a;a = b;b =c;}/** * @param args */public static void main(String[] args) {int a = 1 ,b =2;swap(a, b);System.out.println(a+","+b);}}

其结果大家都知道,a、b的值没有交换成功。




引用传递

static void addA(StringBuffer sBuf){sBuf.append("A");}/** * @param args */public static void main(String[] args) {StringBuffer sBuf = new StringBuffer();addA(sBuf);System.out.println(sBuf.toString());}
因为是引用传递,能找到对象的真实地址从而改变对象的值,其输出结果为A。


static void addA(String str){str = str + "A";}/** * @param args */public static void main(String[] args) {String str = "";addA(str);System.out.println(str);}

问题一:同样是引用传递,但结果并没有输出A,而是输出了空串,这是为什么呢?


我们将StringBuffer的例子改造如下:

static void addAByNew(StringBuffer sBuf){sBuf = new StringBuffer();sBuf.append("A");}/** * @param args */public static void main(String[] args) {StringBuffer sBuf = new StringBuffer();addAByNew(sBuf);System.out.println(sBuf.toString());}

同样上面的运行结果也没有输出A,而是输出了空串。如果传递的是引用,方法内部的代码sBuf = new StringBuffer()将引用指向了新的地址空间,在新的地址空间里append("A")。



问题二:在方法内部对引用的改变为什么没有影响到外部的对象?



对于问题一,大家可以看一下String对象的源码

public final class String    implements java.io.Serializable, Comparable<String>, CharSequence{    /** The value is used for character storage. */    private final char value[];    /** The offset is the first index of the storage that is used. */    private final int offset;    /** The count is the number of characters in the String. */    private final int count;

String类是final修饰的,表示不能被继承。对于被final修饰的属性的赋值有如下几种方式:

 

    1. 如果有static修饰的属性,可以在static{}块里赋值

    2. 如果没有static修饰,可以在代码块{}里赋值

    3. 在定义变量时赋值

    4. 在构造方法里赋值

即一旦一个对象创建,其final修饰的属性值将不可改变,而String对象中存放值的value数组则是final修饰的,我们可以理解为:一个String对象的值改变后,它已经不是原来的它了。理解了String对象的不变性后可知问题一和问题二背后是同一个原因。

 

从JVM出发来看方法的调用

      栈帧是用于支持虚拟机进行方法调用和方法执行的数据结构,每一个方法从调用开始到执行完成的过程,就对应着一个栈帧在虚拟机栈里面从入栈到出栈的过程。

      栈帧包括了局部变量表、方法返回地址等信息。其中局部变量表是一组变量值空间,用于存放方法参数和方法内部定义的局部变量。



     现在可以回答问题二了:参数的引用传递不是单纯的将对象的引用传递给一个方法,而是将对象引用的地址传递给局部变量表里的参数引用,当在方法内部改变引用的地址时,只会影响方法内部的引用,而不会影响到方法外部的变量。但如果改变的是引用地址的值,方法内外的引用都会受影响。



原创粉丝点击