Passing by reference or by value? it is a question

来源:互联网 发布:部落冲突 知乎 编辑:程序博客网 时间:2024/05/01 09:18
Java中的基础类型直接存储在栈中,复合类型采用引用类型,把引用也存储在栈中,而对应的对象存储在 堆中。因此java中把内存分堆内存和栈内存,在函数中定义的一些基本类型或引用都分配栈内存。 堆内存用来存放由new创建的对象和数组,或是static(类装载信息)。 在堆中分配的内存,由jvm的gc治理。 程序只能控制引用的生存期,对象的生存期是jvm控制的。 在java应用程序中,当对象的引用是传递方法一个参数时,传递的时该引用的一个副本,(按值传递)。而 非引用本身,调用方法的对象引用和副本都是指向同一个对象。 对象是按引用传递的,java应用程序中有且仅有一种参数传递机制---值传递 按值传递的语义就是当将一个参数传递给一个函数时,函数接受的是原始值的一个副本。 按引用传递的语义就是当将一个参数传递给一个函数时,函数接受的是原始值的内存地址,而非一个副本。 public class Test { public static void main(String args[]){ StringBuffer sb1 = new StringBuffer("good"); StringBuffer sb2 = sb1; sb2.append(" afternoon"); System.out.println("sb1 == "+sb1); } } 运行的结果:good afternoon 对象的赋值操作是传递对象的引用,sb1和sb2都指的是同一个对象,这里的其实也是传值,传的是指针的值。这里的 赋值是指针之间的赋值。 1. 引用是一种数据类型,保存了对象在内存中的地址,这种类型即不是我们平时所说的简单数据类型也不是类实例(对象); 2. 不同的引用可能指向同一个对象,换句话说,一个对象可以有多个引用,即该类类型的变量。 public class Test1 { public void fun(String s){ s = "hehe"; } public static void main(String args[]){ Test1 test = new Test1(); String str = "haha"; test.fun(str); System.out.println("str == "+str); } } public class Test2 { public void fun(ArrayList al){ al.add("hehe"); al.add("haha"); } public static void main(String args[]){ Test2 test = new Test2(); ArrayList al = new ArrayList(); test.fun(al); Iterator it = al.iterator(); while(it.hasNext()){ System.out.println(""+(String)it.next()); } } } public class Test3{ public void fun(ArrayList al){ ArrayList alA = new ArrayList(); alA.add("haha"); alA.add("hehe"); al = alA; } public static void main (String args[]){ Test3 test = new Test3(); ArrayList al = new ArrayList(); test.fun(al); System.out.println(""+al.size()); } } 通过以上例子可以得出 1、假如参数是不可变对象,如原语类型(i.e.int)或者不可变的对象类型(i.e.BigInteger) 这样的参数是安全的,方法体内的任何动作都不会影响方法外的内容 2、对于可变对象的类型,如ArrayList v 对v调用add()、remove()等方法会导致方法外的变量的变化 这时可能会导致代码的不安全 但是假如在方法体内对该参数有重新赋值的操作,如v = new ArrayList() 这样v指向的地址已经变化,之后对v的任何操作也不会影响方法外的内容
原创粉丝点击