java String的equals,intern方法

本文详细解析了Java中String对象的内部机制,包括equals和==的区别,String对象在内存中的创建过程,以及如何有效利用String的intern()方法减少内存消耗。通过实例演示了String对象的创建方式对内存分配的影响,并提供了优化String比较的方法。

JAVA中的equals和==的区别

 

==比较的是2个对象的地址,而equals比较的是2个对象的内容。

显然,当equals为true时,==不一定为true;

 

基础知识的重要性,希望引起大家的重视,包括自己在内

很多困惑和疑问而且均来自于最基础的知识

折腾了一阵子又查了查书,终于对 String 这个特殊的对象有了点感悟

public class TestString {

    public static void main(String[] args) {

        String s1 = "Monday";

        String s2 = "Monday";

    }

}

有什么问题呢?

1. 来自 String 的忧虑

上面这段程序中,到底有几个对象呢?

可能很多人脱口而出:两个,s1 和 s2

为什么?

String 是 final 类,它的值不可变。

看起来似乎很有道理,那么来检测一下吧,稍微改动一下程序

就可以看到结果了:

public class TestString {

    public static void main(String[] args) {

        String s1 = "Monday";

        String s2 = "Monday";

        if (s1 == s2)

            System.out.println("s1 == s2");

        else

            System.out.println("s1 != s2");

    }

}

呵呵,很多人都会说已经不止两个对象了

编译并运行程序,输出:s1 == s2

啊!

为什么 s1 == s2 ?

== 分明是在说:s1 与 s2 引用同一个 String 对象 -- "Monday"!

2. 千变万化的 String

再稍微改动一下程序,会有更奇怪的发现:

public class TestString {

    public static void main(String[] args) {

        String s1 = "Monday";

        String s2 = new String("Monday");

        if (s1 == s2)

            System.out.println("s1 == s2");

        else

            System.out.println("s1 != s2");

        if (s1.equals(s2))

            System.out.println("s1 equals s2");

        else

            System.out.println("s1 not equals s2");

    }

}

我们将 s2 用 new 操作符创建

程序输出:

s1 != s2

s1 equals s2

嗯,很明显嘛

s1 s2分别引用了两个"Monday"String对象

可是为什么两段程序不一样呢?

3. 在 String 的游泳池中游泳

哈哈,翻了翻书终于找到了答案:

原来,程序在运行的时候会创建一个字符串缓冲池

当使用 s2 = "Monday" 这样的表达是创建字符串的时候,程序首先会

在这个String缓冲池中寻找相同值的对象,在第一个程序中,s1先被

放到了池中,所以在s2被创建的时候,程序找到了具有相同值的 s1

将 s2 引用 s1 所引用的对象"Monday"

第二段程序中,使用了 new 操作符,他明白的告诉程序:

“我要一个新的!不要旧的!”与是一个新的"Monday"Sting对象被创

建在内存中。他们的值相同,但是位置不同,一个在池中游泳

一个在岸边休息。哎呀,真是资源浪费,明明是一样的非要分开做什么呢?

4. 继续潜水

再次更改程序:

public class TestString {

    public static void main(String[] args) {

        String s1 = "Monday";

        String s2 = new String("Monday");

        s2 = s2.intern();

        if (s1 == s2)

            System.out.println("s1 == s2");

        else

            System.out.println("s1 != s2");

        if (s1.equals(s2))

            System.out.println("s1 equals s2");

        else

            System.out.println("s1 not equals s2");

    }

}

这次加入:s2 = s2.intern();

哇!程序输出:

s1 == s2

s1 equals s2

原来,程序新建了 s2 之后,又用intern()把他打翻在了池里

哈哈,这次 s2 和 s1 有引用了同样的对象了

我们成功的减少了内存的占用

5. == 与 equals() 的争斗

String 是个对象,要对比两个不同的String对象的值是否相同

明显的要用到 equals() 这个方法

可是如果程序里面有那么多的String对象,有那么多次的要用到 equals ,

哦,天哪,真慢啊

更好的办法:

把所有的String都intern()到缓冲池去吧

最好在用到new的时候就进行这个操作

String s2 = new String("Monday").intern();

嗯,大家都在水池里泡着了吗?哈哈

现在我可以无所顾忌的用 == 来比较 String 对象的值了

真是爽啊,又快又方便!

### Java 中 `String` 的 `intern()` 方法工作原理 在 Java 中,`String.intern()` 是一种机制,用于将字符串实例存储到 JVM 的 **字符串常量池** (String Constant Pool) 中。如果该字符串已经存在于常量池中,则会返回其对应的引用;否则,当前字符串会被添加至常量池,并返回新加入的引用[^1]。 #### 原理分析 当调用 `s.intern()` 时,JVM 首先会在字符串常量池中查找是否存在与 `s` 内容相同的字符串。如果有匹配项,则直接返回已有字符串的引用;如果没有找到,则将此字符串复制一份存入字符串常量池并返回新的引用[^3]。 以下是实现这一功能的核心逻辑: ```java // 示例代码展示 intern() 方法的行为 public class InternExample { public static void main(String[] args) { String s1 = new String("hello").intern(); // 调用了 intern() String s2 = "hello"; // 直接从常量池加载 System.out.println(s1 == s2); // 输出 true,因为两者指向同一对象 } } ``` 上述代码片段展示了如何通过 `intern()` 将堆上的字符串移入字符串常量池,并验证两个变量是否共享同一个引用[^4]。 --- ### 使用场景 `String.intern()` 主要适用于以下几种情况: 1. **节省内存空间** 当程序中有大量重复的字符串数据时,可以利用 `intern()` 来减少冗余的对象创建,从而降低内存消耗[^2]。 2. **提高比较效率** 如果频繁执行基于内容的字符串比较操作(如使用 `.equals()`),可以通过 `intern()` 和身份比较 (`==`) 提升性能,因为后者通常比前者更快。 3. **处理固定集合的数据** 对于一些固定的字典或者枚举类型的字符串列表,提前将其放入字符串常量池有助于后续快速检索和判断。 --- ### 注意事项 尽管 `String.intern()` 可以为特定应用场景带来好处,但也需要注意潜在的风险或局限性: - 过度依赖可能导致不必要的复杂性和维护成本; - 在某些版本的 JDK 下可能会引发垃圾回收压力增加等问题。 因此,在实际开发过程中应权衡利弊后再决定是否采用这种方法来管理字符串资源。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值