【JAVA】String类

2023-09-12 08:44:06

作者主页:paper jie_的博客

本文作者:大家好,我是paper jie,感谢你阅读本文,欢迎一建三连哦。

本文录入于《JAVASE语法系列》专栏,本专栏是针对于大学生,编程小白精心打造的。笔者用重金(时间和精力)打造,将javaSE基础知识一网打尽,希望可以帮到读者们哦。

其他专栏:《JAVA》《算法详解》《C语言》等

内容分享:本期将会对JAVA中的String类进行分享

目录

String类的引出

String类的常用方法

字符串构造

String对象的比较

比较是否引用的是同一个对象

boolean aquals(Object o)比较方法:按照字典序比较

int compareTo(String s)方法:按照字典序进行比较

字符串查找

转换

数值与字符串转换

大小写转换

字符串与数组转换

格式化

字符串替换

字符串拆分

字符串截取

去处空格

字符串的不可变性

String设计成不可变的原因

字符串的修改

StringBuilder和StringBuffer

String,StringBuilder,StringBuffer的区别


String类的引出

C语言中,我们发现里面是没有字符串类型的,要想表达字符串只能通过字符数组或者字符指针,可以用它的标准库提供的字符串函数完成操作,但是这种将数据和操作数据方法分离的方式它不符合面向对象的思想,因此java中专门提供了String类。

String类的常用方法

字符串构造

String类提供的构造方法一般有三种:

public class Test {
    public static void main(String[] args) {
        //使用常量串构造
        String s1 = "hello word";
        System.out.println(s1);
        
        //直接new一个String对象
        String s2 = new String("hello word");
        System.out.println(s2);
        
        //使用字符数组进行构造
        char[] array = {'h','e','l','l','0'};
        String s3 = new String(array);
    }
}

这里要注意一个地方:

String是一个引用类型,内部并不会存储字符串本身,我们可以看看string类的源码:

    public static void main(String[] args) {

// s1和s2引用的是不同对象 s1和s3引用的是同一对象
        String s1 = new String("hello");
        String s2 = new String("world");
        String s3 = s1;

        System.out.println(s1.length()); // 获取字符串长度---输出5
        System.out.println(s3.isEmpty()); // 如果字符串长度为0,返回true,否则返回false
    }

画图分析:

String对象的比较

字符串的比较,java中提供了4种方式

比较是否引用的是同一个对象

对内置类型,==比较的是变量中的值,对于引用类型==比较的是引用中的地址:

    public static void main(String[] args) {
        int a = 10;
        int b = 20;
        int c = 10;
// 对于基本类型变量,==比较两个变量中存储的值是否相同
        System.out.println(a == b); // false
        System.out.println(a == c); // true
// 对于引用类型变量,==比较两个引用变量引用的是否为同一个对象
        String s1 = new String("hello");
        String s2 = new String("hello");
        String s3 = new String("world");
        String s4 = s1;
        System.out.println(s1 == s2); // false
        System.out.println(s2 == s3); // false
        System.out.println(s1 == s4); // true
    }
    

boolean aquals(Object o)比较方法:按照字典序比较

String类重写父类Object中的equals方法,Object类中equals方法默认按==比较,String重写equals方法后,按照以下规则比较:

equals方法的源码:

    public boolean equals(Object anObject) {
// 1. 先检测this 和 anObject 是否为同一个对象比较,如果是返回true
        if (this == anObject) {
            return true;
        } // 2. 检测anObject是否为String类型的对象,如果是继续比较,否则返回false
        if (anObject instanceof String) {
// 将anObject向下转型为String类型对象
            String anotherString = (String) anObject;
            int n = value.length;
// 3. this和anObject两个字符串的长度是否相同,是继续比较,否则返回false
            if (n == anotherString.value.length) {
                char v1[] = value;
                char v2[] = anotherString.value;
                int i = 0;
// 4. 按照字典序,从前往后逐个字符进行比较
                while (n-- != 0) {
                    if (v1[i] != v2[i])
                        return false;
                    i++;
                }
                return true;
            }
        }
        return false;
    }

