浅析java中String类型中“==”与“equal”的区别

一、前言

1.1、首先很多人都知道,String中用“==”比较的是地址,用equals比较的是内容,很多人对此用的是记忆法,通过记忆来加强此的引用,但是其真正的原理其实并不难,当我们真正明白其为什么的时候,用起来也会更加灵活,更加有底气(形容得不太好,朋友别见怪);

二相关知识的准备

  • 类型常量池
  • 运行时常量池
  • 字符串常量池

我们今天讨论的主题是当然是字符串常量池:
为什么在这要把另外两个常量池拿出说一下呢,首先小生我在网上或者cnds上看到很多人在争论字符串常量池是存在与方法区还是堆里面,因此我在这里面非常负责任的告诉各位读者:
1、类型常量池,存放在方法区里面,每个class文件都有一个
2、运行时常量池、存放在方法区里面,所有class共用
3、字符串常量池:存放在堆区里面
而且字符串常量池有一个特点:存放的常量唯一:三、 开始

3.1,情况一

public class Test2 {
	public static void main(String[] args) {

		String a ="张敬轩";
		String b ="张敬轩";

		System.out.println(a == b);
		System.out.println(a.equals(b));

	}
}

运行结果:

  • true
  • true

解释:因为当我们String a = “张敬轩”;的时候,会在堆里面的字符串常量池里面“搜索”是否有“张敬轩”这个对象,

  • 有:就会将“张敬轩这个对象的地址指向a”
  • 没有:就会在字符串常量池里面新建立一个“张敬轩”,然后就会把地址引用赋值给a

当再声明String b = “张敬轩";和第一次声明String = “张敬轩”;一样,先”搜索“,然后因为已经存在了”张敬轩“这个对象,那么就不会再创建对象,而是将存在的”张敬轩“的地址引用赋值给b,所以
a和b的地址一样,内容自然也一样,所以两个结果为true,不懂各位朋友懂没懂?

3.2,情况二;

public class Test2 {
	public static void main(String[] args) {

		String a ="张敬轩";

		String b = new String("张敬轩");

		System.out.println(a == b);
		System.out.println(a.equals(b));

	}
}

结果是

false,true
解释
String a = “张敬轩”;
已经解释过了,是在堆里面的字符串常量池创建对象
String b = new String(“张敬轩”);
这个也是在堆区里面创建对象,但是不是在字符串常量池里面创建对象
两个不同的对象,地址自然不同,类容都是张敬轩,所以结果是false,true

3.3、情况三

public class Test2 {
	public static void main(String[] args) {

		String a = new String("张敬轩");

		String b = new String("张敬轩");

		System.out.println(a == b);
		System.out.println(a.equals(b));

	}

}

结果:false,true

解释,因为new (“张敬轩”);表示在堆区的非字符串常量池里面创建了两个不同的对象,两次new就创建了两个,所以地址是不同的,因此结果是false,true

3.4、情况四

public class Test2 {
	public static void main(String[] args) {

		String a = "张"+"敬轩";
		String b = "张敬轩";
		System.out.println(a == b);
		System.out.println(a.equals(b));

	}
}

结果是:true,true

解释:String = “张敬” + ”轩“;
在编译的时候就已经开始进行计算:这是Java编译的优化机制,所以a指向的地址依旧是”张敬轩“;所以情况和前面的情况一一致,所以是true,true
(注意Java编译优化机制只针对常量不针对变量 a = a+ b(这个是没有优化机制的,关于Java编译机制,各位小伙伴可以去百度了解一下))

3.5、情况五

public class Test2 {
	public static void main(String[] args) {

		String a = "张";
		String b = "敬轩";
		String c =a+b;
		String d = "张敬轩";
		System.out.println(d == c);
		System.out.println(d.equals(c));

	}
}

结果:

解释:因为a 和 b是变量,不存在什么Java优化机制,而是将两个c = a + b存放在的是堆区的非字符常量池里面,所以是两个不同的对象,自然为false,true
3.6、最后一种情况

public class Test2 {
	public static void main(String[] args) {

		String a = "张";
		String b = "敬轩";
		String c =(a+b).intern();
		String d = "张敬轩";
		System.out.println(d == c);
		System.out.println(d.equals(c));

	}

}

结果:

解释:intern();方法是将堆区里面的非字符常量池里面的对象强行放到字符常量池里面,因为字符常量池里面对象的唯一性,如果字符串常量池里面已经有了和”张敬轩“一样的对象,就会将其地址引用赋值给c,没有就相当于创建一个(也就是和之前堆区里面那个对象一样的对象),有了的话,就会将原来有的那个“张敬轩”对象赋值给c,因为地址一样,所以为true

