• Java当中聊一聊String吧


    1、String的基本特性

    • String:字符串,使用“ ”引用来表示

      • String s1 = “java”;
      • String s2 = new String(”java“);
    • String被final关键字修饰,不可被继承

    • String 实现了Serializable接口,表示字符串是支持序列化的。同时实现了Comparable接口,表示String可以比较大小

    • String早jdk8以前内部定义了final char[] value用于存储字符串数据。jdk9时改为byte[]

    • String:代表不可变的字符序列。简称:不可变性

    	@Test
        public void test01(){
            String s1 = "abc";//字面量定义的方式,"abc"存储在字符串常量池中
            String s2 ="abc";
            System.out.println(s1==s2);//判断地址:true
            s1 = "hello";
            System.out.println(s1);//hello
            System.out.println(s2);//abc
            System.out.println(s1==s2);//判断地址:false
        }
    	
    	@Test
        public void test02(){
            String s1 = "abc";
            String s2 ="abc";
            s2 += "def";
            System.out.println(s2);//abcdef
            System.out.println(s1);//abc
        }
    
        @Test
        public void test03(){
            String s1 = "abc";
            String s2 =s1.replace("a","m");
            System.out.println(s2);//mbc
            System.out.println(s1);//abc
        }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27

    字符串常量池中是不会存储相同内容的字符串的。

    2、字符串拼接操作

    • 常量与常量的拼接结果在常量池,原理是编译期优化
    • 常量池中不会存在相同内容的常量
    • 只要其中有一个是变量,结果就在堆中。变量拼接的原理是StringBuilder
    • 如果拼接的结果调用intern()方法,则主动将常量池中还没有的字符串对象放入池中,并返回此对象地址。
    	@Test
        public void test1(){
            String s1 = "a" + "b" + "c";
            String s2 = "abc";//"abc" 一定是放在字符串常量池中,将此地址赋给s2
            System.out.println(s1 == s2);//true
            System.out.println(s1.equals(s2));//true
        }
    
        @Test
        public void test2(){
            String s1 = "javaEE";
            String s2 = "hadoop";
            String s3 = "javaEEhadoop";
            String s4 = "javaEE"+"hadoop";//编译期优化
            //如果拼接符号的前后出现了变量,则相当于在堆空间当中new String(),具体的内容为拼接的结果:javaEEhadoop
            String s5 = s1+"hadoop";
            String s6 = "javaEE" +s2;
            String s7 = s1+ s2;
    
            System.out.println(s3 == s4 );//true 
            System.out.println(s3 == s5 );//false
            System.out.println(s3 == s6 );//false
            System.out.println(s3 == s7 );//false
            System.out.println(s5 == s6 );//false
            System.out.println(s5 == s7 );//false
            System.out.println(s6 == s7 );//false
    		//intern():判断字符串常量池中是否存在javaEEhadoop值,如果存在,则返回常量池中javaEEhadoop的地址;
            //如果字符串常量池中不存在javaEEhadoop,则在常量池中加载一份javaEEhadoop,并返回次对象的地址。
            String s8 = s6.intern();
            System.out.println(s3 == s8);//true
        }
    	@Test
        public void test3(){
            String s1 = "a";
            String s2 = "b";
            String s3 = "ab";
            /**
             * 注意的是变量之间的拼接
             * 如下 s1+s2 的执行细节:(s是临时定义的)
             * ① StringBuilder s = new StringBuilder();
             * ② s.append("a")
             * ③ s.append("b")
             * ④ s.toString()  ---》 约等于 new String("ab")
             */
            String s4 = s1 + s2;
            System.out.println(s3 == s4);//false
        }
    
    
        @Test
        public void test4(){
            String s1 = "javaEEhadoop";
            String s2 = "javaEE";
            String s3 = s2+"hadoop";
            System.out.println(s1 == s3); //false
    
            final String s4 = "javaEE";//s4:常量
            String s5 = s4 +"hadoop";
            System.out.println(s1==s5);//true
        }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    • 55
    • 56
    • 57
    • 58
    • 59
    • 60

    总结一下

    • 字符串拼接操作不一定使用的是StringBuilder!如果拼接符号左右两边都是字符串常量或常量引用,则仍然使用编译期优化,即非StringBuilder的方式
    • 针对于final修饰类、方法、基本数据类型、引用数据类型的量的结构时,能使用上final的时候建议使用上。
    • 拼接效率:StringBuilder的append()的方式添加字符串的效率远高于使用String的字符串拼接方式。使用String的字符串拼接方式:创建过多个StringBuilder和String的对象。并且内存占用更大。

    3、intern()的使用

    如果不是用双引号声明的String对象,可以使用String提供的intern方法:intern方法会从字符串常量池中查询当前字符串是否存在,若不存在就会将当前字符串放入常量池中。intern就是确保字符串在内存里只有一份拷贝,这样可以节约内存空间,加快字符串操作任务的执行速度。这个值会被存放在字符串内部池

    4、几道面试题

    new String(“ab”) 会创建几个对象?

    看字节码文件,可以查看到创建了两个对象

    • 一个对象是: new 关键字在堆空间创建的
    • 另一个对象是:字符串常量池中的对象 字节码指令:ldc

    new String(“a”) + new String(“b”)会创建几个对象?

    • 对象1: new StringBuilder()
    • 对象2: new String(“a”)
    • 对象3: 常量池中的"a"
    • 对象4: new String(“b”)
    • 对象5: 常量池中的"b"

    总结String的intern()的使用:

    • jdk1.6中,将这个字符串对象尝试放入串池。
      • 如果串池中有,则并不会放入。返回已有的串池中的对象的地址
      • 如果没有,会把此对象复制一份,放入串池,并返回串池中的对象地址。
    • jdk1.7起,将这个字符串对象尝试放入串池
      • 如果串池中有,则并不会放入。返回已有的串池中的对象的地址
      • 如果没有,会把此对象的引用地址复制一份,放入串池,并返回串池中的引用地址。

    String、StringBuffer、StringBuild的区别?

    • String中的对象是不可变的,也就可以理解为常量,是线程安全的。
    • StringBuffer和StringBuilder继承自AbstractStringBuilder,StringBuffer对相关方法加了同步锁,所以是线程安全的
    • StringBuilder并没有对相关方法进行加同步锁,所以是非线程安全的
  • 相关阅读:
    Hadoop、Spark、Storm、Flink区别及选择
    Spring Boot集成微信支付JSAPIV3保姆教程
    猿创征文|网络原理——UDP/TCP协议
    【Autopsy数字取证篇】Autopsy数字取证软件的下载安装与优化配置
    对思维进化的理解
    【蓝桥杯2025备赛】素数判断:从O(n^2)到O(n)学习之路
    go操作rgb图片
    Python3 基础语法
    SSH协议&在IDEA中使用Git及git GUI的简单使用
    持续集成工具jenkins操作
  • 原文地址:https://blog.csdn.net/qq_44763720/article/details/125897764