初始Java篇(JavaSE基础语法)(8)认识String类(上)

找往期文章包括但不限于本期文章中不懂的知识点:

个人主页:我要学编程(ಥ_ಥ)-CSDN博客

所属专栏:JavaSE

简单介绍:在C语言中已经涉及到字符串了,但是在C语言中要表示字符串只能使用字符数组或者字符指针,可以使用标准库提供的字符串系列函数完成大部分操作,但是这种将数据和操作数据方法分离开的方式不符合面相对象的思想,而字符串应用又非常广泛,因此Java语言专门提供了String类。

常用操作String的方法

字符串构造

String类提供的构造方式非常多,常用的就以下三种:

方式一:直接赋值。

String s1 = "Hello";

方式二:使用创建对象的方法。

 String s2 = new String("Hello");

方式三:在方法二的基础上,传入字符数组作为参数。 

char[] ch = {'H','e','l','l','o'};
String s3 = new String(ch);

示例: 

上面三种构造情况,用专业术语总结就是:使用常量串构造,new Sting对象,使用字符数组进行构造。

注意:

1. 我们在前面简单学习了一下String类,它是一个引用数据类型,内部存储的是一个地址,而不是字符串本身。

2. String类的字符串常量是储存在一个名为value的字符数组中的。

3. Java中" "(双引号)里的内容都被认作是String类型对象。简单理解就是字符串。

String对象的比较

下面我们就来深入理解为什么==比较的不是String对象的内容?

public class Test {
    public static void main(String[] args) {
        String s1 = new String("Hello");
        String s2 = new String("Hello");
    }
}

因为数组也是对象,因此也是在堆区中的。只不过为了更好的观察,我就写了两个堆区,实际上只有一个堆区。 

当然,我们也可以通过调试来观察。

 那么要怎么比较呢?其实我们在多态的学习时,已经初步的学习过了。

equals方法——比较对象的内容

equals方法比较的对象里面的内容是啥?比如,我们刚刚写的s1与s2,这两个引用的对象是不一样的,但是这两个引用的对象的内容却是一样的,都是Hello。

注意:

String类重写了父类Object中equals方法,Object中equals默认按照==比较,String重写equals方法后,按照如下规则进行比较 :

1. 先检测this 和 anObject 是否为同一个对象比较,如果是返回true。也就是上面的s1与s2所引用的对象。

2. 检测anObject是否为String类型的对象,如果是继续比较,否则返回false。

3. this和anObject两个字符串的长度是否相同,是继续比较,否则返回false。

4. 按照字典序,从前往后逐个字符进行比较,如果是就返回true,否则就返回false。

注意:

如果上面我们在比较的字符串引用是直接赋值创建的,那么使用==比较的最终结果就是true。

这里就会有小伙伴疑惑这是为啥呢?难道这两个引用所指向的对象是一样的?没错,这两个引用所指向的对象就是一个对象。

在Java中,字符串字面量(例如 "Hello")是存储在字符串常量池中。当你使用 String s1 = "Hello";

和 String s1 = "Hello"; 这种方式声明字符串时,实际上s1和s2都指向了常量池中同一个"Hello"实例。因此,使用 s1 == s2 比较时,结果都是 true,因为它们引用的是同一个对象。

然而,当你使用 String s1 = new String("Hello"); 方式创建字符串时,会创建一个新的字符串对象,即使内容相同,这个新对象也会在堆上分配独立的内存空间,而不是从字符串常量池中获取。所以,String s1 = new String("Hello"); 和 String s1 = new String("Hello"); 分别创建了两个不同的字符串对象,尽管它们的内容都是 "Hello",但它们在内存中的地址不同。因此,使用 s1 == s2 比较时结果为 false,因为它们指向不同的对象实例。但是,如果你使用 s1.equals(s2) 方法进行比较,结果会是 true,因为 equals 方法比较的是对象的内容,而不是它们的引用。

compareTo——比较的是字符串大小

compareTo和C语言中的strcmp函数的功能是相似的。都是比较字符串的大小。既然是比较大小,那么返回的就是一个 int 类型的值。

比较规则:

1. 先按照字典次序大小比较,如果出现不等的字符,直接返回这两个字符的大小差值。

2. 如果前k个字符相等(k为两个字符长度最小值),返回值两个字符串长度差值(与strcmp的差异)

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("abg");
        
        //长度不同,返回的结果就是长度的差值
        System.out.println(s1.compareTo(s2));
        //长度一样,返回的就是一个一个的结果(都一样,返回0)
        System.out.println(s1.compareTo(s3));
        //长度一样,返回的就是一个一个的结果(c和g不同,返回的就是c-g的值)
        System.out.println(s1.compareTo(s4));
    }
}

