Day11-Java中String的equals方法如何实现

简介: 笔记

在我们面试的时候经常会看到== 与 equals()的区别,我们在网上背一些这种题目的答案:


==:


如果是基本数据类型的比较,是值的比较

如果是引用类型的比较,⽐较的是两个引⽤是否指向相同的对象,也就是比较内存地址是否相同

equals():


equals 是比较内存地址上面的值是否相同

往往我们还是需要从源码的角度分析一下Java中String的equals方法如何实现。


首先我们都知道Java中所有的类都继承于Object这个类,在Object类中定义了一个equals的方法,equals的源码是这样写的:

15.png

可以看到,这个方法的初始默认行为是比较对象的内存地址值,一般来说,意义不大。所以在一些类库中被重写了(String、Integer等),在这些类当中equals有其自身的实现(一般都是用来比较对象的成员变量值是否相同),而不再是比较类在堆内存中的存放地址了。


因此,对于复合数据类型之间进行equals比较,在没有覆写equals方法的情况下,他们之间的比较还是内存中的存放位置的地址值,跟双等号(==)的结果相同;如果被复写,按照复写的要求来。


下面是关于String中equals() 方法的源码:

16.png

/**
 * String.equals()源码解析
 */
public boolean equals(Object anObject) {
    /**
     * 第一步:判断传入对象是否与当前对象的地址相同 this == anObject, 如果对象的地址相同, 可知对象相等, 返回 true
     * 看到下面这里行, 我们可以说 equals() 就是基于 == 来实现的
     */
    if (this == anObject) {
        return true;
    }
    /**
     * 第二步: 判断传递的参数 anObject 是否是 String, 如果不是直接返回 false
     */
    if (anObject instanceof String) {
        /**
         * 第三步: 如果 anObject 是 String 类型, 就强制转化为字符串, 然后判断当前对象与传递的对象的长度时候相同
         * 不相同则证明不等于, 返回 false
         */
        String anotherString = (String)anObject;
        int n = value.length;
        if (n == anotherString.value.length) {
            /**
             *  第四步: 将两个对象转换为 char 数组,使用 while 循环逐一进行比较
             *  只有当 v1[i] == v2[i] 全部等于 true 时, 返回 true
             */
            char v1[] = value;
            char v2[] = anotherString.value;
            int i = 0;
            while (n-- != 0) {
                if (v1[i] != v2[i])
                    return false;
                i++;
            }
            return true;
        }
    }
    return false;
}

总结:


判断传入对象是否与当前对象的地址相同,如果对象的地址相同, 可知对象相等, 返回 true

判断传递的参数 anObject 是否是 String, 如果不是直接返回 false

第三步: 如果 anObject 是 String 类型, 就强制转化为字符串, 然后判断当前对象与传递的对象的长度时候相同,不相同则证明不等于, 返回 false

将两个对象转换为 char 数组,使用 while 循环逐一进行比较,只有当 v1[i] == v2[i] 全部等于 true 时, 返回 true

一句话:先比较地址,再比较内容,满足其中一个就可以返回true


测试代码

/**
 * @author :caizhengjie
 * @description :
 * @date :2021/7/25 20:09
 */
public class TestEquals {
    public static void main(String[] args) {
        // 字符串常量
        String s9 = "Hello";
        String s10 = "Hello";
        // 使用new关键字创建
        String s7 = new String("Hello");
        String s8 = new String("Hello");
        System.out.println(s7 == s9); // false
        System.out.println(s9 == s10); // true
        System.out.println(s7 == s8); // false
        System.out.println("--------------");
        System.out.println(s7.equals(s9)); // true
        System.out.println(s7.equals(s8)); // true
    }
}

文字解析:


从上面的运行结果可见,s7和s8指的是不同对象, s9和s10指向的是相同对象。

