`

java中传值与传引用的解析

 
阅读更多

public class Example{

      String str=new String("good");

      char[]ch={'a','b','c'};

      public static void main(String args[]){

             Example ex=new Example();

             ex.change(ex.str,ex.ch);

             System.out.print(ex.str+" and ");

             System.out.print(ex.ch);

      }

      public void change(String str,char ch[]){

             str="test ok";

             ch[0]='g';

      }

}

看看输出结果? good and gbc

java中没有了c++中 这样的引用符号,也没像c#中那样提供了outref 那么它是怎么做的呢

做什么事情都要去除例外的东西,String类就是此类问题的一个特殊情况

为什么特殊呢?

因为它是一个引用类型,确执行的是值传递。这样说有些抽象,还是举个例子吧

值传递:

class Str

public static void mainString[] args

int i  900;

System.out.printlni;

changeInti;

System.Out.printlni;

public static void changeIntint s

 34234;

结果:

900

900

这就是所谓的值传递。i把自己的副本给了函数changeInt的形参,而在changeInt中虽然将s赋值34234。但是对原来的i值并没有影响,因为它所修改的只是icopy品而已。

引用传递:

class Str

public static void mainString[] args

Yinyong  y  new Yinyong();

System.Out.printlny.age;

changeObjecty;

System.Out.printlny.age;

public static void changeObjectYinyong obj

Obj.age  34234;

class Yinyong

int age  22;

声明了个简单的类Yinyong,当把Yinyong的对象y传递给函数changeObject后,看下前后结果:

22

34234

值被改变了,这就是引用调用。

下面再看看传递String对象会发生什么结果?

class Str

public static void mainString[] args

String s  "java test";

System.Out.printlns;

changeStrings;

System.out.printlns;

public static void changeStringString str

str  "3gg over right";

看看结果吧:

java test

java test

你惊奇的发现s的值并没有改变!你会问了,String不也是引用类型的吗?怎么它的值没有改变呢?

因为是这样的:

String被设计为不可修改的类型,也就是对String对象的任何修改都将重新创建一个对象而放弃以前的内存空间的引用!

如上例:

比如 s指向0x2344 当它赋值给strstr也同样指向了0x2344。而当 执行str"3gg over right"后,str指向了别的地方。也许是

0x2222或者其他,反正不是0x2344了。所以当你输出s的时候,它的值并没有被修改!

 

综上所述:基本数据类型传递的是值的拷贝;对象类型传递的是引用的拷贝;而String类型传递的虽然也是对象,但它不同于一般的对象类型,它String被设计为不可修改的类型,也就是对String对象的任何修改都将重新创建一个对象而放弃以前的内存空间的引用!

 

 

按值传递还是按引用传递

这个在Java里面是经常被提起的问题,也有一些争论,似乎最后还有一个所谓的结论:Java里面参数传递都是按值传递。事实上,这很容易让人迷惑,下面先分别看看什么是按值传递,什么是按引用传递,只要能正确理解,至于称作按什么传递就不是个大问题了。

1:按值传递是什么
指的是在方法调用时,传递的参数是按值的拷贝传递。示例如下:

public class TempTest {
private void test1(int a){
//做点事情
}
public static void main(String[] args) {
TempTest t = new TempTest();
int a = 3;
t.test1(a);//这里传递的参数a就是按值传递
}
}

按值传递重要特点:传递的是值的拷贝,也就是说传递后就互不相关了。
示例如下:

public class TempTest {
private void test1(int a){
a = 5;
System.out.println("test1方法中的a==="+a);
}
public static void main(String[] args) {
TempTest t = new TempTest();
int a = 3;
t.test1(a);//传递后,test1方法对变量值的改变不影响这里的a
System.out.println(”main方法中的a===”+a);
}
}

运行结果是:
test1
方法中的a===5
main
方法中的a===3

2:按引用传递是什么
指的是在方法调用时,传递的参数是按引用进行传递,其实传递的引用的地址,也就是变量所对应的内存空间的地址。
示例如下:

public class TempTest {
private void test1(A a){

}
public static void main(String[] args) {
TempTest t = new TempTest();
A a = new A();
t.test1(a); //
这里传递的参数a就是按引用传递
}
}
class A{
public int age = 0;
}

3:按引用传递的重要特点
传递的是值的引用,也就是说传递前和传递后都指向同一个引用(也就是同一个内存空间)。
示例如下:

1 public class TempTest {
2 private void test1(A a){
3 a.age = 20;
4 System.out.println("test1方法中的age="+a.age);
5 }
6 public static void main(String[] args) {
7 TempTest t = new TempTest();
8 A a = new A();
9 a.age = 10;
10 t.test1(a);
11 System.out.println(”main方法中的age=”+a.age);
12 }
13 }
14 class A{
15 public int age = 0;
16 }

运行结果如下:
test1
方法中的age=20
main
方法中的age=20

4:理解按引用传递的过程——内存分配示意图
要想正确理解按引用传递的过程,就必须学会理解内存分配的过程,内存分配示意图可以辅助我们去理解这个过程。
用上面的例子来进行分析:
1):运行开始,运行第8行,创建了一个A的实例,内存分配示意如下:

 
 2):运行第9行,是修改A实例里面的age的值,运行后内存分配示意如下:
 
 

3):运行第10行,是把main方法中的变量a所引用的内存空间地址,按引用传递给test1方法中的a变量。请注意:这两个a变量是完全不同的,不要被名称相同所蒙蔽。

内存分配示意如下:

 
 

由于是按引用传递,也就是传递的是内存空间的地址,所以传递完成后形成的新的内存示意图如下:

 
 


也就是说:是两个变量都指向同一个空间。

4):运行第3行,为test1方法中的变量a指向的A实例的age进行赋值,完成后形成的新的内存示意图如下:

 
 

此时A实例的age值的变化是由test1方法引起的

5):运行第4行,根据此时的内存示意图,输出test1方法中的age=20
6):运行第11行,根据此时的内存示意图,输出main方法中的age=20

5:对上述例子的改变
理解了上面的例子,可能有人会问,那么能不能让按照引用传递的值,相互不影响呢?就是test1方法里面的修改不影响到main方法里面呢?
方法是在test1方法里面新new一个实例就可以了。改变成下面的例子,其中第3行为新加的:


1 public class TempTest {
2 private void test1(A a){
3 a = new A();//新加的一行
4 a.age = 20;
5 System.out.println("test1方法中的age="+a.age);
6 }
7 public static void main(String[] args) {
8 TempTest t = new TempTest();
9 A a = new A();
10 a.age = 10;
11 t.test1(a);
12 System.out.println(”main方法中的age=”+a.age);
13 }
14}
15class A{
16 public int age = 0;
17}

运行结果为:
test1
方法中的age=20
main
方法中的age=10

为什么这次的运行结果和前面的例子不一样呢,还是使用内存示意图来理解一下

6:再次理解按引用传递
1):运行开始,运行第9行,创建了一个A的实例,内存分配示意如下:

 
 

2):运行第10行,是修改A实例里面的age的值,运行后内存分配示意如下:

 
 

3):运行第11行,是把main方法中的变量a所引用的内存空间地址,按引用传递给test1方法中的a变量。请注意:这两个a变量是完全不同的,不要被名称相同所蒙蔽。

内存分配示意如下:

 
 

由于是按引用传递,也就是传递的是内存空间的地址,所以传递完成后形成的新的内存示意图如下:

 
 

也就是说:是两个变量都指向同一个空间。

4):运行第3行,为test1方法中的变量a重新生成了新的A实例的,完成后形成的新的内存示意图如下:

 
 

5):运行第4行,为test1方法中的变量a指向的新的A实例的age进行赋值,完成后形成的新的内存示意图如下:

 
 

注意:这个时候test1方法中的变量aage被改变,而main方法中的是没有改变的。

6):运行第5行,根据此时的内存示意图,输出test1方法中的age=20
7):运行第12行,根据此时的内存示意图,输出main方法中的age=10

7:说明
1):Java里面参数传递都是按值传递这句话的意思是:按值传递是传递的值的拷贝,按引用传递其实传递的是引用的地址值,所以统称按值传递。
2):在Java里面只有基本类型和按照下面这种定义方式的String是按值传递,其它的都是按引用传递。就是直接使用双引号定义字符串方式:String str = “Java私塾”;

分享到:
评论

相关推荐

Global site tag (gtag.js) - Google Analytics