int compareTo(String s)方法:按照字典序进行比较

与equals方法不同是,equals返回的是对还是错,而compareTo返回的是整数。比较方式:

1 先按照字典序比较,出现不同的字符,返回这两个字符的大小差值。

2 如果一直到一个字符串比较完字符都相等的话但他们的长度不同,则返回值是这两个字符串的长度差值

public class Test {
        public static void main(String[] args) {
            String s1 = new String("abc");
            String s2 = new String("ac");
            String s3 = new String("abc");
            String s4 = new String("abcdef");
            System.out.println(s1.compareTo(s2)); // 不同输出字符差值-1
            System.out.println(s1.compareTo(s3)); // 相同输出 0
            System.out.println(s1.compareTo(s4)); // 前k个字符完全相同,输出长度差值 -3
        }
}

字符串查找

字符串查找String类提供了许多查找方法:

public class Test {
    public static void main(String[] args) {
        //通过下标找字符
        String s1 = "abcdefgh";
        char ch = s1.charAt(2);
        System.out.println(ch);

        //通过字符找下标
        int index1 = s1.indexOf('c');
        System.out.println(index1);

        //跳着找
        int index2 = s1.indexOf('e', 3);
        System.out.println(index2);

        //通过子字符串找第一次出现的下标
        int index3 = s1.indexOf("def");
        System.out.println(index3);

        //跳着找子字符串返回第一次出现的下标
        int index4 = s1.indexOf("def", 2);
        System.out.println(index4);
        
        //倒着找字符
        int lastindex1 = s1.lastIndexOf('c');
        System.out.println(lastindex1);

        //跳着倒着找字符
        int lastindex2 = s1.lastIndexOf('c', 5);
        System.out.println(lastindex2);

        //倒着找字符串
        int lastindex3 = s1.lastIndexOf("ef");
        System.out.println(lastindex3);

        //倒着跳着找字符串
        int lastindex4 = s1.lastIndexOf("ef", 6);
        System.out.println(lastindex4);
    }

转换

数值与字符串转换

数字转字符串:

    public static void main(String[] args) {
        //数字转字符串
        String s1 = String.valueOf(1111);
        String s2 = String.valueOf(11.11);
        String s3 = String.valueOf(true);
        String s4 = String.valueOf(new person("baga", 19));
        System.out.println(s1);
        System.out.println(s2);
        System.out.println(s3);
        System.out.println(s4);
    }

字符串转数字:

    public static void main(String[] args) {
        //字符串转数字 integer Double 是包装类型
        int data = Integer.parseInt("12345");
        double data2 = Double.parseDouble("12.43");
        System.out.println(data);
        System.out.println(data2);
    }

大小写转换

    public static void main(String[] args) {
        //大写转小写
        String s1 = "hello word";
        String str1 = s1.toUpperCase();
        System.out.println(str1);
        //小写转大写
        String s2 = "HELLO WORD";
        String str2 = s2.toLowerCase();
        System.out.println(str2);
    }

字符串与数组转换

public static void main(String[] args) {
        //字符串转数组
        String s1 = "hello word";
        char[] ch = s1.toCharArray();
        for (int i = 0; i < ch.length; i++) {
            System.out.print(ch[i]+" ");
        }

        //数组转字符串
        char[] array = new char[]{'f','f','f','o','o','o'};
        String s2 = new String(array);
        System.out.println(s2);
    }

格式化

    public static void main(String[] args) {
        String s = String.format("%d %d %d ", 12, 34, 56);
        System.out.println(s);
    }

字符串替换

    public static void main(String[] args) {
        String s1 = "asdfasfasafaiughesfsfafil";
        //替换全部的子字符
        String str1 = s1.replaceAll("fa", "000");
        System.out.println(str1);
        //替换第一个个子字符
        String str2 = s1.replaceFirst("fd", "99");
        System.out.println(str2);
    }

字符串拆分

