Java基本类型 和 引用类型的不同之处

一:搞清楚 基本类型 和 引用类型的不同之处

int num = 10;
String str = "hello";

 

166032bc90958c21604110441ad03f45_r.jpg

如图所示,num是基本类型,值就直接保存在变量中。而str是引用类型,变量中保存的只是实际对象的地址。一般称这种变量为"引用",引用指向实际对象,实际对象中保存着内容。

二:搞清楚赋值运算符(=)的作用

num = 20;
str = "java";

111421_BNZi_2927338.png

对于基本类型 num ,赋值运算符会直接改变变量的值,原来的值被覆盖掉。
对于引用类型 str,赋值运算符会改变引用中所保存的地址,原来的地址被覆盖掉。但是原来的对象不会被改变(重要)。
如上图所示,"hello" 字符串对象没有被改变。(没有被任何引用所指向的对象是垃圾,会被垃圾回收器回收)

三:调用方法时发生了什么?参数传递基本上就是赋值操作

第一个例子:基本类型
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"。

 

重点理解为什么,第三个例子和第四个例子结果不同?

下面是第三个例子的图解:

111444_9bY9_2927338.png

builder.append("4")之后

111458_NsgN_2927338.png

下面是第四个例子的图解:

111509_zAev_2927338.png

 

builder = new StringBuilder("ipad"); 之后

111518_JjBS_2927338.png

以下各种类型数据在内存中的存储方式:

从局部变量/方法参数开始讲起:

局部变量和方法参数在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的数组。

112006_vP6B_2927338.png

所以当我们传递一个数组引用给一个方法时,数组的元素是可以被改变的,但是无法让数组引用指向新的数组。

你还可以这样声明:int[][] arr3 = new int[3][],这时内存情况如下图

112018_qMZu_2927338.png

你还可以这样 arr3[0] = new int [5]; arr3[1] = arr2[0];

112029_l4OO_2927338.png

关于String:

上面关于String的图解是简化过的,实际上String对象内部仅需要维护三个变量,char[] chars, int startIndex, int length。而chars在某些情况下是可以共用的。但是因为String被设计成为了不可变类型,所以你思考时把String对象简化考虑也是可以的。

String str = new String("hello")

112043_XKMF_2927338.png

当然某些JVM实现会把"hello"字面量生成的String对象放到常量池中,而常量池中的对象可以实际分配在heap中,有些实现也许会分配在方法区,当然这对我们理解影响不大。

转载于:https://my.oschina.net/wangfushu/blog/1647689

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java 中的数据类型可以分为基本数据类型和引用数据类型基本数据类型有 boolean、byte、short、int、long、float 和 double,它们存储的是简单的值。而引用数据类型则包括类、接口、数组等类型,它们存储的是对象的引用,也称为句柄。 以下是java基本数据类型的例子: 1.boolean类型,表示逻辑值,取值为 true 和 false。 ```java boolean b = true; ``` 2.byte类型,8 位二进制补码。在 Java 中,byte 类型用于节省空间,可取值范围为 -128 到 127。 ```java byte b = 100; ``` 3.short类型,16 位二进制补码。在 Java 中,short 类型同样用于节省空间,可取值范围为 -32768 到 32767。 ```java short s = 1000; ``` 4.int类型,32 位二进制补码。在 Java 中,int 类型是最常用的数据类型,可取值范围为 -2147483648 到 2147483647。 ```java int i = 100000; ``` 5.long类型,64 位二进制补码。在 Java 中,long 类型用于存储较大范围的整数,可取值范围为 -9223372036854775808 到 9223372036854775807。 ```java long l = 100000000L; ``` 6.float类型,32 位单精度浮点数,可取值范围为 1.4E-45 到 3.4028235E38。 ```java float f = 3.1415926f; ``` 7.double类型,64 位双精度浮点数,可取值范围为 4.9E-324 到 1.7976931348623157E308。 ```java double d = 3.1415926535897932; ``` 以下是java引用数据类型的例子: 1.类类型,可以用来定义类的属性和方法。 ```java class Person { String name; int age; } ``` 2.接口类型,可以用来定义接口的方法。 ```java interface Printable { void print(); } ``` 3.数组类型,可以用来定义数组。 ```java int[] arr = {1, 2, 3, 4, 5}; ```

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值