compareToIgnoreCase——忽略大小写的比较

和compareTo的比较方法一样,但是是忽略大小写的比较

字符串查找 

字符串查找也是字符串中非常常见的操作,String类提供的常用查找的方法:

方法功能
char charAt(int index)返回index位置上字符,如果index为负数或者越界,抛出 IndexOutOfBoundsException 异常
int indexOf(int ch)返回ch第一次出现的位置,没有返回-1
int indexOf(int ch, int fromIndex)从fromIndex位置开始找ch第一次出现的位置,没有返回-1
int indexOf(String str)返回str第一次出现的位置,没有返回-1
int indexOf(String str, int fromIndex)从fromIndex位置开始找str第一次出现的位置,没有返回-1
int lastIndexOf(int ch)从后往前找,返回ch第一次出现的位置,没有返回-1
int lastIndexOf(int ch, int fromIndex)从fromIndex位置开始找,从后往前找ch第一次出现的位置,没有返回-1
int lastIndexOf(String str)从后往前找,返回str第一次出现的位置,没有返回-1
int lastIndexOf(String str, int fromIndex)从fromIndex位置开始找,从后往前找str第一次出现的位置,没有返回-1

示例:

public class Test {
    public static void main(String[] args) {
        String s = "aaabbbcccaaabbbccc";
        System.out.println(s.charAt(3));//获取下标为3位置的字符
        System.out.println(s.indexOf(97));//返回ASCII码值为97的字符在s中第一次出现的位置
        System.out.println(s.indexOf(97,5));//从下标为5的位置的开始找
        System.out.println(s.indexOf("cc"));//返回的是"cc"在s1中第一次出现的位置
        //而带有last前缀的,则是从字符串的末尾开始寻找
    }
}

注意:

1. 字符串是的下标是从0开始的和数组是一致的。 

2. 上述方法都是实例方法,也就是我们得先创建一个类的实例对象才行。

字符串的转化

数值和字符串转化

valueOf方法  

从上面,我们可以看出来:String有很多重载的 valueOf 方法。

既然数值可以转换为字符串,那么字符串也可以转换位数值。

parse+包装类 

整数变成字符串还有一种方法:整数+字符串——>字符串。

下面我们就来看看一个神奇的代码:

public class Test {
    public static void main(String[] args) {
        //前面是简单的数字运算,但是后面加了字符串,因此结果就变成了一个字符串
        System.out.println(99+1+" Hello");
        //前面是字符串,虽然后面是数字相加,但是按照执行顺序的话是就是字符串加数字,因此最终的结果就是把数字加到了字符串后面
        System.out.println("Hello "+99+1);
        //这里加了括号,改变了执行顺序,是先执行数字运算,再和字符串相加
        System.out.println("Hello "+(99+1));
    }
} 
       

因此,这个方法虽然不够正规,但是我们也是可以了解的。 

其实,从这里我们也就可以看出来,Java当中只要知道使用一个类其中一个方法,那么其他的重载方法也就知道怎么使用了。 

大小写转换

public class Test {
    public static void main(String[] args) {
        String s = "Hello";
        System.out.println(s.toUpperCase());
        System.out.println(s.toLowerCase());
    }
}

toUpper——就是将字符串中所有字母全部变成大写。

toLower——就是把字符串中所有字母全部变成小写。

字符串转数组 

把字符串的内容转换为数组内容,并且一个一个的输出。

格式化

public class Test {
    public static void main(String[] args) {
        //因为这里变量还没有创建完,因此不适合用变量名来调用这个format方法
        //这里就两个参数,应该是按照什么样的格式,一个是传给格式的参数
        String s = String.format("%d-%d-%d",2024,5,10);
        System.out.println(s);
    }
}

字符串替换 

使用一个指定的字符串替换掉已有的字符串数据,可用的方法如下:

方法功能
String replaceAll(String regex, String replacement)替换所有的指定内容
String replaceFirst(String regex, String replacement)替换遇到的首个符合要求内容(符合要替换的内容)

注意:由于字符串是不可变对象,替换不会修改当前字符串,而是产生一个新的字符串对象,给到我们原来的引用。 (至于过程,我们后面会学习这个)

示例:

字符串拆分

可以将一个完整的字符串按照指定的分隔符划分为若干个子字符串。

可用方法如下:

方法功能
String[] split(String regex)将字符串按照给的字符串全部拆分
String[] split(String regex, int limit)将字符串以给的字符串的格式,拆分为limit组

示例: 

