一、什么是不可变
如果一个对象它被创建后,状态不能改变,则这个对象被认为是不可变的。
二、怎么实现不可变
当使用final修饰基本类型变量时,不能对基本类型变量重新赋值,因此基本类型变量不能被改变。但对于引用类型变量而言,它保存的仅仅是一个引用,final只保证这个引用变量所引用的地址不会改变,即一直引用同一个对象,但这个对象完全可以发生改变。例如某个指向数组的final引用,它必须从此至终指向初始化时指向的数组,但是这个数组的内容完全可以改变。
即:被final修饰的java值不可变,首先java是值传递,
1、对于基本数据类型是不可变,
2、对象保存的是引用,保证引用地址不变,但是引用内容可变。例如String类
String类的两个主要成员变量
value与hash。这里只对value做分析
public final class String
implements java.io.Serializable, Comparable<String>, CharSequence {
/** The value is used for character storage. */
private final char value[];
/** Cache the hash code for the string */
private int hash; // Default to 0
value指向的是一个字符串数组,字符串中的字符就是用这个value变量存储起来的,并且用final修饰,也就是说value一旦赋予初始值之后,value指向的地址就不能再改变了。虽然value指向的数组是可以改变的,但是String也没有提供相应的方法让我们去修改value指向的数组的元素。然而在StringBuilder中是提供了响应的方法让我们去修改value指向的数组的元素,这也是StringBuilder的字符串序列可变的原因。
即:String类的地址没有改变,value指向的值可以改变。
例:
public static void main(String[] args) {
String str1 = "hello";
// 打印str1的内存地址
System.out.println("str1的内存地址:" + System.identityHashCode(str1));
String str2 = "world";
str1 += str2;
// str1的内存地址已经改变了
System.out.println("执行+=后str1的内存地址:" + System.identityHashCode(str1));
System.out.println("拼接之后str1的值:" + str1);
结果:
str1的内存地址:1922154895
执行+=后str1的内存地址:883049899
拼接之后str1的值:helloworld
改变的是String类中value的值,而不是String类型的引用值。
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。
文章由极客之音整理,本文链接:https://www.bmabk.com/index.php/post/13007.html