`
men4661273
  • 浏览: 275651 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

java中的值传递,引用传递

    博客分类:
  • java
阅读更多
        无意中看到一个问题,java是值传递还是引用传递,有人说只有值传递,有人说值传递和引用传递都有,我一直认为是值传递和引用传递都有的,原始数据类型以及其封装类型是值传递,如int,Integer,再被别的方法使用的时候,传递的只是值,或者说是复制了一份内容,给新的方法,方法的处理不会影响方法体之外的该对象。复杂数据类型为引用传递,如自定义的一个Dog,传递的是该类的对象的引用,在方法体内修改所传递的对象的属性值,方法体外的对象的相应属性值也会被改变,所以相当于传递的是对象的本身。如果说非要说只有值传递的话,那么针对我理解的引用传递来说,传递的是对象引用的内存地址的值。
       但是有个类型比较特殊,String,他不是基本数据类型,也不是基本数据类型的封装类型,应该是属于一个复杂数据类型,但是它为什么也不能进行引用传递呢,因为String是不可变对象,一旦声明初始化值之后就会回再发生值的变化,在方法体内对其进行操作,其实是常量池中新建了一个常量值。所以String类型的值传递进去也是不可以被更改的,其内部实现是重载了+号操作符,采用StringBuilder的方式重新创建了String的对象。
包括字符串拼接方法 concat,是返回了一个new String的对象
      如有理解有误的地方,欢迎拍砖。
ps:
     通过别人的指导和资料的查阅,纠正一下,确实就是值传递,或者说是拷贝传递,针对复杂对象来说,作为参数传入的时候,不是传入的原对象的引用地址,而是一份副本,是两个引用,而这个副本和原引用都指向堆上的同一个对象,如果不从新给副本指定其所引用的对象(o = new Object())的话,通过副本可以修改堆上那个对象的状态,如果重新指定了其所引用的对象(o = new Object()),那么这个副本会引用堆上另外一个对象,那么它就与之前的对象没有关系,也就无法修改之前对象的状态了。
2
6
分享到:
评论
6 楼 finallygo 2014-11-21  
可以理解为
void foo(int i) {

}

void foo(Dog *g) {

}
5 楼 113.com 2014-11-19  
men4661273 写道
113.com 写道
men4661273 写道
113.com 写道
其实java 里只有值传递,说有引用传递的,都是没看过真正什么叫引用传递,C++里引用传递,object a =..;
object &c =a;
c和a的地址是一样的,c和c地内存里都是同一块地,a和c指向的都是同一个地址,而已c能改变a的值。
而java
object a =..;
object c =a;
a和c在栈区有各自的地址,只是什么指向同一个堆区的内存地址而已。这只能改变指向对象内部属性。
很明显,如果是真正的引用,c是能改变a的值的,但在java c的值只是a的值的一份拷贝,所以操作c不能改变a的指向。而C++里那才叫什么真正的引用,因为他能改变他引用的对象的值。

c是可以改变a值的啊


object a =new object();

public void op(object c){
c=new object();
}

op(a);
a还是指向原来的对象,c能改变a的值?改变个毛线,java最基础的你都没弄懂

我明白了,c和a是两个不同的引用,但是他们都引用堆上的对象new object();如果不重新定义引用c所引用的对象时,c可以改变的是所引用对象new object()的值,如果给引用c更改了它所引用的对象,那么c就无法引用原来的new object(),所以无法改变new object()的值。是这个意思吧?


理解得差不多,简单点来说
object a =new object();

object a 这引用变量在栈区,new object()在堆区。a的值其实也就是指向的对象在堆区的首地址,说白了也就是一个a =X1223455。你再让c=a;也就是c=X1223455,你可以通过c来改变X1223455地址对象里面的属性,但是你就是改变不了a=X1223455这个事实。所以其实也就是值传递没法改变a变量的值的原因。






4 楼 men4661273 2014-11-19  
113.com 写道
men4661273 写道
113.com 写道
其实java 里只有值传递,说有引用传递的,都是没看过真正什么叫引用传递,C++里引用传递,object a =..;
object &c =a;
c和a的地址是一样的,c和c地内存里都是同一块地,a和c指向的都是同一个地址,而已c能改变a的值。
而java
object a =..;
object c =a;
a和c在栈区有各自的地址,只是什么指向同一个堆区的内存地址而已。这只能改变指向对象内部属性。
很明显,如果是真正的引用,c是能改变a的值的,但在java c的值只是a的值的一份拷贝,所以操作c不能改变a的指向。而C++里那才叫什么真正的引用,因为他能改变他引用的对象的值。

c是可以改变a值的啊


object a =new object();

public void op(object c){
c=new object();
}

op(a);
a还是指向原来的对象,c能改变a的值?改变个毛线,java最基础的你都没弄懂

我明白了,c和a是两个不同的引用,但是他们都引用堆上的对象new object();如果不重新定义引用c所引用的对象时,c可以改变的是所引用对象new object()的值,如果给引用c更改了它所引用的对象,那么c就无法引用原来的new object(),所以无法改变new object()的值。是这个意思吧?
3 楼 113.com 2014-11-19  
men4661273 写道
113.com 写道
其实java 里只有值传递,说有引用传递的,都是没看过真正什么叫引用传递,C++里引用传递,object a =..;
object &c =a;
c和a的地址是一样的,c和c地内存里都是同一块地,a和c指向的都是同一个地址,而已c能改变a的值。
而java
object a =..;
object c =a;
a和c在栈区有各自的地址,只是什么指向同一个堆区的内存地址而已。这只能改变指向对象内部属性。
很明显,如果是真正的引用,c是能改变a的值的,但在java c的值只是a的值的一份拷贝,所以操作c不能改变a的指向。而C++里那才叫什么真正的引用,因为他能改变他引用的对象的值。

c是可以改变a值的啊


object a =new object();

public void op(object c){
c=new object();
}

op(a);
a还是指向原来的对象,c能改变a的值?改变个毛线,java最基础的你都没弄懂
2 楼 men4661273 2014-11-19  
113.com 写道
其实java 里只有值传递,说有引用传递的,都是没看过真正什么叫引用传递,C++里引用传递,object a =..;
object &c =a;
c和a的地址是一样的,c和c地内存里都是同一块地,a和c指向的都是同一个地址,而已c能改变a的值。
而java
object a =..;
object c =a;
a和c在栈区有各自的地址,只是什么指向同一个堆区的内存地址而已。这只能改变指向对象内部属性。
很明显,如果是真正的引用,c是能改变a的值的,但在java c的值只是a的值的一份拷贝,所以操作c不能改变a的指向。而C++里那才叫什么真正的引用,因为他能改变他引用的对象的值。

c是可以改变a值的啊
1 楼 113.com 2014-11-19  
其实java 里只有值传递,说有引用传递的,都是没看过真正什么叫引用传递,C++里引用传递,object a =..;
object &c =a;
c和a的地址是一样的,c和c地内存里都是同一块地,a和c指向的都是同一个地址,而已c能改变a的值。
而java
object a =..;
object c =a;
a和c在栈区有各自的地址,只是什么指向同一个堆区的内存地址而已。这只能改变指向对象内部属性。
很明显,如果是真正的引用,c是能改变a的值的,但在java c的值只是a的值的一份拷贝,所以操作c不能改变a的指向。而C++里那才叫什么真正的引用,因为他能改变他引用的对象的值。

相关推荐

Global site tag (gtag.js) - Google Analytics