day10-字符串

目录

  • 字符串
    • 1、API 和 API 帮助文档
    • 2、String概述
    • 3、String构造方法代码实现 和 内存分析
      • 3.1 创建String对象的两种方式
      • 3.2 Java的内存模型
    • 4、字符串的比较
      • 4.1 ==号的作用
      • 4.2 equals方法的作用
    • 练习
      • 5、用户登录
      • 6、遍历字符串和统计字符个数
      • 7、字符串拼接和翻转
      • 8、较难练习-金额转换
      • 9、手机号屏蔽
    • 10、StringBuilder
    • 11、StringJoiner
    • 12、字符串相关类的底层原理
      • 扩展底层原理1:字符串存储的内存原理
      • 扩展底层原理2:==号比较的到底是什么?
      • 扩展底层原理3:字符串拼接的底层原理
      • 扩展底层原理4:StringBuilder提高效率原理图
      • 扩展底层原理5:StringBuilder源码分析
    • 练习
      • 13、较难练习-罗马数字的两种写法
      • 14、调整字符串的内容并比较
      • 15、后续练习思路分析

字符串

1、API 和 API 帮助文档

通过网盘分享的文件:资料
链接: https://pan.baidu.com/s/1jlQvGN1PHiud6NXhSYpIaA?pwd=reyn 提取码: reyn

什么是API?
​API (Application Programming Interface) :应用程序编程接口。
简单理解: API就是别人已经写好的东西,我们不需要自己编写,直接使用即可。

Java API:指的就是JDK中提供的各种功能的Java类。
这些类将底层的实现封装了起来,我们不需要关心这些类是如何实现的,只需要学习这些类如何使用即可,我们可以通过帮助文档来学习这些API如何使用。

**API帮助文档:**帮助开发人员更好的使用API和查询API的一个工具。

如何使用API帮助文档

  • ①打开API帮助文档
    在这里插入图片描述

  • ②点击显示,找到索引选项卡中的输入框
    在这里插入图片描述

  • ③在输入框中输入 类名 并点击显示
    在这里插入图片描述

  • ④查看类所在的包
    在这里插入图片描述

  • ⑤查看类的描述
    在这里插入图片描述

  • ⑥查看构造方法
    在这里插入图片描述

  • ⑦查看成员方法
    在这里插入图片描述

🍊API文档练习:
需求:按照帮助文档的使用步骤学习 Scanner 类的使用,并实现接收键盘录入一个小数,最后输出在控制台.

public class ScannerDemo1 {public static void main(String[] args) {//1.创建对象Scanner sc = new Scanner(System.in);System.out.println("请输入一个小数");//2.接收一个小数double result = sc.nextDouble();//3.输出打印System.out.println(result);}
}

2、String概述

String 类代表字符串,Java 程序中的所有字符串文字(例如“abc”)都被实现为此类的实例。也就是说,Java 程序中所有的双引号字符串,都是 String 类的对象。
String 类在 java.lang 包下,所以使用的时候不需要导包!

String类的特点

  • 字符串不可变,它们的值在创建后不能被更改
  • 虽然 String 的值是不可变的,但是它们可以被共享
  • 字符串效果上相当于字符数组( char[] ),但是底层原理是字节数组( byte[] )
String name = "阿伟";
String schoolName = "黑马程序员";
System.out.println(name + schoolName);
// 三个字符串
String name = "阿伟";
name = "三连.阿伟";
// 两个字符串
// 不是改变了字符串的内容,而是创建了一个新的字符串,
// 把新的字符串复制给了前面的变量 name,此时并没有改变第一个字符串里面的内容

3、String构造方法代码实现 和 内存分析

3.1 创建String对象的两种方式

1、直接赋值

2、通过构造方法

构造方法说明
public String()创建一个空白字符串对象,不含有任何内容
public String(String original)根据传入的字符串,创建字符串对象
public String(char[] chs)根据字符数组的内容,来创建字符串对象
public String(byte[] bys)根据字节数组的内容,来创建字符串对象

🌰代码示例:

public class StringDemo1 {public static void main(String[] args) {//1.使用直接赋值的方式获取一个字符串对象String s1 = "abc";System.out.println(s1);//abc//2.使用new的方式来获取一个字符串对象//public String():创建一个空白字符串对象,不含有任何内容//空参构造:可以获取一个空白的字符串对象String s2 = new String();System.out.println("@" + s2 + "!");//"@!"//传递一个字符串,根据传递的字符串内容再创建一个新的字符串对象//这种方式只要知道有它存在就好了String s3 = new String("abc");System.out.println(s3);//public String(char[] chs):根据字符数组的内容,来创建字符串对象//传递一个字符数组,根据字符数组的内容再创建一个新的字符串对象//需求:我要修改字符串的内容。  abc --> Qbc//abc -->  {'a','b','c'}  -->  {'Q','b','c'} --> "Qbc"char[] chs = {'a', 'b', 'c', 'd'};String s4 = new String(chs);System.out.println(s4);//abcd//public String(byte[] bys):根据字节数组的内容,来创建字符串对象//传递一个字节数组,根据字节数组的内容再创建一个新的字符串对象//应用场景:以后在网络当中传输的数据其实都是字节信息//我们一般要把字节信息进行转换,转成字符串,此时就要用到这个构造了。byte[] bytes = {97, 98, 99, 100};String s5 = new String(bytes);System.out.println(s5);//abcd}
}

通过构造方法创建字符串中,前面两种方法少用,后面两种方法以后用得上。

3.2 Java的内存模型

在这里插入图片描述

串池,可以理解成 字符串常量池 ,但是只有 直接赋值 的字符串才存在这个 串池 中。

🍎直接赋值
在这里插入图片描述

给 s1 赋值时系统会先去串池中观察,有没有abc,没有的话就会创建一个新的abc,然后把她得 地址值 赋值给 s1。
给 s2 赋值时,串池已经有一个abc了,它就不会创建abc,而是会复用已经存在的abc。

⭐️小结:
当使用双引号直接赋值时,系统会检查该字符串在串池中是否存在:
不存在:创建新的
存在:复用

🍎手动new出来的
在这里插入图片描述

以字符数组为例,给 s1 赋值时遇到 new 关键字,在堆里面创建了一个小空间,这个小空间里面的内容就是字符数组的内容abc,然后再把这个小空间的地址值,赋给 s1,s1 通过地址值就能找到创建出来的字符串了。
给 s2 赋值时遇到 new 关键字,其实就是把上述过程重复了一遍,假设开辟出来的第三个小空间的地址值是 0x0033,此时就会把 0x0033 赋给 s2 ,s2 通过它就能找到第二个字符串对象。

⭐️每new一次就会开辟一个小空间,不会复用。

4、字符串的比较

4.1 ==号的作用

