基本类型 和 引用类型的不同
int num = 10;
String str = "hello";
如图所示:num是基本类型,值直接保存在变量中。而str是引用类型,变量中保存实际对象的地址。一般称这种变量为"引用",引用指向实际对象,实际对象中保存着内容。
赋值运算符(=)的作用
num = 20;
str = "java";
对于基本类型num,赋值运算会直接覆盖掉原来的值。
对于引用类型str,赋值运算会改变引用中保存的地址。但是原来的对象不会被改变。
调用方法时参数传递等同于赋值操作。
第一个例子:基本类型
void foo(int value) {
value = 100;
}
foo(num); // num 没有被改变
第二个例子:没有提供改变自身方法的引用类型
void foo(String text) {
text = "windows";
}
foo(str); // str 也没有被改变
第三个例子:提供了改变自身方法的引用类型
StringBuilder sb = new StringBuilder("iphone");
void foo(StringBuilder builder) {
builder.append("4");
}
foo(sb); // sb 被改变了,变成了"iphone4"。
第四个例子:提供了改变自身方法的引用类型,但是不使用,而是使用赋值运算符。
StringBuilder sb = new StringBuilder("iphone");
void foo(StringBuilder builder) {
builder = new StringBuilder("ipad");
}
foo(sb); // sb 没有被改变,还是 "iphone"。
下面是第三个例子的图解:
调用foo(sb);方法时可以理解为:
void foo(StringBuilder builder) {
--->builder = sb; //builder与sb保存的是同一个地址
builder.append("4");
}
builder.append(“4”)之后
下面是第四个例子的图解:
和第三个例子一样,调用方法传参等同于赋值操作:
void foo(StringBuilder builder) {
--->builder = sb; //builder与sb保存的是同一个地址
builder = new StringBuilder("ipad");//new了一个对象,builder引用存的就是“ipad”的地址了,原来的sb没有被改变
}
builder = new StringBuilder(“ipad”); 之后
原答主之后添加的内容:
各种类型数据在内存中的存储方式
局部变量和方法参数在jvm中的储存方法是相同的,都是在栈上开辟空间来储存的,随着进入方法开辟,退出方法回收。以32位JVM为例,boolean/byte/short/char/int/float以及引用都是分配4字节空间,long/double分配8字节空间。对于每个方法来说,最多占用多少空间是一定的,这在编译时就可以计算好。
我们都知道JVM内存模型中有,stack和heap的存在,但是更准确的说,是每个线程都分配一个独享的stack,所有线程共享一个heap。对于每个方法的局部变量来说,是绝对无法被其他方法,甚至其他线程的同一方法所访问到的,更遑论修改。
当我们在方法中声明一个 int i = 0,或者 Object obj = null 时,仅仅涉及stack,不影响到heap,当我们 new Object() 时,会在heap中开辟一段内存并初始化Object对象。当我们将这个对象赋予obj变量时,仅仅是stack中代表obj的那4个字节变更为这个对象的地址。
数组类型引用和对象
当我们声明一个数组时,如int[] arr = new int[10],因为数组也是对象,arr实际上是引用,stack上仅仅占用4字节空间,new int[10]会在heap中开辟一个数组对象,然后arr指向它。
当我们声明一个二维数组时,如 int[][] arr2 = new int[2][4],arr2同样仅在stack中占用4个字节,会在内存中开辟一个长度为2的,类型为int[]的数组,然后arr2指向这个数组。这个数组内部有两个引用(大小为4字节),分别指向两个长度为4的类型为int的数组。
所以当我们传递一个数组引用给一个方法时,数组的元素是可以被改变的,但是无法让数组引用指向新的数组。
例如:
int[] a = new int[3];
void func(int[] b){
int[]c = {1,2,3};
b = c;//这样不会改变a
}
你还可以这样声明:int[][] arr3 = new int[3][],这时内存情况如下图
你还可以这样 arr3[0] = new int [5]; arr3[1] = arr2[0];
关于String
原本回答中关于String的图解是简化过的,实际上String对象内部仅需要维护三个变量,char[] chars, int startIndex, int length。而chars在某些情况下是可以共用的。但是因为String被设计成为了不可变类型(没有提供setter或其他改变自身状态的方法),所以你思考时把String对象简化考虑也是可以的。
String str = new String(“hello”)
当然某些JVM实现会把"hello"字面量生成的String对象放到常量池中,而常量池中的对象可以实际分配在heap中,有些实现也许会分配在方法区,当然这对我们理解影响不大。
Java没有栈上对象,只有堆上对象,Java的引用都是指针,从这个意义上来说,Java始终是传值的。
作者:Intopass
链接:https://www.zhihu.com/question/31203609/answer/50992895
来源:知乎