总结:

== 比较的地址 (如果对象存在字符串常量池,而且类容相同,那么 == 返回的是true )
equal 比较的是内容 (内容相同返回便是true)(所以在以后的开发中一般是用equal比较字符串的

各位朋友如果有错的及时评论区指出,谢谢大家的支持

到此这篇关于浅析java中String类型中“==”与“equal”的区别的文章就介绍到这了,更多相关java String类型 ==与equal内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

时间: 2020-08-21

java中equals和等号(==)的区别浅谈

java中的数据类型,可分为两类:1.基本数据类型,也称原始数据类型.byte,short,char,int,long,float,double,boolean   他们之间的比较,应用双等号(==),比较的是他们的值. 2.复合数据类型(类)   当他们用(==)进行比较的时候,比较的是他们在内存中的存放地址,所以,除非是同一个new出来的对象,他们的比较后的结果为true,否则比较后结果为false.JAVA当中所有的类都是继承于Object这个基类的,在Object中的基类中定义了一个eq

基于Java字符串 "==" 与 "equals" 的深入理解

虽然关于Java字符串"=="和"equals"的问题是Java学习中的最基础的一个问题,但是仍然有不少刚刚学习Java的同学搞不清楚里面的原理,最近和朋友讨论到这个问题,所以写篇文章给大家分享一下我自己的理解.首先让我们看一个例子: 复制代码 代码如下: public class TestStringEquals { public static void main(String[] args) { String a = "test";     

Java中char数组(字符数组)与字符串String类型的转换方法

本文实例讲述了Java中char数组(字符数组)与字符串String类型的转换方法.分享给大家供大家参考,具体如下: 在Java语言编程时,使用"口令字段"jPasswordField组件时,如果要获得密码值,就需要使用该组件的getPassword()方法.jPasswordField的getPassword()方法返回一个char类型的数组,我们经常需要将这个数组转换为String类型,以便进行诸如口令匹配或口令赋值等操作.这时,就需要将char类型的数组进行转换.当然也经常会遇到

JAVA中string数据类型转换详解

在JAVA中string是final类,提供字符串不可以修改,string类型在项目中经常使用,下面给大家介绍比较常用的string数据类型转换: String数据类型转换成long.int.double.float.boolean.char等七种数据类型 复制代码 代码如下: * 数据类型转换 * @author Administrator * */ public class 数据类型转换 { public static void main(String[] args) { String c=

java String 类的一些理解 关于==、equals、null

1.String 的==与equal() 在对字符串的相等判断,==判断的是地址是否相同,equal()判断的是字符值是否相同.大多数时候==跟equal()的结果都是相同的.这是因为String对象是不变模式的,如果你不是明确地new一个String对象,Java对于String对象的保存默认的是会把新生成的String 对象放到一个缓冲区,然后每次判断缓冲区中是否已经有了这个对象,如果有了,那么后建立的同样字符值的String对象也会指向最初建立是该字符值对象的地址.也就是说字符值相同的时候

Java中==符号与equals()的使用详解(测试两个变量是否相等)

Java 程序中测试两个变量是否相等有两种方式:一种是利用 == 运算符,另一种是利用equals()方法. 当使用 == 来判断两个变量是否相等时,如果两个变量是基本类型变量,且都是数值类型(不一定要求数据类型严格相同),则只要两个变量的值相等,就返回true. 但是对于两个引用类型变量,只有它们指向同一个对象时, == 判断才会返回true. == 不可用于比较类型上没有父子关系的两个对象. 很多书上说equals()方法是判断两个对象的值相等.这种说法不准确.实际上equals()方法是O

java对象转换String类型的三种方法

一.采用Object.toString()toString方法是java.lang.Object对象的一个public方法.在java中任何对象都会继承Object对象,所以一般来说任何对象都可以调用toString这个方法.这是采用该种方法时,常派生类会覆盖Object里的toString()方法.但是在使用该方法时要注意,必须保证Object不是null值,否则将抛出NullPointerException异常. 二.采用(String)Object 该方法是一个标准的类型转换的方法,可以将

简单谈谈Java中String类型的参数传递问题

提要:本文从实现原理的角度上阐述和剖析了:在Java语言中,以 String 作为类型的变量在作为方法参数时所表现出的"非对象"的特性. 一.最开始的示例 写代码最重要的就是实践,不经过反复试验而得出的说辞只能说是凭空遐想罢了.所以,在本文中首先以一个简单示例来抛出核心话题: public class StringAsParamOfMethodDemo { public static void main(String[] args) { StringAsParamOfMethodDem

简单谈谈Java中的方法和方法重载

今天我们就讲一点内容,来说说Java中的方法和方法重载以及需要注意的一些地方: 方法: Java的方法类似与其他语言的函数,是一段用来完成特定功能的代码片段, 声明格式: [修饰符1 修饰符2 ....] ,返回值类型 方法名 (形式参数列表) { Java语句: - - -} 形式参数:在方法被调用时用于接受外界输入的数据: 实参: 调用方法时世界传给方法的数据: 返回值: 方法在执行完毕后返回给调用他的环境的数据: 返回值类型: 事先约定好的返回值的数据类型,如无返回值必须给出返回值类型vo

详解Java中String类型与默认字符编码

为什么写这个 至于为什么要写这个,主要是一句mmp一定要讲,绕了一上午,晕死 Java程序中的中文乱码问题一直是一个困扰程序员的难题,自己也不例外,早在做项目时就遇到过很多编码方式的坑,当时想填来着,但是嫌麻烦.这次终于忍不住了,一定要弄个明白 String类型的编码方式 从网上查的资料都说,Java默认的字符编码是Unicode,而String类型的编码方式是与JVM编码方式和本机操作系统默认字符集有关的.于是我做出了测试 在Java中可以这样显示查看本地编码方式(JVM还是OS呢?) //

谈谈Java中整数类型(short int long)的存储方式

在java中的整数类型有四种,分别是 byte  short int long 其中byte只有一个字节 0或1,在此不详细讲解. 其他的三种类型如下: 1. 基本类型:short 二进制位数:16 包装类:java.lang.Short 最小值:Short.MIN_VALUE=-32768 (-2的15此方) 最大值:Short.MAX_VALUE=32767 (2的15次方-1) 2. 基本类型:int 二进制位数:32 包装类:java.lang.Integer 最小值:Integer.M

简单谈谈java中final,finally,finalize的区别

(1) final:修饰符(关键字),如果一个类被声明为final,意味着它不能再派生出新的子类,不能作为父类被继承.因此一个类不能既被声明为 abstract的,又被声明为final的.将变量或方法声明为final,可以保证它们在使用中不被改变.被声明为final的变量必须在声明时给定初值,而在以后的引用中只能读取,不可修改.被声明为final的方法也同样只能使用,不能重载 (2) finally:在异常处理时提供 finally 块来执行任何清除操作.如果抛出一个异常,那么相匹配的 catc

简单谈谈Java 中的线程的几种状态

Java 中的线程有以下状态: 新建状态(New):新创建的线程,还未执行. 就绪状态(Runnable):执行了 start() 方法,等待运行, 运行状态(Running):就绪状态的线程开始执行程序代码. 阻塞状态(Blocked) 同步堵塞:在运行过程中,需要拿到锁才能运行,而锁被其他资源占用,需要等待. 等待堵塞:执行了 wait() 方法,进入了等待. 其他堵塞:执行了 join().sleep() 方法,进入了等待. 终止状态(Terminated):运行完 run() 方法后结束

简单谈谈java中匿名内部类构造函数

先看看下面的代码能不能编译通过: public static void main(String[] args) { List l1 = new ArrayList(); List l2 = new ArrayList(){}; List l3 = new ArrayList(){{}}; System.out.println(l1.getClass() == l2.getClass() ); System.out.println(l2.getClass() == l3.getClass() );

简单谈谈java自定义注解

Java在1.5开始引入了注解,目前流行的框架都在用注解,可想而知注解的强大之处. 以下通过自定义注解来深入了解java注解. 一.创建自定义注解 package com.sam.annotation; import java.lang.annotation.*; /** * @author sam * @since 2017/7/13 */ @Target({ElementType.METHOD, ElementType.FIELD}) @Retention(RetentionPolicy.R

谈谈Java中自定义注解及使用场景

Java自定义注解一般使用场景为:自定义注解+拦截器或者AOP,使用自定义注解来自己设计框架,使得代码看起来非常优雅.本文将先从自定义注解的基础概念说起,然后开始实战,写小段代码实现自定义注解+拦截器,自定义注解+AOP. 一. 什么是注解(Annotation) Java注解是什么,以下是引用自维基百科的内容 Java注解又称Java标注,是JDK5.0版本开始支持加入源代码的特殊语法元数据. Java语言中的类.方法.变量.参数和包等都可以被标注.和Javadoc不同,Java标注可以通过反