  • 比较基本数据类型:比较的是具体的值
  • 比较引用数据类型:比较的是对象地址值
int a = 10;
int b = 20;
System.out.println(a == b); // falseString s1 = new String("abc");
String s2 = new String("abc");
System.out.println(s1 == s2); // false

4.2 equals方法的作用

  • 方法介绍
    • public boolean equals(String s)
      比较两个字符串内容是否相同、区分大小
    • public boolean equalslgnoreCase(String s)
      比较两个字符串内容是否相同、不区分大小写

🌰代码示例:

public class StringDemo2 {public static void main(String[] args) {//1.创建两个字符串对象String s1 = new String("abc");String s2 = "Abc";//2.==号比较//基本数据类型:比的是数据值//引用数据类型:比的是地址值System.out.println(s1 == s2);//false//3.比较字符串对象中的内容是否相等boolean result1 = s1.equals(s2);System.out.println(result1); // false//4.比较字符串对象中的内容是否相等,忽略大小写//1 一 壹 这不行//忽略大小写只能是英文状态下的a Aboolean result2 = s1.equalsIgnoreCase(s2);System.out.println(result2);//true}
}

练习

5、用户登录

需求:
已知用户名和密码,请用程序实现模拟用户登录。总共给三次机会,登录之后,给出相应的提示。

📖代码示例:

public class Test1登录案例 {public static void main(String[] args) {//1.定义两个变量用来记录正确的用户名和密码String rightUsername = "itheima";String rightPassword = "1234qwer";//2.键盘录入用户名和密码//ctrl + alt + T 选择包裹方式for (int i = 0; i < 3; i++) {//0 1 2Scanner sc = new Scanner(System.in);System.out.println("请输入用户名");String username = sc.next();System.out.println("请输入密码");String password = sc.next();//3.判断比较if (username.equals(rightUsername) && password.equals(rightPassword)) {System.out.println("登录成功");//如果正确,循环结束break;} else {//最后一次机会if(i == 2){System.out.println("账户" + username + "被锁定,请联系黑马程序员官方小姐姐:XXXXXXX");}else{//不是最后一次机会System.out.println("用户名或密码错误,登录失败,还剩下" + (2 - i) + "次机会");//2 1 0}}}}
}

6、遍历字符串和统计字符个数

需求:
键盘录入一个字符串,统计该字符串中大写字母字符,小写字母字符,数字字符出现的次数(不考虑其他字符)。

📖代码示例:

public class Test4统计个数 {public static void main(String[] args) {//键盘录入一个字符串,统计大写,小写,数字出现的次数//1.键盘录入一个字符串Scanner sc = new Scanner(System.in);System.out.println("请输入一个字符串");String str = sc.next();//2.统计 --- 计数器count//此时我要统计的有3样东西,所以要定义3个计数器分别进行统计int bigCount = 0;int smallCount = 0;int numberCount = 0;//得到这个字符串里面每一个字符for (int i = 0; i < str.length(); i++) {//i 表示字符串中的索引//c 表示字符串中的每一个字符char c = str.charAt(i);//对c进行判断if (c >= 'a' && c <= 'z') {smallCount++;}else if(c >= 'A' && c <= 'Z'){bigCount++;}else if(c >= '0' && c <= '9'){numberCount++;}}//3.当循环结束之后,三个变量记录的就是对应的个数System.out.println("大写字符有:" + bigCount + "个");System.out.println("小写字符有:" + smallCount + "个");System.out.println("数字字符有:" + numberCount + "个");}
}

7、字符串拼接和翻转

需求:
定义一个方法,把 int 数组中的数据按照指定的格式拼接成一个字符串返回,调用该方法,并在控制台输出结果。例如,数组为 int[] arr = {1,2,3}; ,执行方法后的输出结果为:[1, 2, 3]。

📖代码示例:

public class Test5数组拼接成字符串 {public static void main(String[] args) {//定义一个方法,把 int 数组中的数据按照指定的格式拼接成一个字符串返回,调用该方法,//并在控制台输出结果。例如,数组为 int[] arr = {1,2,3};//执行方法后的输出结果为:[1, 2, 3]int[] arr = {1, 2, 3, 4, 5};String str = arrToString(arr);System.out.println(str);}//作用:把一个数组变成字符串public static String arrToString(int[] arr) {String s = "";//拼接左括号s = s + "["; //此时是拿着长度为0的字符串,跟[进行拼接,产生一个新的字符串。// *这里不用"s+",直接s ="["也可以,//把新的字符串再赋值给s,此时变量s记录的就是新的字符串"["的地址值//下面我想得到数组里面的每一个元素并进行拼接//那么就需要遍历数组,得到每一个元素才行for (int i = 0; i < arr.length; i++) {if(i == arr.length - 1){//如果是最后一个元素,那么不需要拼接逗号空格s = s + arr[i];}else{//如果不是最后一个元素,需要拼接元素和逗号空格s = s + arr[i] + ", ";}}//等循环结束之后,再拼接最后一个右括号s = s + "]";return s;}//用来遍历数组public static void printArr(int[] arr) {System.out.print("[");for (int i = 0; i < arr.length; i++) {if (i == arr.length - 1) {System.out.print(arr[i]);} else {System.out.print(arr[i] + ", ");}}System.out.println("]");//[1, 2, 3, 4, 5]//我们现在要知道,这个最终结果是怎么来的?//从到右依次打印得来的。}
}

8、较难练习-金额转换

需求:
把2135变成:零佰零拾零万贰仟壹佰叁拾伍元
把789变成:零佰零拾零万零仟柒佰捌拾玖元

📖代码示例:

package com.itheima.stringdemo;import java.util.Scanner;public class StringDemo9 {public static void main(String[] args) {//1.键盘录入一个金额Scanner sc = new Scanner(System.in);int money;while (true) {System.out.println("请录入一个金额");money = sc.nextInt();if (money >= 0 && money <= 9999999) {break;} else {System.out.println("金额无效");}}//定义一个变量用来表示钱的大写String moneyStr = "";//2.得到money里面的每一位数字,再转成中文while (true) {//2135//从右往左获取数据,因为右侧是数据的个位int ge = money % 10;String capitalNumber = getCapitalNumber(ge);//把转换之后的大写拼接到moneyStr当中moneyStr = capitalNumber + moneyStr;//第一次循环 : "伍" + "" = "伍"//第二次循环 : "叁" + "伍" = "叁伍"//去掉刚刚获取的数据money = money / 10;//如果数字上的每一位全部获取到了,那么money记录的就是0,此时循环结束if (money == 0) {break;}}//3.在前面补0,补齐7位int count = 7 - moneyStr.length();for (int i = 0; i < count; i++) {moneyStr = "零" + moneyStr;}System.out.println(moneyStr);//零零零贰壹叁伍//4.插入单位//定义一个数组表示单位String[] arr = {"佰","拾","万","仟","佰","拾","元"};//               零    零   零   贰   壹   叁   伍//遍历moneyStr,依次得到 零    零   零   贰   壹   叁   伍//然后把arr的单位插入进去String result = "";for (int i = 0; i < moneyStr.length(); i++) {char c = moneyStr.charAt(i);//把大写数字和单位拼接到result当中result = result + c + arr[i]; // 零,零佰,零佰零,零佰零拾...}//5.打印最终结果System.out.println(result);}//定义一个方法把数字变成大写的中文//1 -- 壹public static String getCapitalNumber(int number) {//定义数组,让数字跟大写的中文产生一个对应关系String[] arr = {"零", "壹", "贰", "叁", "肆", "伍", "陆", "柒", "捌", "玖"};//返回结果return arr[number];}
}

9、手机号屏蔽

需求:
以字符串的形式从键盘接受一个手机号,将中间四位号码屏蔽。
最终效果为:131****9468

📖代码示例:

public class Test8手机号屏蔽 {public static void main(String[] args) {/*以字符串的形式从键盘接受一个手机号,将中间四位号码屏蔽最终效果为:131****9468*///1.键盘录入一个手机号码Scanner sc = new Scanner(System.in);System.out.println("请输入手机号码");String phoneNumber = sc.next();//13112349408//2.截取手机号码中的前三位String star = phoneNumber.substring(0, 3);//3.截取手机号码中的最后四位//此时我用substring方法,是用1个参数的,还是两个参数的?1个参数的会更好//因为现在我要截取到最后,所以建议使用1个参数的。String end = phoneNumber.substring(7); // *从第 7 位到最后//4.拼接String result = star + "****" + end;System.out.println(result);}
}

10、StringBuilder

StringBuilder 可以看成是一个容器,创建之后里面的内容是可变的。

  • 作用:提高字符串的操作效率。
String s1 = "aaa";
String s2 = "bbb";
String s3 = "ccc";
String s4 = "ddd";
String s5 = "eee";String s6 = s1 + s2 + s3 + s4 + s5;
// 在拼接过程中会产生很多没有用的字符串// 用StringBuilder,在拼接过程中不会产生那么多没有用的字符串,所以用StringBuilder效率会更高。

⭐️StringBuilder构造方法

方法名说明
public StringBuilder()创建一个空白可变字符串对象,不含有任何内容
public StringBuilder(String str)根据字符串的内容,来创建可变字符串对象

⭐️StringBuilder常用方法

方法名说明
public StringBuilder append (任意类型)添加数据,并返回对象本身
public StringBuilder reverse()反转容器中的内容
public int length0返回长度(字符出现的个数)
public String toString0通过toString()就可以实现把StringBuilder转换为String

🌰代码示例:

package com.itheima.stringbuilderdemo;public class StringBuilderDemo3 {public static void main(String[] args) {//1.创建对象StringBuilder sb = new StringBuilder("abc");//2.添加元素/*sb.append(1);sb.append(2.3);sb.append(true);*/// abc12.3true//反转sb.reverse();//获取长度int len = sb.length();System.out.println(len);//打印//普及://因为StringBuilder是Java已经写好的类//java在底层对他做了一些特殊处理。//打印对象不是地址值而是属性值。System.out.println(sb);}
}
package com.itheima.stringbuilderdemo;public class StringBuilderDemo4 {public static void main(String[] args) {//1.创建对象StringBuilder sb = new StringBuilder();//2.添加字符串sb.append("aaa").append("bbb").append("ccc").append("ddd");System.out.println(sb);//aaabbbcccddd//3.再把StringBuilder变回字符串String str = sb.toString();System.out.println(str);//aaabbbcccddd}
}

sb 只是一个容器,来帮助我们操作字符串的工具。所以在拼接完之后,还需要把 StringBuilder 变回 字符串 。

⭐️链式编程
当我们在调用一个方法的时候,不需要用变量接收他的结果,可以继续调用其他方法。

🌰代码示例:

package com.itheima.stringbuilderdemo;import java.util.Scanner;public class StringBuilderDemo5 {public static void main(String[] args) {//链式编程://当我们在调用一个方法的时候,不需要用变量接收他的结果,可以继续调用其他方法int len = getString().substring(1).replace("A", "Q").length();System.out.println(len);}public static String getString(){Scanner sc = new Scanner(System.in);System.out.println("请输入一个字符串");String str = sc.next();return str;}
}

链式编程:依赖前一个方法的结果 再去调用后面的方法。
substring 截取完的结果是一个小的字符串,可以继续调用字符串里面的方法 replace (把字符串里面的A替换成Q),然后再调用 length 获取字符串的长度。

public class StringBuilderDemo4 {public static void main(String[] args) {//1.创建对象StringBuilder sb = new StringBuilder();//2.添加字符串sb.append("aaa").append("bbb").append("ccc").append("ddd");System.out.println(sb);//aaabbbcccddd//3.再把StringBuilder变回字符串String str = sb.toString();System.out.println(str);//aaabbbcccddd}
}

🍊练习:对称字符串
需求:
键盘接受一个字符串,程序判断出该字符串是否是对称字符串,并在控制台打印是或不是。

对称字符串:123321、111
非对称字符串:123123

📖代码示例:

public class StringBuilderDemo6 {//使用StringBuilder的场景://1.字符串的拼接//2.字符串的反转public static void main(String[] args) {//1.键盘录入一个字符串Scanner sc = new Scanner(System.in);System.out.println("请输入一个字符串");String str = sc.next();//2.反转键盘录入的字符串String result = new StringBuilder().append(str).reverse().toString();//3.比较if(str.equals(result)){System.out.println("当前字符串是对称字符串");}else{System.out.println("当前字符串不是对称字符串");}}
}

🍊练习:拼接字符串

需求:定义一个方法,把 int 数组中的数据按照指定的格式拼接成一个字符串返回。调用该方法,并在控制台输出结果。
例如:数组为 int[] arr = {1,2,3};
执行方法后的输出结果为:[1, 2, 3]

📖代码示例:

package com.itheima.stringbuilderdemo;public class StringBuilderDemo7 {public static void main(String[] args) {//1.定义数组int[] arr = {1,2,3};//2.调用方法把数组变成字符串String str = arrToString(arr);System.out.println(str);}public static String arrToString(int[] arr){StringBuilder sb = new StringBuilder();sb.append("[");for (int i = 0; i < arr.length; i++) {if(i == arr.length - 1){sb.append(arr[i]);}else{sb.append(arr[i]).append(", ");}}sb.append("]");return sb.toString();}
}

11、StringJoiner

  • StringJoiner跟StringBuilder一样,也可以看成是一个容器,创建之后里面的内容是可变的。
  • 作用:提高字符串的操作效率,而且代码编写特别简洁,但是目前市场上很少有人用。
  • JDK8出现的

⭐️StringJoiner的构造方法

方法名说明
public StringJoiner (间隔符号)创建一个StringJoiner对象,指定拼接时的间隔符号
public StringJoiner (间隔符号,开始符号,结束符号)创建一个StringJoiner对象,指定拼接时的间隔符号、开始符号、结束符号

⭐️StringJoiner的成员方法

方法名说明
public StringJoiner add (添加的内容)添加数据,并返回对象本身
public int length()返回长度(字符出现的个数)
public String toString()返回一个字符串(该字符串就是拼接之后的结果)

🌰代码示例:

package com.itheima.stringjoinerdemo;import java.util.StringJoiner;public class StringJoinerDemo1 {public static void main(String[] args) {//1.创建一个对象,并指定中间的间隔符号StringJoiner sj = new StringJoiner("---");//2.添加元素sj.add("aaa").add("bbb").add("ccc");//3.打印结果System.out.println(sj);//aaa---bbb---ccc}
}
package com.itheima.stringjoinerdemo;import java.util.StringJoiner;public class StringJoinerDemo2 {public static void main(String[] args) {//1.创建对象StringJoiner sj = new StringJoiner(", ","[","]");//2.添加元素sj.add("aaa").add("bbb").add("ccc");int len = sj.length();System.out.println(len);//15 // *逗号后面还有个空格//3.打印System.out.println(sj);//[aaa, bbb, ccc]String str = sj.toString();System.out.println(str);//[aaa, bbb, ccc]}
}

🍎总结:

  1. String
    表示字符串的类,定义了很多操作字符串的方法。

  2. StringBuilder
    一个可变的操作字符串的容器。
    可以高效的拼接字符串,还可以将容器里面的内容反转。

  3. StringJoiner
    JDK8出现的一个可变的操作字符串的容器,可以高效,方便的拼接字符串。
    在拼接的时候,可以制定间隔符号,开始符号,结束符号。

12、字符串相关类的底层原理

扩展底层原理1:字符串存储的内存原理

  • 直接赋值会复用字符串常量池中的东西
  • new出来不会复用,而是开辟一个新的空间

扩展底层原理2:==号比较的到底是什么?

  • 基本数据类型比较数据值
  • 引用数据类型比较地址值

扩展底层原理3:字符串拼接的底层原理

第一种情况:等号的右边没有变量
在这里插入图片描述

第二种情况:等号的右边有变量
🍑画图解释(JDK8以前):
在这里插入图片描述

字符串 和 变量 一次拼接,堆内存中至少会出现两个对象,一个是 StringBuilder对象,一个是 字符串String 对象。
String s2 = s1 + "b";,相当于是,new StringBuilder().append(s1).append("b").toString();,toString的底层原理是 new,所以内存中会有两个对象。

在这里插入图片描述

🍑画图解释(JDK8):
在这里插入图片描述

JDK8 在字符串拼接的时候会做一个 预估。预估成数组,最后再去创建字符串对象。

但是预估也是需要时间的,如果每一行都要进行 字符串和变量的拼接 的话,每一行都要进行预估,再创建数组,再变成字符串。

所以就算JDK优化了,还是一样的浪费时间。字符串拼接的时候有变量参与,在内存中创建了很多对象浪费空间,时间也非常慢。

✅结论:如果很多字符串变量拼接,不要直接+。在底层会创建多个对象,浪费时间,浪费性能。

扩展底层原理4:StringBuilder提高效率原理图

🍑画图解释:在这里插入图片描述

StringBuilder是一个内容可变的容器,我们是把所有的数据往 同一个StringBuilder 里面放的,所以效率会高。

📝常见面试题:
1️⃣
在这里插入图片描述
📚答案:
在这里插入图片描述

2️⃣
在这里插入图片描述
📚答案:
在这里插入图片描述

⭕️字符串原理小结:

扩展底层原理3:字符串拼接的底层原理

  • 如果没有变量参与,都是字符串直接相加,编译之后就是拼接之后的结果,会复用串池中的字符串。
  • 如果有变量参与,每一行拼接的代码,都会在内存中创建新的字符串,浪费内存。

扩展底层原理4:StrinqBuilder提高效率原理图

  • 所有要拼接的内容都会往StringBuilder中放,不会创建很多无用的空间,节约内存.

扩展底层原理5:StringBuilder源码分析

  • 默认创建一个长度为16的字节数组
  • 添加的内容长度小于16,直接存
  • 添加的内容大于16会扩容(原来的容量*2+2)
  • 如果扩容之后还不够,以实际长度为准

🌰代码示例:

package test;import java.util.Scanner;public class A {public static void main(String[] args) {StringBuilder sb = new StringBuilder();// 容量:最多装多少// 长度:已经装了多少System.out.println(sb.capacity()); // 16System.out.println(sb.length()); // 0sb.append("abc");System.out.println(sb.capacity()); // 16System.out.println(sb.length()); // 3sb.append("defghijklmnopqrstuvwxyz");System.out.println(sb.capacity()); // 34System.out.println(sb.length()); // 26sb.append("0123456789");System.out.println(sb.capacity()); // 36System.out.println(sb.length()); // 36}
}

🍎字符串原理小结

扩展底层原理1:字符串存储的内存原理

  • 直接赋值会复用字符串常量池中的东西
  • new出来不会复用,而是开辟一个新的空间

扩展底层原理2:==号比较的到底是什么?

  • 基本数据类型比较数据值
  • 引用数据类型比较地址值

扩展底层原理3:字符串拼接的底层原理

  • 如果没有变量参与,都是字符串直接相加,编译之后就是拼接之后的结果,会复用串池中的字符串。
  • 如果有变量参与,每一行拼接的代码,都会在内存中创建新的字符串,浪费内存。

扩展底层原理4:StrinqBuilder提高效率原理图

  • 所有要拼接的内容都会往StringBuilder中放,不会创建很多无用的空间,节约内存

扩展底层原理5:StringBuilder源码分析

  • 默认创建一个长度为16的字节数组
  • 添加的内容长度小于16,直接存
  • 添加的内容大于16会扩容(原来的容量*2+2)
  • 如果扩容之后还不够,以实际长度为准

练习

13、较难练习-罗马数字的两种写法

键盘录入一个字符串
要求1:长度为小于等于9
要求2:只能是数字
将内容变成罗马数字
下面是阿拉伯数字跟罗马数字的对比关系:
1-1、1-2、1-3、IV-4、V-5、V-6、V-7、V-8、1X-9注意点:
罗马数字里面是没有0的
如果键盘录入的数字包含0,可以变成""(长度为0的字符串)

📖代码示例:

package com.itheima.test;import java.util.Scanner;public class Test1Case1 {public static void main(String[] args) {/* 键盘录入一个字符串,要求1:长度为小于等于9要求2:只能是数字将内容变成罗马数字下面是阿拉伯数字跟罗马数字的对比关系:Ⅰ-1、Ⅱ-2、Ⅲ-3、Ⅳ-4、Ⅴ-5、Ⅵ-6、Ⅶ-7、Ⅷ-8、Ⅸ-9注意点:罗马数字里面是没有0的如果键盘录入的数字包含0,可以变成""(长度为0的字符串)*///1.键盘录入一个字符串//书写Scanner的代码Scanner sc = new Scanner(System.in);String str;while (true) {System.out.println("请输入一个字符串");str = sc.next();//2.校验字符串是否满足规则boolean flag = checkStr(str);if (flag) {break;} else {System.out.println("当前的字符串不符合规则,请重新输入");continue;}}//将内容变成罗马数字//下面是阿拉伯数字跟罗马数字的对比关系://Ⅰ-1、Ⅱ-2、Ⅲ-3、Ⅳ-4、Ⅴ-5、Ⅵ-6、Ⅶ-7、Ⅷ-8、Ⅸ-9//查表法:数字跟数据产生一个对应关系StringBuilder sb = new StringBuilder();for (int i = 0; i < str.length(); i++) {char c = str.charAt(i);int number = c - 48; // 1 2 3 4 5String s = changeLuoMa(number);sb.append(s);}System.out.println(sb);}public static String changeLuoMa(int number) {//定义一个数组,让索引跟罗马数字产生一个对应关系String[] arr = {"", "Ⅰ", "Ⅱ", "Ⅲ", "Ⅳ", "Ⅴ", "Ⅵ", "Ⅶ", "Ⅷ", "Ⅸ"};return arr[number];}public static boolean checkStr(String str) {//123456//要求1:长度为小于等于9if (str.length() > 9) {return false;}//要求2:只能是数字for (int i = 0; i < str.length(); i++) {char c = str.charAt(i);//0~9if (c < '0' || c > '9') {return false;}}//只有当字符串里面所有的字符全都判断完毕了,我才能认为当前的字符串是符合规则return true;}
}
package com.itheima.test;import java.util.Scanner;public class Test1Case2 {public static void main(String[] args) {/* 键盘录入一个字符串,要求1:长度为小于等于9要求2:只能是数字将内容变成罗马数字下面是阿拉伯数字跟罗马数字的对比关系:Ⅰ-1、Ⅱ-2、Ⅲ-3、Ⅳ-4、Ⅴ-5、Ⅵ-6、Ⅶ-7、Ⅷ-8、Ⅸ-9注意点:罗马数字里面是没有0的如果键盘录入的数字包含0,可以变成""(长度为0的字符串)*///1.键盘录入一个字符串//书写Scanner的代码Scanner sc = new Scanner(System.in);String str;while (true) {System.out.println("请输入一个字符串");str = sc.next();//2.校验字符串是否满足规则boolean flag = checkStr(str);if (flag) {break;} else {System.out.println("当前的字符串不符合规则,请重新输入");continue;}}//将内容变成罗马数字//下面是阿拉伯数字跟罗马数字的对比关系://Ⅰ-1、Ⅱ-2、Ⅲ-3、Ⅳ-4、Ⅴ-5、Ⅵ-6、Ⅶ-7、Ⅷ-8、Ⅸ-9//查表法:数字跟数据产生一个对应关系StringBuilder sb = new StringBuilder();for (int i = 0; i < str.length(); i++) {char c = str.charAt(i);String s = changeLuoMa(c);sb.append(s);}System.out.println(sb);}//利用switch进行匹配public static String changeLuoMa(char number) {return switch (number) {case '0' -> "";case '1' -> "Ⅰ";case '2' -> "Ⅱ";case '3' -> "Ⅲ";case '4' -> "Ⅳ";case '5' -> "Ⅴ";case '6' -> "Ⅵ";case '7' -> "Ⅶ";case '8' -> "Ⅷ";case '9' -> "Ⅸ";// *修正:直接返回空字符串// *这部分有报错default -> "";};}public static boolean checkStr(String str) {//123456//要求1:长度为小于等于9if (str.length() > 9) {return false;}//要求2:只能是数字for (int i = 0; i < str.length(); i++) {char c = str.charAt(i);//0~9if (c < '0' || c > '9') {return false;}}//只有当字符串里面所有的字符全都判断完毕了,我才能认为当前的字符串是符合规则return true;}
}

14、调整字符串的内容并比较

给定两个字符串,A 和 B。
A 的旋转操作就是将 A最左边的字符移动到最右边。例如,若 A=‘abcde’,在移动一次之后结果就是’bcdea’如果在若干次调整操作之后,A能变成B,那么返回True。如果不能匹配成功,则返回false。

📖代码示例:

package com.itheima.test;public class Test2Case1 {public static void main(String[] args) {/* 给定两个字符串, A和B。A的旋转操作就是将A 最左边的字符移动到最右边。例如, 若A = 'abcde',在移动一次之后结果就是'bcdea'如果在若干次调整操作之后,A能变成B,那么返回True。如果不能匹配成功,则返回false*///1.定义两个字符串String strA = "abcde";String strB = "ABC";//2.调用方法进行比较boolean result = check(strA, strB);//3.输出System.out.println(result);}public static boolean check(String strA, String strB) {for (int i = 0; i < strA.length(); i++) {strA = rotate(strA);if(strA.equals(strB)){return true;}}//所有的情况都比较完毕了,还不一样那么直接返回falsereturn false;}//作用:旋转字符串,把左侧的字符移动到右侧去//形参:旋转前的字符串//返回值:旋转后的字符串public static String rotate(String str) {//套路://如果我们看到要修改字符串的内容//可以有两个办法://1.用subString进行截取,把左边的字符截取出来拼接到右侧去//2.可以把字符串先变成一个字符数组,然后调整字符数组里面数据,最后再把字符数组变成字符串。//截取思路//获取最左侧那个字符char first = str.charAt(0);//获取剩余的字符String end = str.substring(1);return end + first;}
}
package com.itheima.test;public class Test2Case2 {public static void main(String[] args) {/* 给定两个字符串, A和B。A的旋转操作就是将A 最左边的字符移动到最右边。例如, 若A = 'abcde',在移动一次之后结果就是'bcdea'如果在若干次调整操作之后,A能变成B,那么返回True。如果不能匹配成功,则返回false*///1.定义两个字符串String strA = "abcde";String strB = "ABC";//2.调用方法进行比较boolean result = check(strA, strB);//3.输出System.out.println(result);}public static boolean check(String strA, String strB) {for (int i = 0; i < strA.length(); i++) {strA = rotate(strA);if (strA.equals(strB)) {return true;}}//所有的情况都比较完毕了,还不一样那么直接返回falsereturn false;}//作用:旋转字符串,把左侧的字符移动到右侧去//形参:旋转前的字符串//返回值:旋转后的字符串public static String rotate(String str) {//套路://如果我们看到要修改字符串的内容//可以有两个办法://1.用subString进行截取,把左边的字符截取出来拼接到右侧去//2.可以把字符串先变成一个字符数组,然后调整字符数组里面数据,最后再把字符数组变成字符串。//可以把字符串先变成一个字符数组,然后调整字符数组里面数据,最后再把字符数组变成字符串。//"ABC"   ['A','B','C']  ['B','C','A']   new String(字符数组);char[] arr = str.toCharArray();//拿到0索引上的字符char first = arr[0];//把剩余的字符依次往前挪一个位置for (int i = 1; i < arr.length; i++) {arr[i - 1] = arr[i];}//把原来0索引上的字符放到最后一个索引arr[arr.length - 1] = first;//利用字符数组创建一个字符串对象String result = new String(arr);return result;}
}

15、后续练习思路分析

1️⃣键盘输入任意字符串,打乱里面的内容
📖代码示例:

package com.itheima.test;public class Test3 {public static void main(String[] args) {//键盘输入任意字符串,打乱里面的内容//1.键盘输入任意字符串String str = "abcdefg";//2.打乱里面的内容//修改字符串里面的内容://1.subString//2.变成字符数组char[] arr = str.toCharArray();//['a','b','c','d','e','f','g']//3.打乱数组里面的内容//从0索引开始,跟一个随机索引进行位置的交换//当数组里面的每一个元素都跟一个随机索引进行交换完毕之后,那么内容就打乱了//4.把字符数组再变回字符串String result = new String(arr);System.out.println(result);}
}

2️⃣生成验证码

内容:可以是小写字母,也可以是大写字母,还可以是数字

规则:长度为5。
内容中是四位字母,1位数字。
其中数字只有1位,但是可以出现在任意的位置。

📖代码示例:

package com.itheima.test;public class Test4 {public static void main(String[] args) {/*生成验证码内容:可以是小写字母,也可以是大写字母,还可以是数字规则:长度为5内容中是四位字母,1位数字。其中数字只有1位,但是可以出现在任意的位置。*///1.可以把所有的大写字母,小写字母都放到一个数组当中char[] arr = new char[52];//a-z  A-Z//2.从数组中随机获取4次//3.生成一个0~9之间的随机数拼接到最后//ACFG7//思考,我们把7放到前面,修改了字符串的内容//把生成的验证码先变成一个字符数组//再让最后一个元素跟前面的随机位置的元素进行交换//交换完毕之后再变成字符串就可以了。}
}

3️⃣
给定两个以字符串形式表示的非负整数num1和num2,返回num1和num2的乘积,它们的乘积也表示为字符串形式。
注意:需要用已有的知识完成。

📖代码示例:

package com.itheima.test;public class Test5 {public static void main(String[] args) {/* 给定两个以字符串形式表示的非负整数num1和num2,返回num1和num2的乘积,它们的乘积也表示为字符串形式。注意:需要用已有的知识完成。*///不需要考虑乘积过大之后的结果//就认为乘积一定是小于int的最大值的String num1 = "123456789";String num2 = "12345";//1.把num1和num2变成对应的整数才可以//"123456789"//先遍历字符串依次得到每一个字符 '1'  '2'  '3'  '4'  '5'  '6'  '7'  '8'  '9'//再把字符变成对应的数字即可     1    2    3     4    5    6    7    8    9//把每一个数字组合到一起 123456789//2.利用整数进行相乘//3.可以把整数变成字符串//+""}
}

4️⃣
给你一个字符串 s,由若干单词组成,单词前后用一些空格字符隔开。
返回字符串中最后一个单词的长度。
单词是指仅由字母组成、不包含任何空格字符的最大子字符串。

📖代码示例:

package com.itheima.test;public class Test6 {public static void main(String[] args) {/* 给你一个字符串 s,由若干单词组成,单词前后用一些空格字符隔开。返回字符串中最后一个单词的长度。单词是指仅由字母组成、不包含任何空格字符的最大子字符串。示例 1:输入:s = "Hello World“	输出:5解释:最后一个单词是“World”,长度为5。示例 2:输入:s = "   fly me   to   the moon"	输出:4解释:最后一个单词是“moon”,长度为4。示例 3:输入:s = "luffy is still joyboy"	输出:6解释:最后一个单词是长度为6的“joyboy”。*///倒着遍历//直到遇到空格为止//那么遍历的次数就是单词的长度}
}

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

使用git commit时‘“node“‘ 不是内部或外部命令,也不是可运行的程序

第一种&#xff1a; 使用git commit -m "xxx"时会报错&#xff0c;我看网上的方法是在命令行后面添加--no-verify&#xff1a;git commit -m "主题更新" --no-verify&#xff0c;但是不可能每次都添加。 最后解决办法是&#xff1a;使用git config --lis…

DeepSeek从入门到精通:全面掌握AI大模型的核心能力

文章目录 一、DeepSeek是什么&#xff1f;性能对齐OpenAI-o1正式版 二、Deepseek可以做什么&#xff1f;能力图谱文本生成自然语言理解与分析编程与代码相关常规绘图 三、如何使用DeepSeek&#xff1f;四、DeepSeek从入门到精通推理模型推理大模型非推理大模型 快思慢想&#x…

使用OBS推流,大华摄像头 srs服务器播放

说明&#xff1a; ffmpeg可以推流&#xff0c;但是是命令行方式不太友好&#xff0c;还可以使用主流的OBS开源推流软件&#xff0c;可从官网Open Broadcaster Software | OBS 下载最新版本&#xff0c;目前很多网络主播都是用它做直播。该软件支持本地视频文件以及摄像头推流。…

从大规模恶意攻击 DeepSeek 事件看 AI 创新隐忧:安全可观测体系建设刻不容缓

作者&#xff1a;羿莉&#xff08;萧羿&#xff09; 全球出圈的中国大模型 DeepSeek 作为一款革命性的大型语言模型&#xff0c;以其卓越的自然语言处理能力和创新性成本控制引领行业前沿。该模型不仅在性能上媲美 OpenAI-o1&#xff0c;而且在推理模型的成本优化上实现了突破…

mac下dify+deepseek部署,实现私人知识库

目前deepseek 十分火爆&#xff0c;本地部署实现私有知识库&#xff0c;帮助自己日常工作&#xff0c;上一篇使用工具cherry studio可以做到私人知识库。今天学习了一下&#xff0c;使用Dify链接deepseek&#xff0c;实现私人知识库&#xff0c;也非常不错&#xff0c;这里分享…

react概览webpack基础

react概览 课程介绍 webpack 构建依赖图->bundle 首屏渲染&#xff1a; 减少白屏等待时间 数据、结构、样式都返回。需要服务器的支持 性能优化 ***webpack干的事情 模块化开发 优势&#xff1a; 多人团队协作开发 可复用 单例&#xff1a;全局冲突 闭包 模块导入的顺序 req…

常见的九种二极管

常见的九种二极管 文章目录 常见的九种二极管1、普通二极管2、光电二极管&#xff08;LED&#xff09;3、变容二级管4、发光二极管5、恒流二极管6、快恢复二极管&#xff08;FRD&#xff09;7、肖特基二极管8、瞬态电压抑制二极管(TVS)9、齐纳二极管&#xff08;稳压&#xff0…

LabVIEW在呼吸机测试气体容量计算

在呼吸机测试中&#xff0c;精确测量气体容量变化是评估设备性能的关键步骤。通过监测呼吸机气道内的压力变化&#xff0c;并结合流阻和肺顺应性等参数&#xff0c;可以计算出单位时间内的气体容量变化。本案例基于LabVIEW实现该计算过程&#xff0c;以确保测试数据的准确性和一…

本地部署DeepSeek R1 + 界面可视化open-webui

本地部署DeepSeek R1 界面可视化open-webui ollama是物理机本地安装 open-webui是容器启动 另外&#xff0c;用docker 部署ollama也很方便ollama docker 安裝部署ollama ollama官网 安装 Linux上安装: curl -fsSL https://ollama.com/install.sh | sh使用命令行管理 拉…

使用C语言实现MySQL数据库的增删改查操作指南

使用C语言与MySQL数据库进行交互,通常涉及使用MySQL提供的C API库。这套API允许开发者在C/C++程序中执行SQL查询,从而实现数据库的增删改查操作。下面,我将详细介绍如何在C语言中实现这些基本操作。 准备工作 安装MySQL开发库:确保你的系统上安装了MySQL服务器以及MySQL开发…

在CT107D单片机综合训练平台上实现外部中断控制LED闪烁

引言 在单片机开发中&#xff0c;外部中断是一个非常重要的功能&#xff0c;它可以让单片机在检测到外部信号变化时立即做出响应。本文将详细介绍如何在CT107D单片机综合训练平台上使用外部中断来控制LED灯的闪烁。我们将使用两种不同的方式来实现这一功能&#xff1a;一种是在…

重磅发布!AI 驱动的 Java 开发框架:Spring AI Alibaba

*本文作者系阿里云云原生微服务技术负责人&#xff0c;Spring AI Alibaba 发起人彦林&#xff0c;望陶和隆基对可观测和 RocketMQ 部分内容亦有贡献。 * 摘要 随着生成式 AI 的快速发展&#xff0c;基于 AI 开发框架构建 AI 应用的诉求迅速增长&#xff0c;涌现出了包括 Lang…

防御保护作业二

拓扑图 需求 需求一&#xff1a; 需求二&#xff1a; 需求三&#xff1a; 需求四&#xff1a; 需求五&#xff1a; 需求六&#xff1a; 需求七&#xff1a; 需求分析 1.按照要求进行设备IP地址的配置 2.在FW上开启DHCP功能&#xff0c;并配置不同的全局地址池&#xff0c;为…

MapReduce简单应用(三)——高级WordCount

目录 1. 高级WordCount1.1 IntWritable降序排列1.2 输入输出格式1.3 处理流程 2. 代码和结果2.1 pom.xml中依赖配置2.2 工具类util2.3 高级WordCount2.4 结果 参考 本文引用的Apache Hadoop源代码基于Apache许可证 2.0&#xff0c;详情请参阅 Apache许可证2.0。 1. 高级WordCo…

智慧机房解决方案(文末联系,领取整套资料,可做论文)

智慧机房解决方案-软件部分 一、方案概述 本智慧机房解决方案旨在通过硬件设备与软件系统的深度整合&#xff0c;实现机房的智能化管理与服务&#xff0c;提升机房管理人员的工作效率&#xff0c;优化机房运营效率&#xff0c;确保机房设备的安全稳定运行。软件部分包括机房管…

(五)Spring Boot学习——spring security +jwt使用(前后端分离模式)

一定要熟悉spring security原理和jwt无状态原理&#xff0c;理解了才知道代码作用。 在 Spring Security JWT 认证流程中&#xff0c;通常的做法是&#xff1a; 用户提交用户名和密码Spring Security 认证管理器 (AuthenticationManager) 进行认证如果认证成功&#xff0c;生…

清华DeepSeek手册:从入门到精通(网页版便于阅读)

目录 一、产品概述二、清华DeepSeek从入门到精通三、PDF文件路径 一、产品概述 DeepSeek是国产领先的人工智能技术平台&#xff0c;提供从数据分析到大型语言模型的全栈解决方案。其核心产品包括网页端数据分析工具[1] 、视觉语言模型(DeepSeek-VL)[2] 和670亿参数大型语言模型…

阿里云百炼初探DeepSeek模型调用

阿里云百炼初探DeepSeek模型调用 阿里云百炼为什么选择百炼开始使用百炼方式一&#xff1a;文本对话方式二&#xff1a;文本调试方式三&#xff1a;API调用 DeepSeek调用1、搜索模型2、查看API调用3、开始调用安装依赖查看API Key运行以下代码 4、流式输出 总结 阿里云百炼 阿…

【网络安全】服务器安装Docker及拉取镜像教程

文章目录 1. 安装 Docker2. 拉取镜像3. 运行 Ubuntu 容器4. 执行相关操作5. 退出并停止容器1. 安装 Docker # 更新软件包索引 sudo apt update# 安装必要的依赖 sudo apt install -y ca-certificates curl gnupg

AI刷题-子数组和的最大值问题

目录 问题描述 输入格式 输出格式 输入样例 输出样例 说明 数据范围 解题思路&#xff1a; 问题理解 数据结构选择 算法步骤 具体步骤 代码实现&#xff1a; 1.特判&#xff1a; 不需要删除元素的时候 2.在前面的判断结束后&#xff1a;k1&#xff0c;&#xff…