String类为什么被final修饰以及底层问题

为什么被final修饰:

一.被final修饰的类,不可以被继承,所以不会别其它类改变,这样会更加的安全.

二.string是共享在常量池中的,String str="abc", char data[] = {'a', 'b', 'c'};是等价的,他们都放在了字符串常量池中.

衍生问题:

三.Java有两种创建字符串的方式,

  String str1 = "java"; String str2 = new String("java");

用双引号创建和用new来创建。这两种方式创建出来的String存储的位置上不同的。当使用双引号方式时,相当于显式的声明了字符串的值(字面值),所以是作为一个常量,存储在方法区的常量池中。使用new方式来创建String时,JVM会在堆上分配一块区域,存储一个String对象,值为“java”。

  其中一个String是new的一个非new的

String s = "1";
String s1 = new String("1");
System.out.println(s == s1);

此时返回的也是false:
在这里插入图片描述
s是非new的所以不会在堆中创建引用,s1会在堆中创建引用,所以s指向常量池中的地址,s1指向堆中的地址,所以s==s1为false,反过来先new s再定义非new s1也是一样的。

四.String的==和equals是不同的,

==比较的是两个String在内存中的地址是否相同,equals比较的是两个String的值是否相同。所以,如果String是用双引号方式创建,则两个String都指向常量池中的同一个位置,这时==是成立的,equals也成立。如果String是用new方式创建的,两个String如果不是指向堆上的同一个String对象,则==不成立,而如果值相同,则equals成立.

五.String的运算

如果运算时,等号右边有一个“运算数”不为常量(即使该“运算数”是用双引号创建的常量),则得到的结果相当于new创建的一个新的String对象,如:

        String str1 = "abc";
        String str2 = "abcd";
        String str3 = "abc" + "d";
        String str4 = str1 + "d";
        
        System.out.println(str2 == str3);//true
        System.out.println(str2.equals(str3));//true
        System.out.println(str2 == str4);//false !!!!
        System.out.println(str2.equals(str4));//true

六.intern()方法的用法

intern方法会返回一个字符串对应的常量值。在执行intern方法时,JVM会检查常量池中是否存在和该字符串相同的常量值,如果有,则返回该常量值,若没有,则创建该常量值,并返回。即,intern返回的是值常量池中的String,不是堆上的String,相当于用双引号创建String。

        String str1 = "abc";
        String str2 = "abcd";
        String str3 = "abc" + "d";
        String str4 = (str1 + "d").intern();
        
        System.out.println(str2 == str3);//true
        System.out.println(str2.equals(str3));//true
        System.out.println(str2 == str4);//true
        System.out.println(str2.equals(str4));//true

文章参考:

https://www.cnblogs.com/YLsY/p/5729113.html

https://blog.csdn.net/xlantian/article/details/79533321

猜你喜欢

转载自blog.csdn.net/a480675751/article/details/86626534