public class Test {
    public static void main(String[] args) {
        String s = "I say Hello World";
        //根据 字符串空格 来拆分这个字符串,变成若干个子字符串,并且用数组来接收
        String[] tmp = s.split(" ");
        for (String x : tmp) {
            System.out.println(x);
        }
    }
}

public class Test {
    public static void main(String[] args) {
        String s = "I say Hello World";
        //根据 字符串空格 来拆分这个字符串,变成两个子字符串,并且用数组来接收
        String[] tmp = s.split(" ",2);
        for (String x : tmp) {
            System.out.println(x);
        }
    }
}

上面是正常的拆分,但是还有些是不能正常来拆分的。就像下面这样

那么怎样才能正常划分呢?要加上转义字符才可以。

规则:

1. 字符串 "|"   "*"  "+"  "."  都得加上转义字符,前面加上 "\\" 。

2. 而如果是 "\\" ,那么就得写成 "\\\\" 。也就是两个 \\ 表示一个 \ ,而4个 \ 就表示两个 \ 。

3. 如果一个字符串中有多个分隔符,可以用 "|" 作为连字符.。

第二点想要表达的意思是如果原字符串中存在 \\ (有时候不存在一个 \ 的情况,例如在数字中 \ddd,这就变成了一个转义字符了,因此这时候如果我们要表述一个 \ ,就得用用两个 \\ ),而我们想要以这个来作为划分标准的话,怎么办呢?用两个 \\ ,肯定是不行的,因为这是表示一个 \ ,因此,我们就得用 \\ 表示一个 \ ,因此 \\\\ 就表示 \\ 了。

而第三点是啥意思呢?就是我们想要多个标准来划分字符串的话,那那个标准怎么写呢?如下所示:

上面这个就包含了我们上面的多种规则结合在一起。

好啦!本期 初始Java篇(JavaSE基础语法)(8)认识String类(上)的学习之旅就到此结束了!我们下一期再一起学习吧!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/616270.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

关于画图-一次性搞定各类高级论文作图及配色

关于画图-一次性搞定各类高级论文作图及配色 图(Figure)可以让各类论文的结果更加直观,有时候一张图片比一大段文字更有说服力。 但许多新手作者可能会有一连串的疑惑:数据这么多,什么时候该做什么类型的图&#xff…

alist网盘自动同步

alist网盘自动同步 alist可以设置目录定时转存到各个网盘,做到夸网盘,多备份的效果可以将自己挂载的alist 下的各个目录相互间进行同步,原理是采用alist原始api调用执行,同步原理是匹配文件名称是否相同,相同会跳过同…

ElasticSearch 8.X 源码导入idea并配置环境启动调试(mac环境)

主要是用于自己记录配置流程 环境 IntelliJ IDEA 2024.1.1 (Community Edition) jdk17(可以安装jenv管理) macos 14.4.1 gradle 8.5 资源准备 先在官网下载elasticsearch源码(GitHub - elastic/elasticsearch: Free and Open, Distrib…

事务的基础

📝个人主页:五敷有你 🔥系列专栏:面经 ⛺️稳中求进,晒太阳 事务的基础 1)事务 事务是:一组操作的集合 ,他是不可分割的工作单位。事务会把所有操作作为一个整体一起向系统提…

【初阶数据结构】栈

目录 栈的概念及结构栈的实现栈的结构栈的初始化栈的销毁入栈出栈取栈顶元素判断栈是否为空取栈中元素个数代码测试 完整代码Stack.hStack.ctest.c 栈的概念及结构 栈:是一种特殊的线性表,它只允许在固定的一端进行插入和删除元素的操作。   栈顶&…

远程桌面如何连接?

远程桌面连接是一种可以在不同地点之间共享电脑桌面的技术。通过远程桌面连接,用户可以在远程的计算机上操作另一台计算机,就像是直接坐在前者的前面一样。这种技术可以帮助用户解决在不同地点之间共享数据、协同办公、设备管理等问题。 【天联】的使用场…

C语言实现猜数字小游戏

1.随机数生成 要想实现猜数字小游戏,依赖于随机数的生成 1.1 rand()函数 这个函数是用来生成随机数的,返回值是正整数,他的值的范围是0到rand_max之间的,rand_max的值在大多数编译器上面是32767,rand()函数的使用必…

前端 | TED打卡号分类查询

文章目录 &#x1f4da;实现效果&#x1f4da;模块实现解析&#x1f407;html&#x1f407;css&#x1f407;javascript &#x1f4da;实现效果 提供完整TED打卡号对应TED标题的查询列表 根据分类按需查询 &#x1f4da;模块实现解析 &#x1f407;html 搭框架<div cl…

