先看一段代码:

public static void main(String[] args) {
        String a = "a"+"b"+1;
        String b ="ab1";
        System.out.println(a == b);
    }

输出的结果是true.

1.==和equals的区别

(1).基本数据类型,也称原始数据类型。byte,short,char,int,long,float,double,boolean 他们之间的比较,应用双等号(==),比较的是他们的值。

(2).复合数据类型(对象类型),当他们用(==)进行比较的时候,比较的是他们在内存中的存放地址,所以,除非是同一个new出来的对象,他们的比较后的结果为true,否则比较后结果为false。
JAVA当中所有的类都是继承于Object这个基类的,在Object中的基类中定义了一个equals的方法,这个方法的初始行为是比较对象的内存地
址,但在一些类库当中这个方法被覆盖掉了,如String,Integer,Date在这些类当中equals有其自身的实现,而不再是比较类在堆内存中的存放地址了。
 对于复合数据类型之间进行equals比较,在没有覆写equals方法的情况下,他们之间的比较还是基于他们在内存中的存放位置的地址值的,因为Object的equals方法也是用双等号(==)进行比较的,所以比较后的结果跟双等号(==)的结果相同。

2.编译时优化

当编译器在编译代码String a="a"+"b"+1;时,会将其编译为:String a="ab1";这是为了提升整体工作效率JVM进行的优化。类似的优化还有很多,例如当程序中出现int i=3*4+120时,会在编译时就变成了i=132。
但需要注意的是,JVM只优化可以优化的部分,即在编译阶段就可以确定值的,如果字符串中出现了变量,那么JVM是不会做这样的编译时优化的。例如:

String a="a";

final String c="a";

String b=a+"b";

String c=c+"b";

String e=getA()+"b";

String d="ab";

String f = new String(b);

其中b和e在编译时不会被优化,而c会被优化,因为c是final变量,可认为是常量。所以b==d,e==d为false,c==d为true.

3.常量池

程序在运行的时候会创建一个常量池,它保证同一个值的字符串全局唯一。当使用 String b = "ab1", 这样的表达是创建字符串的时候,程序首先会在这个常良池中寻找相同值的对象,在程序中,a引用的对象先被放到了池中,所以在b被创建的时候,将使b也指向这个对象。所以a==b为true.
String 类有个intern方法,当调用它时,JVM会在常良池中通过equals方法查找是否存在等值的String,如果存在,则直接返回常量池中这个String对象的地址;如果不存在,则会在常良池中创建等值的字符串,然后再返回这个新创建空间的地址。因此,只要是同样的字符串,当调用intern方法时,都会得到常良池中对应String对象的地址。所以上面虽然b==d为false,但是b.intern()==d为true.

4.String类的equals方法实现。

前面说过, 对于复合数据类型之间进行equals比较,在没有覆写equals方法的情况下,他们之间的比较还是基于他们在内存中的存放位置的地址值的,但String,Integer,Date在这些类当中equals有其自身的实现,因此不再是比较类在堆内存中的存放地址了。下面是String类的equals方法实现:

public boolean equals(Object anObject)
{

//1、如果是同一个对象,直接返回true
       
if (this == anObject)
       
{
           
return true;
       
}
       
//2、如果传递进来的参数类型不是String,直接返回false
       
if (anObject instanceof String)
       
{
           
String anotherString = (String)anObject;
           
int n = count;//字符串长度
           
if (n == anotherString.count) //3、如果长度不相等,直接返回false
           
{
               
char v1[] = value;//取每一个位置的字符
               
char v2[] = anotherString.value;
               
int i = offset;
               
int j = anotherString.offset;
               
while (n-- != 0) //4、循环对比两个字符串的char数组,逐个比较字符是否一致,若存在不一致的情况,直接返回false
               
{
                   
if (v1[i++] != v2[j++])
                       
return false;
                }

return true;//5、循环结束都没有找到不匹配的,说明两个字符串相等,返回true.
           
}
       
}
       
return false;
}

在上例中,f==d为false,因为内存地址不一样,但是f.equals(d)为true.

