编程的人,都会遇到值传递与引用传递的困惑,不过很快都会迎刃而解。本文通过图文并茂的形式,解释Java的值传递与引用传递。并且会通过String这个特殊的类,进一步加深您的对值传递与引用传递的印象。
?声明:
为了图解方便,图中的术语不精确、甚至是“自创的”,请不要把图中的概念与JVM或者真正的内存相结合,只是为了说明方便!!
?防止误解。?
?
说明:?
图的标号在图的下方;
栈1表示main方法的栈,栈2表示doSomething的栈;
绿色的栈,表示当前的正在运行的栈;红色的栈,表示挂起的栈;白色的栈,表示废弃的栈。
?
?
一般的解释:
?
Java代码??
- public?class?ReferenceCrack?{??
- ??
- ????public?void?doSomething(int?a,?Name?b)?{??
- ????????a?=?100;??
- ????????b.setName("World");??
- ????}??
- ??
- ????public?static?void?main(String[]?args)?{??
- ????????int?numb?=?1;??
- ????????Name?obj?=?new?Name();??
- ????????obj.setName("Hello");???
- ????????new?ReferenceCrack().doSomething(numb,?obj);???
- ??????????
- ????????System.out.println("numb?=?"?+?numb?+?"?;?obj.name?=?"?+?obj.getName());??
- ????}??
- ??
- }??
- ??
- class?Name?{??
- ????private?String?name;??
- ??
- ????public?String?getName()?{??
- ????????return?name;??
- ????}??
- ??
- ????public?void?setName(String?name)?{??
- ????????this.name?=?name;??
- ????}??
- }??
?
?? 输出结果:
numb = 1 ; obj.name = World
图1.1? 执行到obj.setName("Hello"); //--> 见图1.1时,值栈中的内容
?
?图1.2 public void doSomething(int a, Name b) {// --> 见图1.2 刚刚进入另一个函数
?当调用函数doSomething时,Main函数挂起。
?注意: 此时栈2中的a指向的是另一个值“1”。这就是常说的 值传递!!
?
?图1.3 被调用的函数执行过程中。
?doSomething的方法,改变了一些内容?。
?
?图1.4 函数执行完毕,返回Main函数时,值栈中的内容:
?由于栈1中的obj 和 栈2中的b 指向的是 同一个内容,而该内容被b修改了,所以obj的内容就是修改后的内容。
?因此输出是: numb = 1 ; obj.name = World
?
?
传统的方式,大家看完图后都明白的。让我们更进一步:
?
Java代码??
- public?class?ReferenceCrack01?{??
- ??
- ????public?void?doSomething(int?a,?String?b)?{??
- ????????a?=?100;??
- ????????b?=?b.trim();??
- ????}??
- ??
- ????public?static?void?main(String[]?args)?{??
- ??
- ????????int?numb?=?1;??
- ????????String?str?=?"Hello?World?????";??
- ??????????
- ????????new?ReferenceCrack01().doSomething(numb,?str);??
- ??????????
- ????????System.out.println(numb);??
- ????????System.out.println(str+"|");??
- ????}??
- ??
- }??
?
?图2.1
?图2-2
?图2-3
?注意:这个图与1-3的图不一样。(下文解释)
?
DBA9106DA3C.png">
?图2-4
图2-3的解释:
Java中的String类是Final的,是不允许修改的。因此在对String做任何操作时,要么返回自身(this)要么返回一个新的对象!
?
Java代码??
- public?String?trim()?{??
- ????????int?len?=?count;??
- ????????int?st?=?0;??
- ????????int?off?=?offset;???
- ????????char[]?val?=?value;???
- ??
- ????????while?((st?<?len)?&&?(val[off?+?st]?<=?'?'))?{??
- ????????????st++;??
- ????????}??
- ????????while?((st?<?len)?&&?(val[off?+?len?-?1]?<=?'?'))?{??
- ????????????len--;??
- ????????}??
- ????????return?((st?>?0)?||?(len?<?count))???substring(st,?len)?:?this;??
- ????}??
?
Java代码??
- public?String?substring(int?beginIndex,?int?endIndex)?{??
- ????if?(beginIndex?<?0)?{??
- ????????throw?new?StringIndexOutOfBoundsException(beginIndex);??
- ????}??
- ????if?(endIndex?>?count)?{??
- ????????throw?new?StringIndexOutOfBoundsException(endIndex);??
- ????}??
- ????if?(beginIndex?>?endIndex)?{??
- ????????throw?new?StringIndexOutOfBoundsException(endIndex?-?beginIndex);??
- ????}??
- ????return?((beginIndex?==?0)?&&?(endIndex?==?count))???this?:??
- ????????new?String(offset?+?beginIndex,?endIndex?-?beginIndex,?value);??
- ????}??
?
有JDK源码可以清楚地看到,生成了一个新的String。所以在图2-3中,栈2中的b变量的指向发生了变化。