关于String 的 == equals 的区别

Java中==与equals的区别
本文详细解析了Java中==与equals的区别,包括基本类型的比较、引用类型比较以及String类的特殊情况。通过实例展示了不同场景下两者的工作原理。

一.  == 与equals 的本质区别

   1.  == 是指比较  两个比较对象 在内存中放的值(一般都是在方法中进行逻辑处理,所以默认是放在栈中)。

   如: int a = 1 ; int b =1;  因为int 是原始类型(byte , short, int , long,char , float,double, boolean等),所以 a 存放的 是 1 , b存放的 也是1,在进行 a== b 操作时,返回的是true。因为 原始类型没有实现 equals方法(或者可以理解为 原始类型不是对象,没有这样的方法),所以不能 进行 如 a.equals(b) 这样的操作。

 

    又如: Integer o1 = new Integer(2);  Integer o2 = new Integer(2);  因为 o1 和  o2 在这儿指向的是对象,所以在进行  o1==o2  操作时,比较的是 栈中存放的值(地址), 他们分别指向不同的对象(放在堆中),所以执行结果是false。

    注: 这儿进行 System.out.println(o1); 时,得到的是数值 2,而不是o1存储的内容,是因为Integer  这个对象对toString()方法进行了重写,返回的是 o1的值。

   2.  equals  是对象的祖先Object类中的一个实现了的方法,他的源码

     public boolean equals(Object paramObject)
     {
       return this == paramObject;
    }

    所以 也是 进行 == 比较的。即比较内存中的内容。

 

二. 关于String的 == 和equals

     首先得说下String这个东西。java代码 被编译为class文件时,有一个常量池,代码中的非关键字(感觉这样的描述也不对,具体点就是 变量名,变量值,方法名,方法参数名,类名,包名 等等 ,太多列举不完)都以字符串的形式存储在里面。在class被classloader 加载进内存时,会在内存中 建一块 这个class的 运行时常量池 ,存放class文件的常量池(这是个人理解,还没去求证,但是应该大部分会加载进来)及其他更多的信息。

    package com.one.study;

public class TestString {

    private String s1 = "hello";
    private static String s2 = "hello";
    public static void main(String[] args) {
     String hello = "hello";

     String s3 = new String("hello");
   }
 
   public void hello()
   {}
}

   经过对编译出来的class进行查看,可以发现  s1 , s2 ,s3, hello 变量(自身)  的 值  和 方法名 hello 都指向 class的常量池中的同一个字符串 "hello" (被优化)。而在class被加载进内存之后,一个字符串也只会生成一个,所以他们的使用都是从这一个字符串中引用出来的。

  所以s1 ,s2 ,hello 他们在内存中的值(地址) 都是指向 运行时常量池中的同一个字符串,所以在他们之间进行 == 操作的时候 ,结果是true。

   而s3 因为是new 了一个String 对象,这个对象是存放在堆上的,所以s3中的值(地址) 是指向堆的,而不是指向运行时常量池的,但是在初始化时,传入的 "hello" 值,还是是从常量池中的"hello"  copy 去的。

   所以s1 和 s3 进行 == 操作时, 返回的是false。

  

  下面说下String 的equals方法 。

   因为String对equals方法进行了重写。贴一下源码

    public boolean equals(Object paramObject)
  {
    if (this == paramObject) {
      return true;
      }
   if ((paramObject instanceof String)) {
     String str = (String)paramObject;
     int i = this.count;
    if (i == str.count) {
       char[] arrayOfChar1 = this.value;
        char[] arrayOfChar2 = str.value;
      int j = this.offset;
       int k = str.offset;
       while (i-- != 0) {
          if (arrayOfChar1[(j++)] != arrayOfChar2[(k++)])
           return false;
       }
      return true;
      }
    }
   return false;
  }

  他是 先 还是 对两个object 的内存值进行比较,如果相同,则直接返回true,如果不同,则判断比较对象是否是String,如果是再转换为String进行 内容上的比较。

   所以 s1,s2,s3 之间进行 equals操作时,返回的都市true。

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值