无论你是一个编程新手还是老手,提到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")的堆内存和栈内存的变化

我们首先来看一段代码
  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是相等的。所以“==”是用来进行地址值比较的。
   那么怎么判断两个字符串的内容是否相等呢?我们先看个例子
  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对象,而使用这种方法还有另外一个好处,就是如果一个字符串已经被一个名称所引用,则以后再有相同的字符串声明时,就不会再重新开辟空间,而继续使用已经开辟好的堆内存,啥意思?我们通过一个例子来进行说明,这个例子的代码如下
  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.字符串的内容不可变
字符串的内容不可变?可能有的说你别在这忽悠我了,咋可能这不是胡扯吗?先别急,我们同样先看个例子代码如下
  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)这种写法可以避免空指针,为啥?可能有好多人经常这样写,但是不明白原理是啥?我们从一个例子说起
  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. }

再来看段代码
  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的源码如下
  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";     //不创建对象,只是创建一个新的引用
关于String类型的分析到这里就结束了,下班后加了会班,完成了此博客,如果你觉着对你有用,我不介意你留言顶一个,哈哈,今天是周五,祝大家周末愉快啊。。。。

Java基础之String中equals,声明方式,等大总结的更多相关文章

  1. java基础解析系列(十一)---equals、==和hashcode方法

    java基础解析系列(十一)---equals.==和hashcode方法 目录 java基础解析系列(一)---String.StringBuffer.StringBuilder java基础解析系 ...

  2. Java基础之String、StringBuffer、StringBuilder浅析

    Java基础之String.StringBuffer.StringBuilder浅析 一.前言: 位于java.lang包下的String.StringBuilder.StringBuffer一般都是 ...

  3. Java 字符串比较,String 中的一些方法 == 和 equals 的详解

    "==" 是比较的是两个对象的内存地址,而equals方法默认情况下是比较两个对象的内存地址. 1.String str = "hello"  生成的字符串,首 ...

  4. java基础63 JavaScript中的Number、Math、String、Date对象(网页知识)

    本文知识点(目录): 1.Number对象    2.Math对象    3.String对象    4.Date对象 (日历例子) 1.Number对象 1.1.Number对象的创建方式 方式1: ...

  5. 【Java基础】String 相关知识点总结

    String 相关知识点总结 字符串的不可变性 概述 String 被声明为 final,因此它不可继承 在 Java8 中,String 内部使用 char 数组存储数据 public final ...

  6. Java基础:hashCode与equals个人学习记录

    摘要: 本文主要记录本人对hashCode和对equals两个知识点的学习过程. 从学生时期初学java,就知道hashCode和equals这两个方法,工作中equals方法使用也是特别频繁,要说e ...

  7. Spring基础——IOC九种bean声明方式

    Spring简介 Spring不是服务于开发web项目的功能,或业务.而是服务于项目的开发,方便各层间的解耦调用,方便对类的批量管理,是提高软件开发效率,降低后期维护成本的框架. Spring的核心思 ...

  8. Java 基础之 String 类

    String String 被声明为 final,因此不能被继承.(Integer 等包装类也不能被继承) 在 java8 中,String 内部使用 char 数组 来存储数据 public fin ...

  9. String基础: String两种创建对象方式的比较

    字符串常量 在一般的语言中常量一旦声明则不可改变,在java中的字符串常量是以匿名对象来表示的 javaz中字符串两种定义方法: String strA= new String("hello ...

随机推荐

  1. python 网络并发 :理论部分

    1.今日内容大纲 进程的介绍(理论部分) 进程的创建以及分析 获取进程的pid 进程之间的隔离 1.进程的介绍(理论部分) 1.1什么是进程 一个正在被cpu执行的程序就是一个进程,一个程序可以开启多 ...

  2. Kotlin for Java Developers 学习笔记

    Kotlin for Java Developers 学习笔记 ★ Coursera 课程 Kotlin for Java Developers(由 JetBrains 提供)的学习笔记 " ...

  3. 第6.3节 Python动态执行之动态编译的compile函数

    Python支持动态代码主要三个函数,分别是compile.eval和exec.本节介绍compile函数的语法和相关使用.compile函数用来编译一段字符串的源码,将其编译为字节码或者AST(抽像 ...

  4. PyQt(Python+Qt)学习随笔:QDockWidget停靠部件的allowedAreas属性

    专栏:Python基础教程目录 专栏:使用PyQt开发图形界面Python应用 专栏:PyQt入门学习 老猿Python博文目录 QDockWidget的allowedAreas属性用于控制停靠部件在 ...

  5. Python正则表达式处理的组是什么?

    在学习正则表达式处理开始阶段,对于匹配对象的group数据没有理解,查了资料进行验证测试,终于理解了. 组其实与组匹配模式相关,就是在匹配的正则表达式中使用小括号"()"括起来的任 ...

  6. 【JAVA并发第一篇】Java的进程与线程

    1.进程与线程 1.1.进程 进程可以看作是程序的执行过程.一个程序的运行需要CPU时间.内存空间.文件以及I/O等资源.操作系统就是以进程为单位来分配这些资源的,所以说进程是分配资源的基本单位. ( ...

  7. 使用cmd制作图片木马

    我们可以使用windows下自带的cmd制作图片木马,配合文件包含漏洞可以达到getshell的目的 我们找到一张图片:kiss.jpg 如图: 写好一句话木马:chopper.php 将两者放在同一 ...

  8. Linux相关知识基础

    目录 前言 第一章 Linux远程连接管理 1. 为什么要远程连接Linux系统 2. 连接前的小知识 2.2.1 IP地址 2.2.2 端口的概念 2.2.3 协议的概念 3. 远程连接Linux的 ...

  9. Android之window机制token验证

    前言 很高兴遇见你~ 欢迎阅读我的文章 这篇文章讲解关于window token的问题,同时也是Context机制和Window机制这两篇文章的一个补充.如果你对Android的Window机制和Co ...

  10. Java File类的简单使用

    Java File的简单使用(创建.删除.遍历.判断是否存在等) Java文件类以抽象的方式代表文件名和目录路径名.该类本身不能用来读数据或写数据,它主要用于磁盘上文件和目录的创建.文件的查找和文件的 ...