查看API文档我们可以发现,String类代表字符串,Java程序中的所有字符串字面值(如”abc”)都作为此类的实例实现。
其实就是说:程序当中所有的双引号字符串,都是String类的对象。(就算没有new,也照样是。)
String字符串的特点:
1.字符串的内容永不可变。
2.因为String对象是不可变的,所以共享。
3.字符串效果上相当于是char[]字符数组,但是底层原理是byte[]字节数组。
创建字符串的常见3+1种方式:
三种构造方法:
public String():创建一个空白字符串,不含有任何内容
public String(char[] array):根据字符数组的内容,来创建对应的字符串。
public String(byte[] array):根据字节数组的内容,来创建对应的字符串。
一种直接创建:
String str = "Hello":右边直接用双引号
注意:直接写上双引号,就是字符串对象。
public static void main(String[] args) {
// 使用空参构造
String str1 = new String(); // 小括号留空,说明字符串什么内容都没有
System.out.println("第一个字符串:" + str1);
// 根据字符数组的内容来创建对应的字符串
char[] charArray = {'A','B','C'};
String str2 = new String(charArray);
System.out.println("第二个字符串:" + str2); // 注意是拼接数组元素,没有存入逗号
// 根据字节数组的内容来创建对应的字符串
// 小范围到大范围,范围没超,byte类型:ASCII码隐式强制转换
// 回忆知识点:65--A;97--a;48--0
byte[] byteArray = {97,98,99};
String str3 = new String(byteArray); // 查看源码ctrl+点击
System.out.println("第三个字符串:" + str3);
String str4 = "Hello";
System.out.println("第三个字符串:" + str3);
}
我们来分析这个代码在内存里的走向。
首先提一个关于JDK迭代知识点:
JDK1.7之前,运行时常量池(字符串常量池也在里面)是存放在方法区,此时方法区的实现是永久带。
JDK1.7,字符串常量池被单独从方法区移到堆里,运行时常量池剩下的还在永久带(方法区)。
JDK1.8,永久带更名为元空间(方法区的新实现),但字符串常量池还在堆里,运行时常量池在元空间(方法区)。
字符串常量池当中的字符串对象保存的是byte[]字节数组的地址值。
JVM帮忙创建双引号内的字符串对象时,是先将它化作一个字节数组,再把字节数组的地址值保存在字符串常量池中的这个字符串对象里,然后把池中的字符串对象的地址赋值给栈里的局部变量。
而创建char[]数组的对象时,也相当于在堆里new出来的一个数组,找到地址赋值,但注意并不在常量池里。
最后在堆里根据char[]数组new一个字符串对象时,是先将这个在堆里的char[]数组转换为字节数组,然后在堆里new出一个字符串对象,将这个字节数组的地址赋值给字符串对象,然后将字符串对象的地址赋值给局部变量。
注意事项:
1.对于基本类型的数据来说,==进行的是数值比较;对于引用类型来说,==进行的是地址值的比较。
2.双引号直接写的字符串在常量池中,new的不在池中(但都在堆中)。
————————————————
版权声明:本文为CSDN博主「wardoluo」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/wardo_l/article/details/113824135