WHCSRL 技术网

[Java] Analysis of String type string ❤

目录

一、前言

二、String类概述

三、字符串的特点

四、String 构造方法

图示: 

代码演示:

五、String类对象的特点

六、比较字符串的方法

用法:

代码图示:

七、判断两个字符串地址是否相等

StringBuilder和StringBuffer的区别


一、前言

在java中,和C语言一样,也有关于字符串的定义,并且有他自己特有的功能,下面我们

一起来学习一下。

二、String类概述

string在软件包java.lang下,所以不需要导包。

String字符串是java中的重点,String 类表示字符串类 ,所有的字符串(如"adf")都属于

此类,也就是说有" "双引号括起来的都属于此类,

三、字符串的特点

字符串不可变,他们的值在创建之后不能被改变。

虽然String的值的不可变的,但是他们可以被共享。共享就是其他成员也可以拥有这个值,

字符串效果相当于数组(char[]),但是底层原理是字节数组(byte[])

jdk8以前是字符数组,jdk9以后是字节数组。

四、String 构造方法

public String()  //创建一个空白字符串对象,不含有任何内容。

public String(char[] ch)//根据字符数组的内容来创建字符对象。

public String (byte[] b)//根据字节数组的内容来创建字节对象。

String s=“abc”         //字节赋值的方式创建对象,内容就是abc。

图示: 

代码演示:

  1. public class StringDemo {
  2. public static void main(String[] args) {
  3. String s1=new String();//创建空的字符串,其中不包括任何内容
  4. System.out.println("s1:"+s1);
  5. char[]chs={'a','b','c'};
  6. String s2=new String(chs);//给chs创建对象
  7. System.out.println("chs:"+s2);//输出ch:abc
  8. byte[]bys={97,98,99};
  9. String s3=new String(bys);
  10. System.out.println("bys:"+s3);//输出bys:abc,他会 转化成对应的Ascll码值
  11. String s4="abc";
  12. System.out.println("s4:"+s4);//输出s4:abc
  13. }
  14. }

综上看,推荐使用直接赋值的方式得到字符串对象。

五、String类对象的特点

通过new创建的的字符串对象,每一次new都会申请一个内存空间,虽然内容一样,

但是地址不同

通过直接赋值的方式来创建对象,赋值相同的值,因为是常量在堆内存的常量池中,

进入常量池的规则是:如果常量池中没有这个常量,则在常量池中创建一个,如果有

,则把已经存在的常量地址赋值给他,所以创建不同的变量去接受相同的值,他的内容

是一样的,地址也是一样的 。

六、比较字符串的方法

我们在比较两个数字是否相同时,一般用的是==来判断,那么要比较两个字符串相等

用的是什么呢,答案是用equals。

==用来判断两个字符串的地址是否相同,相同返回true,不同返回false。

equals用来比较两个字符串的值是否相同,相同返回true,不同返回false。

用法:

  1. public class String1 {
  2. public static void main(String[] args) {
  3. String s1="hello";
  4. String s2="world";
  5. String s3= "helloworld";
  6. String s4=s1+s2;
  7. System.out.println(s7==s8);//0
  8. System.out.println(s3==s4);//比较两个字符串的地址是否相同
  9. System.out.println(s3.equals(s4));//比较两个字符串中的值是否相同
  10. }
  11. }

代码图示:

 原因下文分析。

七、判断两个字符串地址是否相等

在字符串中,两个字符串相加可以的到一个新的字符串,这是我们知道的,但是地址会是一样的吗

看下列代码:

  1. public class String1 {
  2. public static void main(String[] args) {
  3. String s1="hello";
  4. String s2="world";
  5. String s3= "helloworld";
  6. String s4=s1+s2;
  7. String s5="he"+"llo";
  8. String s6="hello"+"world";
  9. String s7="hello"+s2;
  10. String s8=s1+"world";
  11. System.out.println(s3==s6);//比较两个地址是否相同
  12. System.out.println(s1==s5);
  13. System.out.println(s3==s7);
  14. System.out.println(s3==s8);
  15. System.out.println(s7==s8);
  16. System.out.println(s3==s4);
  17. System.out.println(s3.equals(s4));//比较两个字符串中的值是否相同
  18. }
  19. }

我们仔细分析:

第一个:

  1. String s3= "helloworld";
  2. String s6="hello"+"world";

s3首先在常量池中创建了一个helloworld的常量,s6是有两个新的字符串连接起来的,

这两个字符串常量创建新的字符串常量,存储在常量池中时,因为helloworld已经存在,

所以常量池就不会创建新的字符串了,直接把已经存在的s3地址赋值给s6,所以他们地址

相同。

第二个:

  1. String s1="hello";
  2. String s5="he"+"llo";

这个分析和第一个一样,地址相同。

第三个:

  1. String s2="world";
  2. String s3= "helloworld";
  3. String s7="hello"+s2;

s2在常量池中创建world,s3在常量池中创建helloworld,s7是由一个变量s2连接一

个新的字符串"world",首先会在常量池创建字符串"world",然后两者之间进行"+"

操作,根据字符串的串联规则,s7会在堆内存中创建StringBuilder(或StringBuffer)

对象,通过append方法拼接s2和字符串常量"world”,此时拼接成的字符串"helloworld"

是StringBuilder(或StringBuffer)类型的对象,通过调用toString方法转成String对

象"helloworld",所以s7此时实际指向的是堆内存中的"helloworld"对象,堆内存中对

象的地址和常量池中对象的地址不一样。

StringBuilder和StringBuffer的区别

1.StringBuffer 对几乎所有的方法都实现了同步,线程比较安全,在多线程系统中可以保

证数据同步。

2.StringBuilder 没有实现同步,线程不安全,在多线程系统中不能使用 StringBuilder。

3.当需要考虑线程安全的场景下使用 StringBuffer,如果不需要考虑线程安全,追求效率的场

景下可以使用 StringBuilder。

第四个:

  1. String s1="hello";
  2. String s3= "helloworld";
  3. String s8=s1+"world";

解释同上,重新简单的说一下,s3在先在常量池中创建helloworld,s8是由变量s1和常量

world加起来的,会先在常量池中创建world,然他他们现在之后会在堆内存中存在,所以

他们的地址不同。

 总结:一般带有变量的相加操作是在堆中创建的

 第五个:

  1. String s1="hello";
  2. String s2="world";
  3. String s7="hello"+s2;
  4. String s8=s1+"world";

解释和上面有些相似之处,他们都是有变量加常量,所以他们都是在堆内存中创建的,

堆内存的地址是不会相同的。

 第六个:

  1. String s1="hello";
  2. String s2="world";
  3. String s3= "helloworld";
  4. String s4=s1+s2;
首先在常量池中创建唯一的常量,然后再,s4进行两个变量的相加操作,所生成的是在

堆内存中的,所以地址不同。

推荐阅读