【Java】深层次解析字符串比较

本文涉及的产品
云解析 DNS,旗舰版 1个月
全局流量管理 GTM,标准版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
简介: 【Java】深层次解析字符串比较

1. 字符串创建

在我们的JVM中,有常量池-存放字符串、堆-存放new的字符串、栈-存放引用地址

我们看一下创建字符串的两种方式:

String s1 = "china";
        String s2 = "china";
        String s3 = "china";
        String ss1 = new String("china");
        String ss2 = new String("china");
        String ss3 = new String("china");
  • 对于第一种创建:在我们的常量池中创建出一个"china"的字符串,在栈中将s1、s2、s3指向其常量池
  • 对于第二种创建:首先去我们的常量池中寻找有没有"china"字符串,没有则创建,再去我们的堆进行对象的创建,在栈中将ss1、ss2、ss3指向堆中

2. 字符串比较

2.1 普通比较

String str1 = "a";
    String str2 = "b";
    String str3 = "ab";
    String str4 = str1 + str2;
    String str5 = new String("ab");
    String str6 = "a" + "b";
    System.out.println(str5.equals(str3));
    System.out.println(str5 == str3);
    System.out.println(str4 == str3);
    System.out.println(str5.intern() == str3);
    System.out.println(str5.intern() == str4);
    System.out.println(str5.intern() == str6);
    System.out.println(str3 == str6);

先想想自己的答案:

依次分析:

首先看下创建的过程

  1. String str1 = "a"; 直接常量池创建"a",栈创建str1指向
  2. 同上
  3. 同上
  4. String str4 = str1 + str2;,因为我们的JVM没有办法将str1和str2自动识别成字符串,所以会将他们new 一个放置在堆内存中
  5. String str5 = new String("ab");直接创建即可
  6. String str6 = "a" + "b";我们的JVM先将"a" + “b"变成"ab”,再去赋值给str6

比较过程:

前面的不说了,重点说一下关于:System.out.println(str5.intern() == str3);


官方API:

intern

public String intern()

返回字符串对象的规范化表示形式。

一个初始时为空的字符串池,它由类 String 私有地维护。

当调用 intern 方法时,如果池已经包含一个等于此 String 对象的字符串(该对象由 equals(Object) 方法确定),则返回池中的字符串。否则,将此 String 对象添加到池中,并且返回此 String 对象的引用。

它遵循对于任何两个字符串 s 和 t,当且仅当 s.equals(t) 为 true 时,s.intern() == t.intern() 才为 true。

所有字面值字符串和字符串赋值常量表达式都是内部的。

返回:一个字符串,内容与此字符串相同,但它保证来自字符串池中。


上面的理解比较官方,简单来说就是,我调用了这个函数,你给我返回一个存在于常量池中的字符串,要是没有,你给我加一个

比如:str5.intern()就是调用的常量池中的ab字符串,也就是只要指向ab字符串的,就可以判定为true

System.out.println(str5.equals(str3));
    System.out.println(str5 == str3);
    System.out.println(str4 == str3);
    System.out.println(str5.intern() == str3);
    System.out.println(str5.intern() == str4);
    System.out.println(str5.intern() == str6);
    System.out.println(str3 == str6);

答案分别是:

true

false

false

true

false

true

true

2.2 final比较

看完上面的,是不是感觉自己超级牛逼,感觉啥都懂了

别慌 看这个

final String str1 = "a";
    final String str2 = "b";
    String str3 = "ab";
    String str4 = str1 + str2;
    System.out.println(str4 == str3);

按照咱们上面说的,这个不细心的可能直接就false了

细心的看到对于str1和str2被final修饰了

回想一下,final修饰变量的作用,不可变性对吧。

如果被final修饰后,JNM在使用的时候,会把他当成一个字符串来使用,也就是类似于"a" + "b"这种操作,所以,最后的值是true

3. 总结

画图---->记住intern引用---->final变量

字符串比较不过如此~简简单单


相关文章
|
25天前
|
SQL Java 索引
java小工具util系列2:字符串工具
java小工具util系列2:字符串工具
137 83
|
22天前
|
Java 编译器
Java 泛型详细解析
本文将带你详细解析 Java 泛型,了解泛型的原理、常见的使用方法以及泛型的局限性,让你对泛型有更深入的了解。
34 2
Java 泛型详细解析
|
25天前
|
Java 数据库
java小工具util系列1:日期和字符串转换工具
java小工具util系列1:日期和字符串转换工具
53 26
|
23天前
|
缓存 监控 Java
Java线程池提交任务流程底层源码与源码解析
【11月更文挑战第30天】嘿,各位技术爱好者们,今天咱们来聊聊Java线程池提交任务的底层源码与源码解析。作为一个资深的Java开发者,我相信你一定对线程池并不陌生。线程池作为并发编程中的一大利器,其重要性不言而喻。今天,我将以对话的方式,带你一步步深入线程池的奥秘,从概述到功能点,再到背景和业务点,最后到底层原理和示例,让你对线程池有一个全新的认识。
51 12
|
20天前
|
存储 算法 Java
Java内存管理深度解析####
本文深入探讨了Java虚拟机(JVM)中的内存分配与垃圾回收机制,揭示了其高效管理内存的奥秘。文章首先概述了JVM内存模型,随后详细阐述了堆、栈、方法区等关键区域的作用及管理策略。在垃圾回收部分,重点介绍了标记-清除、复制算法、标记-整理等多种回收算法的工作原理及其适用场景,并通过实际案例分析了不同GC策略对应用性能的影响。对于开发者而言,理解这些原理有助于编写出更加高效、稳定的Java应用程序。 ####
|
20天前
|
存储 监控 算法
Java虚拟机(JVM)垃圾回收机制深度解析与优化策略####
本文旨在深入探讨Java虚拟机(JVM)的垃圾回收机制,揭示其工作原理、常见算法及参数调优方法。通过剖析垃圾回收的生命周期、内存区域划分以及GC日志分析,为开发者提供一套实用的JVM垃圾回收优化指南,助力提升Java应用的性能与稳定性。 ####
|
22天前
|
Java 数据库连接 开发者
Java中的异常处理机制:深入解析与最佳实践####
本文旨在为Java开发者提供一份关于异常处理机制的全面指南,从基础概念到高级技巧,涵盖try-catch结构、自定义异常、异常链分析以及最佳实践策略。不同于传统的摘要概述,本文将以一个实际项目案例为线索,逐步揭示如何高效地管理运行时错误,提升代码的健壮性和可维护性。通过对比常见误区与优化方案,读者将获得编写更加健壮Java应用程序的实用知识。 --- ####
|
25天前
|
存储 缓存 监控
Java中的线程池深度解析####
本文深入探讨了Java并发编程中的核心组件——线程池,从其基本概念、工作原理、核心参数解析到应用场景与最佳实践,全方位剖析了线程池在提升应用性能、资源管理和任务调度方面的重要作用。通过实例演示和性能对比,揭示合理配置线程池对于构建高效Java应用的关键意义。 ####
|
1月前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
73 2
|
2月前
|
缓存 Java 程序员
Map - LinkedHashSet&Map源码解析
Map - LinkedHashSet&Map源码解析
78 0

推荐镜像

更多