Java中的不可变字符串String常量,采用字符串池(String Pool)管理技术,字符串池是 一种字符串驻留技术。采用字符串常量赋值时会在字符串池中查 找"Hello"字符串常量,如果已经存在把引用赋值给s9,否则创建"Hello"字符串对象,并放到池中。根 据此原理,可以推定s10与s9是相同的引用,指向同一个对象。但此原理并不适用于new所创建的字符 串对象,它并没有放到字符串池中。s7和s8是不同的引用,指向不同的对象。


图解:20.png


相关文章
|
8月前
|
Java
Java语言实现字母大小写转换的方法
Java提供了多种灵活的方法来处理字符串中的字母大小写转换。根据具体需求,可以选择适合的方法来实现。在大多数情况下,使用 String类或 Character类的方法已经足够。但是,在需要更复杂的逻辑或处理非常规字符集时,可以通过字符流或手动遍历字符串来实现更精细的控制。
500 18
|
9月前
|
算法 安全 Java
除了类,Java中的接口和方法也可以使用泛型吗?
除了类,Java中的接口和方法也可以使用泛型吗?
262 11
|
8月前
|
Java 编译器 Go
【Java】(5)方法的概念、方法的调用、方法重载、构造方法的创建
Java方法是语句的集合,它们在一起执行一个功能。方法是解决一类问题的步骤的有序组合方法包含于类或对象中方法在程序中被创建,在其他地方被引用方法的优点使程序变得更简短而清晰。有利于程序维护。可以提高程序开发的效率。提高了代码的重用性。方法的名字的第一个单词应以小写字母作为开头,后面的单词则用大写字母开头写,不使用连接符。例如:addPerson。这种就属于驼峰写法下划线可能出现在 JUnit 测试方法名称中用以分隔名称的逻辑组件。
347 4
|
8月前
|
编解码 Java 开发者
Java String类的关键方法总结
以上总结了Java `String` 类最常见和重要功能性方法。每种操作都对应着日常编程任务,并且理解每种操作如何影响及处理 `Strings` 对于任何使用 Java 的开发者来说都至关重要。
468 5
|
9月前
|
Java 开发者
Java 函数式编程全解析:静态方法引用、实例方法引用、特定类型方法引用与构造器引用实战教程
本文介绍Java 8函数式编程中的四种方法引用:静态、实例、特定类型及构造器引用,通过简洁示例演示其用法,帮助开发者提升代码可读性与简洁性。
|
10月前
|
存储 SQL 缓存
Java字符串处理:String、StringBuilder与StringBuffer
本文深入解析Java中String、StringBuilder和StringBuffer的核心区别与使用场景。涵盖字符串不可变性、常量池、intern方法、可变字符串构建器的扩容机制及线程安全实现。通过性能测试对比三者差异,并提供最佳实践与高频面试问题解析,助你掌握Java字符串处理精髓。
|
10月前
|
算法 Java
Java语言实现链表反转的方法
这种反转方法不需要使用额外的存储空间,因此空间复杂度为,它只需要遍历一次链表,所以时间复杂度为,其中为链表的长度。这使得这种反转链表的方法既高效又实用。
664 0
|
10月前
|
存储 Java 数据处理
Java映射操作:深入Map.getOrDefault与MapUtils方法
结合 `getOrDefault`方法的简洁性及 `MapUtils`的丰富功能,Java的映射操作变得既灵活又高效。合理地使用这些工具能够显著提高数据处理的速度和质量。开发人员可以根据具体的应用场景选择适宜的方法,以求在性能和可读性之间找到最佳平衡。
337 0
|
10月前
|
缓存 人工智能 NoSQL
Java中实现Token设置过期时间的方法
本文介绍了在Java应用中实现Token设置过期时间的多种方法,包括使用JWT和Redis缓存,并结合定时任务清理过期Token,以提升系统安全性与用户隐私保护。
1128 0
|
存储 编译器 C语言
关于string的‘\0‘与string,vector构造特点,反迭代器与迭代器类等的讨论
你真的了解string的'\0'么?你知道创建一个string a("abcddddddddddddddddddddddddd", 16);这样的string对象要创建多少个对象么?你知道string与vector进行扩容时进行了怎么的操作么?你知道怎么求Vector 最大 最小值 索引 位置么?
287 0