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


相关文章
|
18天前
|
存储 Java 程序员
Java基础的灵魂——Object类方法详解(社招面试不踩坑)
本文介绍了Java中`Object`类的几个重要方法,包括`toString`、`equals`、`hashCode`、`finalize`、`clone`、`getClass`、`notify`和`wait`。这些方法是面试中的常考点,掌握它们有助于理解Java对象的行为和实现多线程编程。作者通过具体示例和应用场景,详细解析了每个方法的作用和重写技巧,帮助读者更好地应对面试和技术开发。
67 4
|
29天前
|
Java API
Java 对象释放与 finalize 方法
关于 Java 对象释放的疑惑解答,以及 finalize 方法的相关知识。
48 17
|
23天前
|
Java 测试技术 Maven
Java一分钟之-PowerMock:静态方法与私有方法测试
通过本文的详细介绍,您可以使用PowerMock轻松地测试Java代码中的静态方法和私有方法。PowerMock通过扩展Mockito,提供了强大的功能,帮助开发者在复杂的测试场景中保持高效和准确的单元测试。希望本文对您的Java单元测试有所帮助。
63 2
|
30天前
|
Java 开发者
在Java多线程编程中,创建线程的方法有两种:继承Thread类和实现Runnable接口
【10月更文挑战第20天】在Java多线程编程中,创建线程的方法有两种:继承Thread类和实现Runnable接口。本文揭示了这两种方式的微妙差异和潜在陷阱,帮助你更好地理解和选择适合项目需求的线程创建方式。
20 3
|
30天前
|
Java 开发者
在Java多线程编程中,选择合适的线程创建方法至关重要
【10月更文挑战第20天】在Java多线程编程中,选择合适的线程创建方法至关重要。本文通过案例分析,探讨了继承Thread类和实现Runnable接口两种方法的优缺点及适用场景,帮助开发者做出明智的选择。
19 2
|
30天前
|
安全 Java
Java多线程通信新解:本文通过生产者-消费者模型案例,深入解析wait()、notify()、notifyAll()方法的实用技巧
【10月更文挑战第20天】Java多线程通信新解:本文通过生产者-消费者模型案例,深入解析wait()、notify()、notifyAll()方法的实用技巧,包括避免在循环外调用wait()、优先使用notifyAll()、确保线程安全及处理InterruptedException等,帮助读者更好地掌握这些方法的应用。
19 1
|
30天前
|
Java 开发者
Java多线程初学者指南:介绍通过继承Thread类与实现Runnable接口两种方式创建线程的方法及其优缺点
【10月更文挑战第20天】Java多线程初学者指南:介绍通过继承Thread类与实现Runnable接口两种方式创建线程的方法及其优缺点,重点解析为何实现Runnable接口更具灵活性、资源共享及易于管理的优势。
34 1
|
30天前
|
Java
在Java多线程编程中,`wait()`和`notify()`方法的相遇如同一场奇妙的邂逅
在Java多线程编程中,`wait()`和`notify()`方法的相遇如同一场奇妙的邂逅。它们用于线程间通信,使线程能够协作完成任务。通过这些方法,生产者和消费者线程可以高效地管理共享资源,确保程序的有序运行。正确使用这些方法需要遵循同步规则,避免虚假唤醒等问题。示例代码展示了如何在生产者-消费者模型中使用`wait()`和`notify()`。
28 1
|
20天前
|
JavaScript 前端开发 开发者
|
24天前
|
Java Spring
JAVA获取重定向地址URL的两种方法
【10月更文挑战第17天】本文介绍了两种在Java中获取HTTP响应头中的Location字段的方法:一种是使用HttpURLConnection,另一种是使用Spring的RestTemplate。通过设置连接超时和禁用自动重定向,确保请求按预期执行。此外,还提供了一个自定义的`NoRedirectSimpleClientHttpRequestFactory`类,用于禁用RestTemplate的自动重定向功能。

热门文章

最新文章

下一篇
无影云桌面