    public static void main(String[] args) {
        //全部拆分
        String s1 = "hello word wo lai le";
        String[] str = s1.split(" ");
        for (int i = 0; i < str.length; i++) {
            System.out.println(str[i]);
        }
        System.out.println();
        //拆分部分
        String[] str2 = s1.split(" ", 3);
        for (int i = 0; i < str2.length; i++) {
            System.out.println(str2[i]);
        }
    }

使用特殊符号"|","*","+"需要用一个斜杠表示,而两个斜杠表示一个斜杠

多个字符分割时,用竖线|隔开

字符串截取

从一个完整的字符串中截取部分内容

方法功能
String substring(int beginIndex)从指定索引截取到结尾
 

String substring(int beginIndex, int endIndex)
 
截取部分内容
    public static void main(String[] args) {
        //从指定索引到末尾
        String s1 = "sadfdsag";
        String str1 = s1.substring(3);
        System.out.println(str1);
        //截取部分内容 遵循左闭右开规则
        String str2 = s1.substring(1, 3);
        System.out.println(str2);

    }

1. 索引从0开始 

2. 注意前闭后开区间的写法, substring(0, 5) 表示包含 0 号下标的字符, 不包含 5 号下标 

去处空格

trim会去除字符串开头和结尾的空白字符(空格,换行,制表符)

    public static void main(String[] args) {
        String s1 ="   hello werd sad   ";
        String str = s1.trim();
        System.out.println(str);
    }

字符串的不可变性

String类是一种不可变的对象,它的内容是不可以改变的。字符串它为什么不能改变呢?

1 String类它在设计的时候就是不可以改变的。字符串实现描述中已经说的很明白了:

通过1观察源码,我们发现:

String类中的字符实际保存在内部维护的value字符数组中

String类被final修饰,不可以被继承

value被final修饰,表明value自身的值是不能被改变的,也就是不可以引用其他字符数组,但是它空间中的内容可以改变。

所有涉及到修改字符串的操作,都是创建一个新对象,改变的是新对象

栗子:

注意:

这里字符串不能修改不是因为value被final修饰了,final修饰只是类是表示不能被继承,修饰引用类型表示不能引用其他对象,但是引用对象里面的内容还是可以修改的。

不能修改是因为value被private修饰,只能被当前类访问,在当前类的外部是访问不到的。

栗子:

public static void main(String[] args) {
final int array[] = {1,2,3,4,5};
array[0] = 100;
System.out.println(Arrays.toString(array));
// array = new int[]{4,5,6}; // 编译报错:Error:(19, 9) java: 无法为最终变量array分配值
}

String设计成不可变的原因

方便实现字符串对象池 ,如果String类可变,那么对象池就需要考虑拷贝的问题了

不可变对象的线程是安全的

不可变对象更方便缓存hash code,作为key的时候可以更高效的保存到HashMap中

字符串的修改

对于String类对象进行修改,它是不能直接修改的,会创建新的对象,效率就会变得地下,特别是在循环中使用的时候。

public static void main(String[] args) {
        String s = "hello";
        s = s + "word";
        System.out.println(s);
    }

我们可以通过cmd观察底层实现逻辑:

通过观察,发现它在这中间创建了一些临时变量,因此效率地下

public static void main2(String[] args) {
        long start = System.currentTimeMillis();
        String s = "";
        for(int i = 0; i < 10000; ++i){
            s += i;
        } long end = System.currentTimeMillis();
        System.out.println(end - start);
        start = System.currentTimeMillis();
        StringBuffer sbf = new StringBuffer("");
        for(int i = 0; i < 10000; ++i){
            sbf.append(i);
        } end = System.currentTimeMillis();
        System.out.println(end - start);
        start = System.currentTimeMillis();
        StringBuilder sbd = new StringBuilder();
        for(int i = 0; i < 10000; ++i){
            sbd.append(i);
        } end = System.currentTimeMillis();
        System.out.println(end - start);
    }

通过结果就可以发现他们相差的时间巨大,我们看看底层的执行逻辑:

因此,我们需要尽量避免对String类的直接需要,要修改尽量使用StringBuffer和StringBuilder

StringBuilder和StringBuffer

因为String类的不可改变性,为了方便字符串的修改,java中提供了StringBuilder和Stringbuffer两个类。他们两个类的功能差不多相同。方法和String其实也差不多

