`

Java基础之String中equals,声明方式,等大总结

    博客分类:
  • Java
阅读更多

转载请注明出处:http://blog.csdn.net/dmk877/article/details/49420141 

    无论你是一个编程新手还是老手,提到String你肯定感觉特别熟悉,因为String类我们在学习java基础的时候就已经学过,但是String类型有我们想象的那么简单吗?其实不然,String类型的知识点还是比较多的。今天就和大家来一起讨论一下,关于String的一些容易让人疑惑的地方,废话不多说进入正题。。。如有谬误请批评指正,如果有疑问请留言。我会在第一时间修改或回答

通过本篇博客你将学到以下知识

   ①==和equals的区别,String a="abc"和String a=new String("abc")的堆内存和栈内存的变化

   ②String a="abc"和String a=new String("abc")两种声明方式的区别

   ③"abc".equals(abc)和a.equals("abc")的区别,从源码分析为什么"abc".euqals(str)可以避免空指针
   ④String对象的值是不能修改的,为啥可以重新赋值,并且改变
   ⑤String a;和String a = null,String a =""的区别
   ⑥String s1 = new String("s1") ;String s2 = new String("s1") ;总共创建了几个对象?
 
1、==和equals的区别,String a="abc"和String a=new String("abc")的堆内存和栈内存的变化
 
我们首先来看一段代码
[java] view plaincopy
  1. public class StringDemo {  
  2.     public static void main(String[] args) {  
  3.         String str1 = "hello";  
  4.         String str2 = new String("hello");  
  5.         String str3 = str2;  
  6.         System.out.println("str1==str2:"+(str1==str2));  
  7.         System.out.println("str1==str2:"+(str1==str3));  
  8.         System.out.println("str1==str2:"+(str2==str3));  
  9.     }  
  10. }  
打印结果如下
why?为什么会这样打印呢?上面的例子str1、str2、str3的内容都是一样的但比较的结果却是有的相等,有的不相等,这是为啥?首先看一张内存上述的内存分配图
    从图中可以清楚的发现每个String对象的内容实际是保存到堆内存中的,而且堆中的内容是相等的。但是对于str1和str2来说所指向的地址堆内存地址是不等的,所以尽管内容是相等的,但是地址值是不相等的,“==”是用来进行数值比较的,所以str1和str2比较不相等,因为str2和str3指向同一个内存地址所以str2和str3是相等的。所以“==”是用来进行地址值比较的。
   那么怎么判断两个字符串的内容是否相等呢?我们先看个例子
[java] view plaincopy
  1. public class StringDemo {  
  2.     public static void main(String[] args) {  
  3.         String str1 = "hello";  
  4.         String str2 = new String("hello");  
  5.         String str3 = str2;  
  6.         System.out.println("str1 equals str2:"+(str1.equals(str2)));  
  7.         System.out.println("str1 equals str2:"+(str1.equals(str3)));  
  8.         System.out.println("str1 equals str2:"+(str2.equals(str3)));  
  9.     }  
  10. }  
运行结果如下

因为equals方法的作用是将内容进行比较,所以此处的返回结果都为true。
总结以上两个例子我们可以总结出"=="和equals的区别是:“==”是用来进行数值比较的,在String中用“==”进行地址值的比较,而equals比较的是String的内容。
 
2.两种声明方式的区别
首先必须明白的一点就是一个字符串就是String的匿名对象,为什么这样说呢?我们可以通过"hello".equals("hello")的打印结果为true进行验证,因为“hello”可以通过“hello”.equals()直接调用String中的方法,因此对于String str1="hello";实际上就是把一个在堆内存中开辟好的堆内存空间的使用权给了str1对象,而使用这种方法还有另外一个好处,就是如果一个字符串已经被一个名称所引用,则以后再有相同的字符串声明时,就不会再重新开辟空间,而继续使用已经开辟好的堆内存,啥意思?我们通过一个例子来进行说明,这个例子的代码如下
[java] view plaincopy
  1. public class StringDemo {  
  2.     public static void main(String[] args) {  
  3.         String str1 = "hello";  
  4.         String str2 = "hello";  
  5.         String str3 = "hello";  
  6.         System.out.println("str1==str2:" + (str1 == str2));  
  7.         System.out.println("str1==str3:" + (str1 == str3));  
  8.         System.out.println("str2==str3:" + (str2 == str3));  
  9.     }  
  10. }  
打印结果如下

咦,这是啥子情况啊?上面我们刚得出结论"=="在String中比较的是地址值,为什么打印结果都是true呢?其实这是java中一种共享设计,这种设计思路是,在java中形成一个对象池,在这个池中保存多个对象,新实例化的对象如果已经在池中定义了则不再重新定义,而从池中取出继续使用。String采用了这种设计,在Java运行环境中有一个字符串池,由String类维护。执行语句String str1="hello"时,首先查看字符串池中是否存在字符串"hello",如果存在则直接将"hello"赋给str1,如果不存在则先在字符串池中新建一个字符串"hello",然后再将其赋给str1。执行语句String str=new String("hello")时,不管字符串池中是否存在字符串"hello",直接新建一个字符串"hello"(注意:新建的字符串"hello"不是在字符串池中),然后将其付给str。前一语句的效率高,后一语句的效率低,因为新建字符串占用内存空间。String str = new String()创建了一个空字符串,与String str=new String("")相同。
 
我们总结一下:单独是用""引号创建的字符串都是常量,编译期就已经确定好存储到String池中,使用new String("")创建的对象会存储到堆内存(head)中是运行时期创建的。
 
通过以上的两种实现方式的比较可以知道哪种方式更合适,对于字符串的操作直接采用直接赋值的方式完成,而不要采用构造方法传递字符串的方式完成,这样可以避免产生垃圾空间。
 
3.字符串的内容不可变
字符串的内容不可变?可能有的说你别在这忽悠我了,咋可能这不是胡扯吗?先别急,我们同样先看个例子代码如下
[java] view plaincopy
  1. public class StringDemo {  
  2.     public static void main(String[] args) {  
  3.         String str="hello";  
  4.         str=str+" world!";  
  5.         System.out.println("str="+str);  
  6.     }  
  7. }  
打印结果

可能有的人会说你看这不是变了吗?开始是"hello"后面是"hello world!"。从程序运行结果发现,String对象的内容确实已经修改了,但是内容真的修改了吗?下面通过内存的分配图说明字符串内容不可更改的含义

从上图可以清楚的发现,一个String对象内容的改变实际上时通过内存地址的"断开-连接"变化完成的,而本身字符串中的内容并没有任何的变化。
 
4."abc".equals(str)和str.equals("abc")的区别,从源码分析为什么"abc".euqals(str)可以避免空指针
我们应该都听说过"abc".equals(str)这种写法可以避免空指针,为啥?可能有好多人经常这样写,但是不明白原理是啥?我们从一个例子说起
[java] view plaincopy
  1. public class StringDemo {  
  2.     public static void main(String[] args) {  
  3.         String str1 = "hello";  
  4.         String str2 = null;  
  5.         System.out.println(str1.equals(str2));  
  6.     }  
  7. }  
再来看段代码
[java] view plaincopy
  1. public class StringDemo {  
  2.     public static void main(String[] args) {  
  3.         String str1 = "hello";  
  4.         String str2 = null;  
  5.         System.out.println(str2.equals(str1));  
  6.     }  
  7. }  
和上面的代码唯一不同的是上面是str1.equals(str2),这里是str2.equals(str1),我们看看打印结果会是啥样。

我曰它大爷,报空指针,为啥嘞?为啥st1.equals(str2)会打印false,而str2.equals(str1)就报空指针,这不科学啊。要想知道原因唯独源码最具说服力,我们来看看String类中的equals源码不就行了,好咱们去看看呗String类中equals的源码如下
[java] view plaincopy
  1. /**  
  2.      * Compares this string to the specified object.  The result is {@code  
  3.      * true} if and only if the argument is not {@code null} and is a {@code  
  4.      * String} object that represents the same sequence of characters as this  
  5.      * object.  
  6.      *  
  7.      * @param  anObject  
  8.      *         The object to compare this {@code String} against  
  9.      *  
  10.      * @return  {@code true} if the given object represents a {@code String}  
  11.      *          equivalent to this string, {@code false} otherwise  
  12.      *  
  13.      * @see  #compareTo(String)  
  14.      * @see  #equalsIgnoreCase(String)  
  15.      */    
  16.     public boolean equals(Object anObject) {    
  17.         //判断是否是和自己比较  
  18.         if (this == anObject) {    
  19.             return true;    
  20.         }    
  21.         //判断传过来的anObject是否是String类型的实例  
  22.         if (anObject instanceof String) {    
  23.             String anotherString = (String) anObject;    
  24.             int n = value.length;    
  25.             if (n == anotherString.value.length) {    
  26.                 char v1[] = value;    
  27.                 char v2[] = anotherString.value;    
  28.                 int i = 0;    
  29.                 //逐个字符进行比较  
  30.                 while (n-- != 0) {    
  31.                     if (v1[i] != v2[i])    
  32.                             return false;    
  33.                     i++;    
  34.                 }    
  35.                 return true;    
  36.             }    
  37.         }    
  38.         return false;    
  39.     }  
从源码中我们看到首先判断是否是和自己进行比较,然后判断传过来的对象是不是String类型的实例,注意 instanceof这个关键字的作用就是判断一个对象是哪个类的实例,注意这里是实例,我们直接String str2=null,此时str2并不是String类型的实例,不信你可以去验证,因为String str2=null,申明一个String类型的str2,同时在内存里申请了一个地址,但是该地址不指向任何引用地址,所以在执行这个判断时直接跳出if语句返回false。所以就不会报空指针啦啦啦。。。
 
5.String a;和String a = null,String a =""的区别
 
   String a;申明一个string类型的 a,即没有在申请内存地址,更没有在内存任何指向引用地址;

   String a = null ;申明一个string类型的 a,同时在内存里申请了一个地址,但是该地址不指向任何引用地址;

   String a = "" ;申明一个string类型的 a,既在内存里申请了地址,该地址又指向一个引用该字符串的引用地址;

   string a=null,String b="hello"; system.out.println(a+b);输出nullhello;
 
6.String s1 = new String("s1") ;String s2 = new String("s1") ;总共创建了几个对象?
对于这个问题,通过上面的分析其实也不难理解,首先String s1 = new String("s1"),我们看到("s1"),此时"s1"作为常量被读入,所以会在String池中创建一个对象,之后又看到了new此时会在堆内存(head)中再创建一个对象,所以执行String s1=new String("s1")时创建了两个对象,而接着执行String s2=new String("s1")时,("s1")也会被作为常量读入,但是由于String池中已经有了"s1"所以这一次不会在String池中创建对象了,而执行new时是必须要创建对象的,所以执行String s2=new String("s1");时创建了一个对象。看下面两段代码:
第一段代码:
String   s1   =   new   String("s1");     //创建二个对象,一个引用 
String   s2   =   new   String("s1");     //创建一个对象,并且以后每执行一次创建一个对象,一个引用 
第二段代码:
String   s3   =   "xyz";     //创建一个对象,一个引用   
String   s4   =   "xyz";     //不创建对象,只是创建一个新的引用
分享到:
评论

相关推荐

    JAVA基础课程讲义

    第一个JAVA程序的总结和提升 20 常用Java开发工具 20 常用dos命令 21 本章笔试作业 21 本章上机操作 21 第二章(1) 编程的基本概念 22 注释 22 标识符 22 关键字/保留字 23 变量(variable) 24 常量(Constant) 25 命名...

    java 面试题 总结

    JAVA相关基础知识 1、面向对象的特征有哪些方面 1.抽象: 抽象就是忽略一个主题中与当前目标无关的那些方面,以便更充分地注意与当前目标有关的方面。抽象并不打算了解全部问题,而只是选择其中的一部分,暂时不用...

    java 面对对象编程.pdf.zip

    面向对象基础 面向对象和面向过程的区别 成员变量与局部变量的区别 创建一个对象用什么运算符?对象实体与对象引用有何不同? 对象的相等和引用相等的区别 类的构造方法的作用是什么? 如果一个类没有声明构造方法,该...

    面试必问Java面试题,对标初级Java

    1.Java泛型(约束类型) 2.Java中函数式编程-Stream流-Lambda表达式 3.JVM虚拟机 4.Springcloud里面的组件 ...18.java中==equals有哪些区别 19.ArrayList和LinkedList有什么区别 20.Java面向对象有哪些特征

    java 编程基础题

    Employee类具有保护成员String变量employeeNo,Teacher类有String类型的teano和zc变量,并分别为两个子类编写一个无参的构造函数。若使两个子类正常的编译,请在父类中解决该问题。 public class Employee extends ...

    JAVA面试题最全集

    一、Java基础知识 1.Java有那些基本数据类型,String是不是基本数据类型,他们有何区别。 2.字符串的操作: 写一个方法,实现字符串的反转,如:输入abc,输出cba 写一个方法,实现字符串的替换,如:输入...

    java核心知识点整理

    3.Java是典型的强类型语言,即必须声明变量的类型,Java中有8种类型,6种数值类型(4个整数型和2个浮点型)、一个字符类型和一个boolean类型。 想学习java可以来这个群,首先是二二零,中间是一四二,最后是九零六...

    Java 2实用教程(第三版)实验指导与习题解答

    Java 2实用教程(第三版)实验指导与习题解答 清华大学出版社 (编著 耿祥义 张跃平) 实验模版代码 建议使用文档结构图 (选择Word菜单→视图→文档结构图) 上机实践1 初识Java 4 实验1 一个简单的应用程序 ...

    Java初学者都必须理解的六大问题

    其实不只String,很多Java标准类库中的类都是不可变的。在开发一个系统的时候,我们有时候也需要设计不可变类,来传递一组相关的值,这也是面向对象思想的体现。不可变类有一些优点,比如因为它的对象是只读的,所以...

    java面试宝典

    47、在java 中一个类被声明为final 类型,表示了什么意思? 12 48、下面哪些类可以被继承? 12 49、指出下面程序的运行结果: 【基础】 13 52、关于内部类: 13 53、数据类型之间的转换: 14 54、字符串操作:如何实现...

    Java精华(免费版)

    2、equalsIgnoreCase:忽略大小写的比较,上例中如果您输入的是BYE,则不会退出,因为大小写不同,但是如果使用这个方法,则会退出。 3、indexOf(int ch);返回字符ch在字符串中首次出现的位置 4、substring(int ...

    JAVA入门1.2.3:一个老鸟的JAVA学习心得 PART1(共3个)

    3.4 小结:基本数据类型—— Java中一切数据和运算的基础 63 3.5 习题 65 第4章 Java中的程序执行流程 67 教学视频:1小时57分钟 4.1 顺序执行 67 4.2 使用if-else让程序懂得判断 68 4.2.1 if语句 68 4.2.2 ...

    Java复习题及答案

    1、在java中如果声明一个类为final,表示什么意思? 答:final是最终的意思,final可用于定义变量、方法和类但含义不同,声明为final的类不能被继承。 2、父类的构造方法是否可以被子类覆盖(重写)? 答:父类的...

    Java的六大问题你都懂了吗

    这些问题对于认真学习java的人都要必知的,当然如果你只是初学者就没必要那么严格了,那如果你认为... 最后呢,还有些java的技术,包括EJB3.0等,可以选择学习,与三大轻量级框架相比,EJB就是当之无愧的重量级了。

    Java入门1·2·3:一个老鸟的Java学习心得.PART3(共3个)

    3.4 小结:基本数据类型—— Java中一切数据和运算的基础 63 3.5 习题 65 第4章 Java中的程序执行流程 67 教学视频:1小时57分钟 4.1 顺序执行 67 4.2 使用if-else让程序懂得判断 68 4.2.1 if语句 68 4.2.2 ...

    java编的简单计算器

    一个java编译的最简单的计算器 import java.awt.*; import java.awt.event.*; import javax.swing.*; public class jisuan extends JFrame implements ActionListener{ /** * */ private static final ...

    java六大必须理解的问题-来自网络少许有误(非常适合初学者).doc

    我们并没有声明一个String对象,我们只是声明了一个只能指向String对象的引用变量。所以,如果在刚才那句语句后面,如果再运行一句: String string = s; 我们是声明了另外一个只能指向String对象的引用,名为...

    java初学者必须理解的6大问题

    我们并没有声明一个String对象,我们只是声明了一个只能指向String对象的引用变量。所以,如果在刚才那句语句后面,如果再运行一句:  String string = s;  我们是声明了另外一个只能指向String对象的引用,名为...

    Java语言程序设计(第3版)第06章-字符串.pptx

    6.1.1 字符串比较 Java语言程序设计(第3版) 不能使用"=="号来比较字符串内容是否相等 比较内容是否相等: boolean equals(String str) boolean equalsIgnoreCase(String str) s1 s2 Hello Hello 字符串常量池 s1 s2...

Global site tag (gtag.js) - Google Analytics