软件测试报告(交付文档支撑word原件)

软件测试报告在软件开发过程中起着至关重要的作用&#xff0c;主要有以下几个主要原因&#xff1a; 1、确保软件质量 2、提供决策支持 3、记录测试过程和结果 4、促进沟通和协作 5、符合标准和法规要求 6、改进测试流程和策略 7、降低风险 软件开发全套资料获取进主页或者本文…

芯课堂 | 如何使用SWM系列产品的TIMER功能(一)

01 TIMER简介 TIMER是一种定时器工具&#xff0c;用来在一个后台线程计划执行指定任务。它可以计划执行一个任务一次或反复多次。 我司的SWM系列单片机提供的TIMER个数和功能有些微差别&#xff0c;为了让您更加简单的使用这一功能&#xff0c;下面将以SWM190为例&#xff0…

Linux/Brainfuck

Brainfuck Enumeration Nmap 扫描发现对外开放了 22&#xff0c;25&#xff0c;110&#xff0c;143&#xff0c;443 五个端口&#xff0c;使用 nmap 扫描端口详细信息 ┌──(kali㉿kali)-[~/vegetable/HTB/Insane] └─$ nmap -sC -sV -p 22,25,110,143,443 -oA nmap 10.10…

制作跳动的爱心网页效果

html <!DOCTYPE html> <html lang"en"> <head> <meta charset"UTF-8"> <meta name"viewport" content"widthdevice-width, initial-scale1.0"> <title>跳动的爱心</title> <link rel&q…

##16 利用迁移学习和微调提升深度学习模型性能

文章目录 前言什么是迁移学习&#xff1f;迁移学习的主要优势迁移学习的策略1. 特征提取器2. 微调 在PyTorch中实现迁移学习环境设置加载预训练模型修改模型以适应新任务训练和微调模型迁移学习的示例应用 结论 前言 在深度学习的多个领域中&#xff0c;迁移学习技术已经成为了…

每日一题9:Pandas-填充缺失值

一、每日一题 DataFrame products --------------------- | Column Name | Type | --------------------- | name | object | | quantity | int | | price | int | ---------------------编写一个解决方案&#xff0c;在 quantity 列中将缺失的值 编…

Axure PR 10 制作顶部下拉三级菜单和侧边三级菜单教程和源码

在线预览地址&#xff1a;Untitled Document 2.侧边三级下拉菜单 在线预览地址&#xff1a;Untitled Document 文件包和教程下载地址&#xff1a;https://pan.quark.cn/s/77e55945bfa4 程序员必备资源网站&#xff1a;天梦星服务平台 (tmxkj.top)

TMS320F280049 CLB模块--FSM(3)

功能框图 FSM有效状态机内部框图如下图所示&#xff0c;可以看到内部有S0 / S1两个状态和下一状态的跳转查找表。还有个输出查找表。 下图是FSM LUT的示意框图。FSM还可以工作在3输入或4输入的查找表模式下。对于输入&#xff0c;EXTRA_EXT_IN1/0可以替换S0/1。 寄存器 参考文…

Ubuntu 22.04: VS Code 配置 C++ 编译及 CMake

一、VS Code 安装以及 C 编译环境配置 1. 在 Ubuntu 中安装 VS Code 笔者直接在 Ubuntu Software 中心安装 VS Code。也可以从VS Code官网下载 deb&#xff0c;解压 dpkg -i 安装。 2. VS Code 中配置 g/gcc 1) 安装 C/C 扩展 &#xff08;CtrlShiftX&#xff09; 2&#x…

【Unity Shader入门精要 第6章】基础光照(一)

1. 什么是光照模型 光照原理 在真实世界中&#xff0c;我们能够看到物体&#xff0c;是由于眼睛接收到了来自观察目标的光。这里面包括两种情况&#xff1a;一部分是观察目标本身发出的光&#xff08;自发光&#xff09;直接进入我们的眼睛&#xff0c;另一部分是其他物体&am…

感知机和神经网络

引入 什么是神经网络&#xff1f; 我们今天学习的神经网络&#xff0c;不是人或动物的神经网络&#xff0c;但是又是模仿人和动物的神经网络而定制的神经系统&#xff0c;特别是大脑和神经中枢&#xff0c;定制的系统是一种数学模型或计算机模型&#xff0c;神经网络由大量的人…

重写muduo之TcpServer

目录 1、Callbacks.h 2、TcpServer.h 3、TcpServer.cc 1、Callbacks.h 回调操作 #pragma once#include <memory> #include <functional>class Buffer; class TcpConnection;using TcpConnectionPtrstd::shared_ptr<TcpConnection>; using ConnectionCall…