    public static void main(String[] args) {
        StringBuilder sb1 = new StringBuilder("hello");
        StringBuilder sb2 = sb1;
// 追加:即尾插-->字符、字符串、整形数字
        sb1.append(' '); // hello
        sb1.append("world"); // hello world
        sb1.append(123); // hello world123
        System.out.println(sb1); // hello world123
        System.out.println(sb1 == sb2); // true
        System.out.println(sb1.charAt(0)); // 获取0号位上的字符 h
        System.out.println(sb1.length()); // 获取字符串的有效长度14
        System.out.println(sb1.capacity()); // 获取底层数组的总大小
        sb1.setCharAt(0, 'H'); // 设置任意位置的字符 Hello world123
        sb1.insert(0, "Hello world!!!"); // Hello world!!!Hello world123
        System.out.println(sb1);
        System.out.println(sb1.indexOf("Hello")); // 获取Hello第一次出现的位置
        System.out.println(sb1.lastIndexOf("hello")); // 获取hello最后一次出现的位置
        sb1.deleteCharAt(0); // 删除首字符
        sb1.delete(0,5); // 删除[0, 5)范围内的字符
        String str = sb1.substring(0, 5); // 截取[0, 5)区间中的字符以String的方式返回
        System.out.println(str);
        sb1.reverse(); // 字符串逆转
        str = sb1.toString(); // 将StringBuffer以String的方式返回
        System.out.println(str);
    }

String和StringBuffer,StringBuilder的区别就是前者不能修改,后两者可以修改。需要频繁修改字符串可以考虑使用后两者。

这里需要注意几点:

String类和Stringbuffer,StringBuilder不能直接转化

String转换为StringBilder,StringBuffer需要使用后者的构造方法或者append()方法、

StringBuilder和Stringbuffer转换为String需要调用toString方法

String,StringBuilder,StringBuffer的区别

String的内容不可以修改,而StringBuilder和StringBuffer可以修改内容

他们的大部分功能是相同的

StringBuffer是同步操作,线程是安全的;StringBuilder未同步操作,线程是不安全的

因为StringBuffer需要不断的关锁开锁,性能略低于StringBuilder


更多推荐

翻牌闯关游戏

翻牌闯关游戏3关:关卡由少至多12格、20格、30格图案:12个玩法:点击两张卡牌,图案一到即可消除掉记忆时长(秒):memoryDurationTime:5可配置:默认5提示游戏玩法:showTipsFlag:1可配置:1:判断localStorage值,仅一次提示游戏玩法2:每次游戏第一关(12格关卡)都提示游戏玩

基于Java演唱会购票系统设计实现(源码+lw+部署文档+讲解等)

博主介绍:✌全网粉丝30W+,csdn特邀作者、博客专家、CSDN新星计划导师、Java领域优质创作者,博客之星、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战✌🍅文末获取源码联系🍅👇🏻精彩专栏推荐订阅👇🏻不然下次找不到哟2022-2024年最全的计算机软件毕业设计选题

JavaScript知识系列(3)每天10个小知识点

目录系列文章目录JavaScript知识系列(1)每天10个小知识点JavaScript知识系列(2)每天10个小知识点知识点**20.AJAX**的概念、作用、原理、特性、优点、缺点、区别、使用场景**,实现一个AJAX请求****21.尾调用**的概念、作用、原理、特性、优点、缺点、区别、使用场景**22.ES6模

互联网数字化管理升级,制造企业一站式智能管理,可定制-亿发

在互联网时代,传统机械制造企业面临着未有的挑战和机遇。信息化管理水平成为企业竞争力的关键因素。然而,许多制造企业在信息化管理中常常陷入以下三大问题:1、盲目随潮流,缺乏总体规划互联网时代,科技发展日新月异,让制造企业感到需要跟上潮流。然而,盲目跟风而行往往导致了技术的不合理使用和资源的浪费。解决这个问题的关键在于建立明

运维:Centos7安装解压版mysql5.7

目录1、卸载Centos7默认自带的mariadb数据库,避免冲突2、下载解压版mysql并安装3、配置mysql4、mysql客户端访问MySQL是一种开源的关系型数据库管理系统(RDBMS),它具有许多优点和一些缺点。以下是MySQL的主要优缺点:优点1.开源免费:MySQL是开源软件,可以免费使用,并且有大量的社

机器视觉常见的问题及解决

应该怎样选择相机?选择相机却往往刻不容缓的的问题摆在机器视觉工程师面前,因此,选择相机了解以下几个方面问题:通常您首先需要知道系统精度要求和相机分辨率,可以通过公式:X方向系统精度(X方向像素值)=视野范围(X方向)/CCD芯片像素数量(X方向);Y方向系统精度(Y方向像素值)=视野范围(Y方向)/CCD芯片像素数量(

面试失败的反思:如何从错误中吸取教训

🌷🍁博主猫头虎(🐅🐾)带您GotoNewWorld✨🍁🦄博客首页——🐅🐾猫头虎的博客🎐🐳《面试题大全专栏》🦕文章图文并茂🦖生动形象🐅简单易学!欢迎大家来踩踩~🌺🌊《IDEA开发秘籍专栏》🐾学会IDEA常用操作,工作效率翻倍~💐🌊《100天精通Golang(基础入门篇)》🐅学会Gol

服务器搭建(TCP套接字)-epoll版(服务端)

epoll是一种在Linux系统上用于高效事件驱动编程的I/O多路复用机制。它相比于传统的select和poll函数具有更好的性能和扩展性。epoll的主要特点和原理:1、事件驱动:epoll是基于事件驱动的模型,它通过监听事件来触发相应的回调函数,而不是像传统的阻塞模型那样持续轮询。这样可以避免无效的轮询操作,提高效

开源项目在面试中的作用:如何用你的贡献加分

🌷🍁博主猫头虎(🐅🐾)带您GotoNewWorld✨🍁🦄博客首页——🐅🐾猫头虎的博客🎐🐳《面试题大全专栏》🦕文章图文并茂🦖生动形象🐅简单易学!欢迎大家来踩踩~🌺🌊《IDEA开发秘籍专栏》🐾学会IDEA常用操作,工作效率翻倍~💐🌊《100天精通Golang(基础入门篇)》🐅学会Gol

STM32F4X UCOSIII 消息队列

STM32F4XUCOSIII消息队列消息队列消息队列的作用消息队列工作机制消息队列创建消息发送消息发送模式FIFO(先进先出)LIFO(后进先出)消息接收消息队列删除消息队列常用函数消息队列创建函数消息队列发送函数消息队列接收函数消息队列删除函数UCOSIII消息队列例程消息队列消息队列的作用消息队列是一种常用于任务

全渠道零售趋势——现代零售业成功的关键

在快节奏的零售时代,无论通过什么渠道购物,消费者越来越习惯无缝购物体验。因此,保持线上和线下购物体验的一致性有助于在品牌与购物者之间建立信任,这也是每个零售商的首要任务。在本文中,我们将介绍现代零售业成功的关键——全渠道零售趋势,帮助零售商实现现代零售业的成功。什么是全渠道零售体验?全渠道零售是一种使零售商能够在所有渠

热文推荐