String类的一些细节的更多相关文章

  1. Java之String类的使用细节

    String类的特点:     字符串对象一旦被初始化就不会被改变,字符串存储在字符串常量池中(字符串缓冲区).如果池中没有就创建,如果有就直接拿过来用.  代码验证如下:     String s ...

  2. 关于String类和String[]数组的获取长度方法细节

    一.在Java中,以下代码段有错误的是第(  )行 public static void main(String[] args) { String name = "小新";     ...

  3. Java中是否可以继承String类,为什么

    Java中,是否可以继承String类?为什么? 答案: 不可以,因为String类有final修饰符,而final修饰的类是不能被继承的,实现细节不允许改变. public final class ...

  4. C/C++关于string.h头文件和string类

    学习C语言时,用字符串的函数例如stpcpy().strcat().strcmp()等,要包含头文件string.h 学习C++后,C++有字符串的标准类string,string类也有很多方法,用s ...

  5. (转)C++——std::string类的引用计数

    1.概念 Scott Meyers在<More Effective C++>中举了个例子,不知你是否还记得?在你还在上学的时候,你的父母要你不要看电视,而去复习功课,于是你把自己关在房间里 ...

  6. Java学习笔记21(String类补充:正则表达式)

    正如python的re模块,不过Java和Python的正则表达式有一些区别,这里做简单介绍,具体的细节可以参考网上其他的文章: 功能:可以用于检验一个字符串,比如验证用户名,验证密码格式,验证是否是 ...

  7. String类的深入理解

    String不是基本数据类型,String和8种包装类型是不可变类.String和8种基本数据类型采用值传递. 关于方法区中的常量区和class文件中的常量区的关系,参考:https://www.cn ...

  8. 转载:C/C++关于string.h头文件和string类

    学习C语言时,用字符串的函数例如stpcpy().strcat().strcmp()等,要包含头文件string.h 学习C++后,C++有字符串的标准类string,string类也有很多方法,用s ...

  9. 逐步解读String类(一)

    一句题外话 面试刚入行的Java新手,侧重基础知识:面试有多年工作经验的老鸟,多侧重对具体问题的解决策略. 从一类面试题说起 考察刚入行菜鸟对基础知识的掌握程度,面试官提出关于String类的内容挺常 ...

随机推荐

  1. 输入5个学生的信息(包括学号,姓名,英语成绩,计算机语言成绩和数据库成绩), 统计各学生的总分,然后将学生信息和统计结果存入test.txt文件中

    题目分析: 1.首先想到的是数组存放数据,数组肯定是String类型. 2.String类型的数组,5行6列.要把从第0行第2列到第4行第4列的数据取出转换成数值型,再统计三科总分.最后把计算出的总分 ...

  2. SS、SP、BP寄存器

    SS, SP, BP 三个寄存器 SS:存放栈的段地址: SP:堆栈寄存器SP(stack pointer)存放栈的偏移地址; BP: 基数指针寄存器BP(base pointer)是一个寄存器,它的 ...

  3. IDEA创建Scala项目

    一.安装插件 见Scala入门篇 二.新建项目 选择new project,其中SBT相当于精简版的maven,其他的待补充.这里选择IDEA 填写信息,选择Scala SDK 在src目录下新建Sc ...

  4. 20155234 exp4 恶意代码分析

    实验4 恶意代码分析 系统运行监控 Schtasks 先建立一个netstat20155234.txt文件,在文件中输入 date /t >> c:\netstat20155234.txt ...

  5. 微信小程序之页面传值(路由、页面栈、globalData、缓存)

    1. 通过url带参数传递 1.1 固定参数传递 例如,从 list 页面到 detail 页面, 传递一个或多个固定值 list页面传值: <!--pages/list/list.js--&g ...

  6. Python_Xlrd&Xlwt

    import xlrd # \U 开始的字符被编译器认为是八进制 解决方法 r objWB = xlrd.open_workbook(r'C:\Users\IBM\Desktop\S1\7月下旬入库表 ...

  7. cadence allegro 封装原点修改

    打开 dra文件后 在菜单栏 setup - change drawing origin 在命令栏输入 新的参考点位置 如想更改新坐标位置为 1,2 .输入  x 1 2

  8. LintCode——交叉字符串

    描述:给出三个字符串:s1.s2.s3,判断s3是否由s1和s2交叉构成. 样例:s1 = "aabcc" s2 = "dbbca" - 当 s3 = &quo ...

  9. 1089. Insert or Merge (25)-判断插入排序还是归并排序

    判断插入排序很好判断,不是的话那就是归并排序了. 由于归并排序区间是2.4.8开始递增的,所以要判断给出的归并排序执行到哪一步,就要k从2开始枚举. 然后再对每个子区间进行一下sort即可. #inc ...

  10. 实验 六:分析linux内核创建一个新进程的过程

    实验六:分析Linux内核创建一个新进程的过程 作者:王朝宪  <Linux内核分析>MOOC课程http://mooc.study.163.com/course/USTC-1000029 ...