【面试】Java 之 String 系列 -- String 为什么不可变?

在 Java 编程中,String 类是一个使用频率极高的类。而 String 对象具有不可变的特性,这一特性在 Java 设计中有着重要的意义。本文将深入探讨 String 不可变的含义、原因以及带来的好处。

一、String 不可变的含义

1. 概念解释

所谓 String 不可变,指的是一旦一个 String 对象被创建,它的内容(即字符序列)就不能被改变。在 Java 里,String 类被设计为 final 类,并且其内部用于存储字符序列的 value 数组也是 privatefinal 的。以下是 String 类中部分相关代码:

java 复制代码
public final class String
    implements java.io.Serializable, Comparable<String>, CharSequence {
    /** The value is used for character storage. */
    private final char value[];

    // 其他代码...
}

从代码中可以看出,value 数组被 final 修饰,这意味着一旦数组被初始化,其引用就不能再指向其他数组。而 private 修饰符则保证了外部无法直接访问和修改这个数组。

2. 示例说明

下面通过一个简单的示例来直观感受 String 的不可变性:

java 复制代码
public class StringImmutabilityExample {
    public static void main(String[] args) {
        String str = "Hello";
        str = str + " World";
        System.out.println(str);
    }
}

就像下面这个图示一样:

在这个例子中,我们可能会认为 str 的内容从 "Hello" 变成了 "Hello World"。但实际上,str 最初指向的 "Hello" 对象并没有改变,当执行 str = str + " World" 时,Java 会创建一个新的 String 对象 "Hello World",然后让 str 引用这个新对象,而原来的 "Hello" 对象仍然存在于内存中。

二、String 不可变的原因

1. 安全性

String 在 Java 中广泛用于存储敏感信息,如用户名、密码、数据库连接信息等。如果 String 是可变的,那么这些敏感信息就可能被恶意修改,从而引发安全问题。例如,在多线程环境下,如果一个线程正在使用一个 String 对象存储的密码进行验证,而另一个线程同时修改了这个 String 对象的内容,那么验证结果就会变得不可靠。

2. 缓存哈希码

String 类重写了 hashCode() 方法,并且 String 对象的哈希码是在对象创建时就计算好并缓存起来的。因为 String 不可变,所以其哈希码不会改变,这样在使用 String 作为哈希表(如 HashMapHashSet)的键时,就可以避免重复计算哈希码,提高了性能。以下是 String 类中 hashCode() 方法的部分代码:

java 复制代码
public final class String
    implements java.io.Serializable, Comparable<String>, CharSequence {
    private int hash; // Default to 0

    public int hashCode() {
        int h = hash;
        if (h == 0 && value.length > 0) {
            char val[] = value;

            for (int i = 0; i < value.length; i++) {
                h = 31 * h + val[i];
            }
            hash = h;
        }
        return h;
    }

    // 其他代码...
}

可以看到,如果 String 可变,那么哈希码就可能会因为内容的改变而改变,这将破坏哈希表的正常工作。

3. 便于字符串常量池的实现

Java 中的字符串常量池是一种特殊的内存区域,用于存储字符串常量。当使用双引号声明一个字符串时,Java 会首先在字符串常量池中查找是否已经存在相同内容的字符串,如果存在,则直接返回该字符串的引用;如果不存在,则在常量池中创建一个新的字符串对象。String 的不可变性保证了常量池中的字符串可以被安全地共享,避免了因内容改变而导致的混乱。例如:

java 复制代码
String str1 = "Hello";
String str2 = "Hello";
System.out.println(str1 == str2); // 输出 true

在这个例子中,str1str2 都指向字符串常量池中的同一个 "Hello" 对象。

三、String 不可变带来的好处

1. 线程安全

由于 String 不可变,所以在多线程环境下,多个线程可以同时访问同一个 String 对象,而不需要担心数据被修改的问题。这使得 String 成为了线程安全的类,开发者可以放心地在多线程程序中使用 String 对象,无需额外的同步机制。

2. 性能优化

正如前面提到的,String 的不可变性使得哈希码可以被缓存,这在使用 String 作为哈希表的键时可以显著提高性能。此外,由于 String 对象可以在字符串常量池中共享,减少了内存的使用,也提高了垃圾回收的效率。

3. 代码可读性和可维护性

String 的不可变性使得代码更加易于理解和维护。因为开发者不需要担心 String 对象的内容会在程序运行过程中被意外修改,所以可以更专注于业务逻辑的实现。

四、利用反射改变 String 的字符数据

虽然我不能不能改变字符串,但是我们可以修改字符串的字符,Java 的反射机制允许我们在运行时检查和修改类的属性、方法等。通过反射,我们可以绕过 private 修饰符的限制,访问并修改 String 对象内部的 value 数组。

以下是一个示例代码:

java 复制代码
package org.example.a;

import java.lang.reflect.Field;

public class Demo {
    public static void main(String[] args) throws NoSuchFieldException, IllegalAccessException {
        String s = "1234";
        System.out.println("改变前:s=" + s);
        // 获取 String 类的 value 属性
        Field f = s.getClass().getDeclaredField("value");
        // 设置该属性可访问,绕过 private 限制
        f.setAccessible(true);
        // 修改 value 数组为新的字符数组
        f.set(s, new char[]{'a', 'b', 'c'});
        System.out.println("改变后:s=" + s);
    }
}

代码解释

  • Field f = s.getClass().getDeclaredField("value");:通过 getClass() 方法获取 s 对象的 Class 对象,然后使用 getDeclaredField("value") 方法获取 String 类中名为 value 的属性。
  • f.setAccessible(true);:将 value 属性的可访问性设置为 true,这样就可以绕过 private 修饰符的限制,对其进行访问和修改。
  • f.set(s, new char[]{'a', 'b', 'c'});:将 s 对象的 value 属性设置为新的字符数组 {'a', 'b', 'c'}

执行结果

复制代码
改变前:s=1234
改变后:s=abc

从执行结果可以看出,我们成功地通过反射修改了 String 对象的字符数据。

3. 这种做法的风险和注意事项

虽然通过反射可以改变 String 的字符数据,但这种做法并不推荐在实际开发中使用,原因如下:

  • 破坏不可变性原则String 的不可变性是 Java 语言设计的重要特性之一,许多系统和库都依赖于这一特性。使用反射修改 String 会破坏这种不可变性,可能导致程序出现难以调试的错误。
  • 安全问题 :如果恶意代码利用反射修改 String 对象,可能会导致安全漏洞,比如篡改敏感信息等。
  • 兼容性问题 :不同的 Java 版本或虚拟机实现可能对反射操作的支持有所不同,使用反射修改 String 可能会导致兼容性问题。
相关推荐
风象南32 分钟前
SpringBoot中6种自定义starter开发方法
java·spring boot·后端
mghio9 小时前
Dubbo 中的集群容错
java·微服务·dubbo
uhakadotcom11 小时前
视频直播与视频点播:基础知识与应用场景
后端·面试·架构
拉不动的猪11 小时前
刷刷题50(常见的js数据通信与渲染问题)
前端·javascript·面试
拉不动的猪11 小时前
JS多线程Webworks中的几种实战场景演示
前端·javascript·面试
uhakadotcom13 小时前
快速开始使用 n8n
后端·面试·github
uhakadotcom13 小时前
Astro 框架:快速构建内容驱动型网站的利器
前端·javascript·面试
uhakadotcom13 小时前
了解Nest.js和Next.js:如何选择合适的框架
前端·javascript·面试
uhakadotcom13 小时前
React与Next.js:基础知识及应用场景
前端·面试·github