目录
前言
一、String类的常见用法
1.1 字符串构造(常见三种)
①使用常量串构造
②直接newString对象
③ 使用字符数组进行构造
注意:
1.2 String对象的比较
1. ==比较是否引用同一个对象
2. boolean equals(Object anObject) 方法:
3. int compareTo(String s) 方法:
4. int compareToIgnoreCase(String str) 方法:
1.3 字符串查找
1.4 四种转化
①数值和字符串转化
②大小写转换
③字符串转数组
④格式化
1.5字符串替换
1.6 字符串拆分
1.7 字符串截取
1.8 其他操作方法
1.9 字符串的不可变性
为什么 String 要设计成不可变的?(了解)
1.10 字符串修改
二、StringBuilder和StringBuffer
2.1 StringBuilder的介绍
面试题:
三、 String类编程题练习(leetcode)
1.字符串相加
2.字符串中第一个唯一字符
3.最后一个单词的长度
4.验证回文串
前言
本篇博客主要讲解Java基础语法中的
认识 String 类 ,了解 String 类的基本用法,知道字符串的三种常见构造方法。
会使用String对象的比较,知道==、equals、compareTo、compareToIgnoreCase比较的用法和区别。等String类常见用法。 熟练掌握 String 类的常见操作 、认识字符串常量池、
认识 StringBuffer 和 StringBuilder、最后是String类编程题的练习。
大家好,本人是普通一本的在校大学生一枚,目前在学习java。之前也学了一段时间,但是没有发布博客。本人现在已经大二结束了,开学就大三了,时间过的真的很快。我会利用好这个暑假,来复习之前学过的内容,并整理好之前写过的博客进行发布。如果博客中有错误或者没有读懂的地方。热烈欢迎大家在评论区进行讨论!!!
喜欢我文章的兄弟姐妹们可以点赞,收藏和评论我的文章。喜欢我的兄弟姐妹们以及也想复习一遍java知识的兄弟姐妹们可以关注我呦,我会持续更新滴,并且追求完整。
望支持!!!!!!一起加油呀!!!!
语言只是工具,不能决定你好不好找工作,决定你好不好找工作的是你的能力!!!!!
学历本科及以上就够用了!!!!!!!!!!!!!!!!!!!!!!!!!!!!
话不多说,直接上干货
一、String类的常见用法
我们都知道String是字符串类型,是引用类型。在java中String也是一个类。
1.1 字符串构造(常见三种)
①使用常量串构造
// 使用常量串构造String s1 = "hello bit";System.out.println(s1);
②直接newString对象
// 直接newString对象String s2 = new String("hello bit");System.out.println(s1);
③ 使用字符数组进行构造
// 使用字符数组进行构造char[] array = {'h','e','l','l','o','b','i','t'};String s3 = new String(array);System.out.println(s1);
注意:
1. String是引用类型,内部并不存储字符串本身,在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()); // 获取字符串长度---输出5System.out.println(s1.isEmpty()); // 如果字符串长度为0,返回true,否则返回false
}
代码图解:
s1和s2引用的是不同对象 s1和s3引用的是同一对象
2. 在Java中“”引起来的也是String类型对象。
// 打印"hello"字符串(String对象)的长度
System.out.println("hello".length());
1.2 String对象的比较
字符串的比较是常见操作之一,比如:字符串排序。Java中总共提供了4种方式:
1. ==比较是否引用同一个对象
- 对于基本类型,==比较的是变量中的值;
- 对于引用类型==比较的是引用中的地址。
public static void main(String[] args) {int a = 10;int b = 20;int c = 10;// 对于基本类型变量,==比较两个变量中存储的值是否相同System.out.println(a == b); // falseSystem.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); // falseSystem.out.println(s2 == s3); // falseSystem.out.println(s1 == s4); // true
}
2. boolean equals(Object anObject) 方法:
按照字典序比较
字典序:字符大小的顺序
String类重写了父类Object中equals方法,Object中equals默认按照==比较,
String重写equals方法后,按照 如下规则进行比较,
比如: s1.equals(s2)
public boolean equals(Object anObject) {// 1. 先检测this 和 anObject 是否为同一个对象比较,如果是返回trueif (this == anObject) {return true;}// 2. 检测anObject是否为String类型的对象,如果是继续比较,否则返回falseif (anObject instanceof String) {// 将anObject向下转型为String类型对象String anotherString = (String)anObject;int n = value.length;// 3. this和anObject两个字符串的长度是否相同,是继续比较,否则返回falseif (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;
}
public static void main(String[] args) {String s1 = new String("hello");String s2 = new String("hello");String s3 = new String("Hello");// s1、s2、s3引用的是三个不同对象,因此==比较结果全部为falseSystem.out.println(s1 == s2); // falseSystem.out.println(s1 == s3); // false// equals比较:String对象中的逐个字符// 虽然s1与s2引用的不是同一个对象,但是两个对象中放置的内容相同,因此输出true// s1与s3引用的不是同一个对象,而且两个对象中内容也不同,因此输出falseSystem.out.println(s1.equals(s2)); // trueSystem.out.println(s1.equals(s3)); // false
}
- ==比较结果全部为false,因为s1、s2、s3引用的是三个不同对象
equals比较:String对象中的逐个字符
- 两个对象中放置的内容相同,因此输出true虽然s1与s2引用的不是同一个对象
- s1与s3引用的不是同一个对象,而且两个对象中内容也不同,因此输出false
3. int compareTo(String s) 方法:
按照字典序进行比较
与equals不同的是
equals返回的是boolean类型,而compareTo返回的是int类型。具体比较方式:
1. 先按照字典次序大小比较,如果出现不等的字符,直接返回这两个字符的大小差值
2. 如果前k个字符相等(k为两个字符长度最小值),返回值两个字符串长度差值
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)); // 不同输出字符差值-1System.out.println(s1.compareTo(s3)); // 相同输出 0System.out.println(s1.compareTo(s4)); // 前k个字符完全相同,输出长度差值 -3
}
4. int compareToIgnoreCase(String str) 方法:
与compareTo方式相同,但是忽略大小写比较
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.compareToIgnoreCase(s2)); // 不同输出字符差值-1System.out.println(s1.compareToIgnoreCase(s3)); // 相同输出 0System.out.println(s1.compareToIgnoreCase(s4)); // 前k个字符完全相同,输出长度差值 -3
}
1.3 字符串查找
String类提供的常用查找的方法:
public static void main(String[] args) {String s = "aaabbbcccaaabbbccc";System.out.println(s.charAt(3)); // 'b'System.out.println(s.indexOf('c')); // 6System.out.println(s.indexOf('c', 10)); // 15System.out.println(s.indexOf("bbb")); // 3System.out.println(s.indexOf("bbb", 10)); // 12System.out.println(s.lastIndexOf('c')); // 17System.out.println(s.lastIndexOf('c', 10)); // 8System.out.println(s.lastIndexOf("bbb")); // 12System.out.println(s.lastIndexOf("bbb", 10)); // 3
}
注意:上述方法都是实例方法。
1.4 四种转化
①数值和字符串转化
数字转字符串:
String.valueOf(1234);
字符串转数字:
Integer.parseInt("1234");
public static void main(String[] args) {// 数字转字符串String s1 = String.valueOf(1234);String s2 = String.valueOf(12.34);String s3 = String.valueOf(true);String s4 = String.valueOf(new Student("Hanmeimei", 18));System.out.println(s1);System.out.println(s2);System.out.println(s3);System.out.println(s4);System.out.println("=================================");// 字符串转数字// 注意:Integer、Double等是Java中的包装类型,这个后面会讲到int data1 = Integer.parseInt("1234");double data2 = Double.parseDouble("12.34");System.out.println(data1);System.out.println(data2);}
运行结果
②大小写转换
s1.toUpperCase();
s2.toLowerCase();
public static void main(String[] args) {String s1 = "hello";String s2 = "HELLO";System.out.println(s1.toUpperCase()); // 小写转大写System.out.println(s2.toLowerCase()); // 大写转小写}
运行结果
③字符串转数组
s.toCharArray();
public static void main(String[] args) {String s = "hello"; // 字符串转数组char[] ch = s.toCharArray();for (int i = 0; i < ch.length; i++) {System.out.print(ch[i]);}System.out.println(); // 数组转字符串String s2 = new String(ch);System.out.println(s2);}
④格式化
String.format();
public static void main(String[] args) {String s = String.format("%d-%d-%d", 2019, 9,14);System.out.println(s);
}
2019-9-14
1.5字符串替换
使用一个指定的新的字符串替换掉已有的字符串数据,可用的方法如下:
代码示例: 字符串的替换处理
String str = "helloworld" ;System.out.println(str.replaceAll("l", "_"));System.out.println(str.replaceFirst("l", "_"));
运行结果
注意事项: 由于字符串是不可变对象, 替换不修改当前字符串, 而是产生一个新的字符串.
1.6 字符串拆分
可以将一个完整的字符串按照指定的分隔符划分为若干个子字符串。
代码示例:
实现字符串的拆分处理
String str = "hello world hello bit" ;
String[] result = str.split(" ") ; // 按照空格拆分
for(String s: result) {System.out.println(s);
}
代码示例:
字符串的部分拆分
String str = "hello world hello bit" ;
String[] result = str.split(" ",2) ;
for(String s: result) {System.out.println(s);
}
拆分是特别常用的操作. 一定要重点掌握.
另外有些特殊字符作为分割符可能无法正确切分, 需要加上转义.
代码示例:
拆分IP地址
String str = "192.168.1.1" ;
String[] result = str.split("\\.") ;
for(String s: result) {System.out.println(s);
}
代码示例:
多次拆分
String str = "name=zhangsan&age=18" ;
String[] result = str.split("&") ;
for (int i = 0; i < result.length; i++) {String[] temp = result[i].split("=") ; System.out.println(temp[0]+" = "+temp[1]);
}
1.7 字符串截取
从一个完整的字符串之中截取出部分内容。可用方法如下:
代码示例:
String str = "helloworld" ;
System.out.println(str.substring(5));
System.out.println(str.substring(0, 5));
注意事项:
1. 索引从0开始
2. 注意前闭后开区间的写法, substring(0, 5) 表示包含0号下标的字符, 不包含5号下标
1.8 其他操作方法
代码示例:
观察trim()方法的使用
String str = " hello world " ;System.out.println("["+str+"]");System.out.println("["+str.trim()+"]");
trim 会去掉字符串开头和结尾的空白字符(空格, 换行, 制表符等).
代码示例:
大小写转换
String str = " hello%$$%@#$%world 哈哈哈 " ;System.out.println(str.toUpperCase());System.out.println(str.toLowerCase());
这两个函数只转换字母。
1.9 字符串的不可变性
String是一种不可变对象. 字符串中的内容是不可改变。
1. String类在设计时就是不可改变的,String类实现描述中已经说明了
被源码中final修饰
- 1. String类被final修饰,表明该类不能被继承
- 2. value被修饰被final修饰,表明value自身的值不能改变,即不能引用其它字符数组
- 但是value引用空间中的内容可以修改。
2. 所有涉及到可能修改字符串内容的操作都是创建一个新对象,改变的是新对象
比如replace方法:
【纠正】
网上有些人说:字符串不可变是因为其内部保存字符的数组被final修饰了,因此不能改变。这种说法是错误的,不是因为String类自身,或者其内部value被final修饰而不能被修改。
final修饰类表明该类不想被继承,
final修饰引用类型表明该引用变量不能引用其他对象,但是其引用对象中的内容是可以修改的。
public static void main(String[] args) {final int array[] = {1, 2, 3, 4, 5};array[0] = 100;System.out.println(Arrays.toString(array));}
我们可以看到是可以被修改的。
只是不能引用其他对象
为什么 String 要设计成不可变的?(了解)
- 方便实现字符串对象池. 如果 String 可变, 那么对象池就需要考虑写时拷贝的问题了.
- 不可变对象是线程安全的.
- 不可变对象更方便缓存 hash code, 作为 key 时可以更高效的保存到 HashMap 中.
1.10 字符串修改
注意:尽量避免直接对String类型对象进行修改,因为String类是不能修改的,
所有的修改都会创建新对象,效率非常低下。
public static void main(String[] args) {String s = "hello";s += " world";System.out.println(s); // 输出:hello world
}
但是这种方式不推荐使用,因为其效率非常低,中间创建了好多临时对象。
在对String类进行修改时,效率是非常慢的,因此:尽量避免对String的直接修改,
如果要修改建议尽量 使用StringBuffer或者StringBuilder。
二、StringBuilder和StringBuffer
2.1 StringBuilder的介绍
由于String的不可更改特性,为了方便字符串的修改,Java中又提供StringBuilder和StringBuffer类。这两个类大 部分功能是相同的,这里介绍 StringBuilder常用的一些方法
public static void main(String[] args) {StringBuilder sb1 = new StringBuilder("hello");StringBuilder sb2 = sb1;// 追加:即尾插-->字符、字符串、整形数字sb1.append(' ');// hellosb1.append("world"); // hello worldsb1.append(123); // hello world123System.out.println(sb1); // hello world123System.out.println(sb1 == sb2); // trueSystem.out.println(sb1.charAt(0)); // 获取0号位上的字符 hSystem.out.println(sb1.length()); // 获取字符串的有效长度14System.out.println(sb1.capacity()); // 获取底层数组的总大小sb1.setCharAt(0, 'H'); // 设置任意位置的字符 Hello world123sb1.insert(0, "Hello world!!!"); // Hello world!!!Hello world123System.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和StringBuilder最大的区别在于String的内容无法修改,而StringBuilder的内容可 以修改。频繁修改字符串的情况考虑使用StringBuilder。
注意:
String和StringBuilder类不能直接转换。
如果要想互相转换,可以采用如下原则:
- String变为StringBuilder: 利用StringBuilder的构造方法或append()方法
- StringBuilder变为String: 调用toString()方法。
面试题:
1.String、StringBuff和StringBulider之间的区别
- String的内容不可修改,StringBuffer与StringBuilder的内容可以修改.
- StringBuffer与StringBuilder大部分功能是相似的
- StringBuffer采用同步处理,属于线程安全操作;而StringBuilder未采用同步处理,属于线程不安全操作
2. 以下总共创建了多少个String对象【前提不考虑常量池之前是否存在】
String str = new String("ab");
- "ab":一个
String
对象存储在常量池中。- new String("ab"):一个新的
String
对象在堆中。总共:2 个对象
String str = new String("a") + new String("b");
这行代码的解析较为复杂,因为涉及到字符串连接操作。
- "a":一个
String
对象存储在常量池中。- "b":一个
String
对象存储在常量池中。- new String("a"):一个新的
String
对象在堆中。- new String("b"):一个新的
String
对象在堆中。- 字符串连接的结果:连接操作会创建一个新的
String
对象在堆中,结果为"ab"
。总共:5 个对象
三、 String类编程题练习(leetcode)
1.字符串相加
class Solution {public String addStrings(String num1, String num2) {int i = num1.length() - 1, j = num2.length() - 1, add = 0;StringBuffer ans = new StringBuffer();while (i >= 0 || j >= 0 || add != 0) {int x = i >= 0 ? num1.charAt(i) - '0' : 0;int y = j >= 0 ? num2.charAt(j) - '0' : 0;int result = x + y + add;ans.append(result % 10);add = result / 10;i--;j--;}// 计算完以后的答案需要翻转过来ans.reverse();return ans.toString();}
}作者:力扣官方题解
链接:https://leetcode.cn/problems/add-strings/solutions/357938/zi-fu-chuan-xiang-jia-by-leetcode-solution/
来源:力扣(LeetCode)
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。
- 将两个字符串表示的非负整数相加,并返回它们的和。
- 首先令i和j分别为字符串num1和num2的最后一个字符。
- 新建一个ans字符串。用于构建结果字符串。
- 循环处理每一位。当i和j非负或者add不为0.那么x和y就分别取num1和num2当前位的数字,如果超出范围则取0。result是当前位的和包括进位。result%10是当前位的数字放进ans中。更新add为result/10就是新的位。移动指针i和j减减
- 反转结果并返回,由于结果是从低位到高位追加的,需要反转。将
StringBuffer
转换为字符串并返回。
2.字符串中第一个唯一字符
class Solution {public int firstUniqChar(String s) {Map<Character, Integer> frequency = new HashMap<Character, Integer>();for (int i = 0; i < s.length(); ++i) {char ch = s.charAt(i);frequency.put(ch, frequency.getOrDefault(ch, 0) + 1);}for (int i = 0; i < s.length(); ++i) {if (frequency.get(s.charAt(i)) == 1) {return i;}}return -1;}
}
- 创建一个
HashMap
变量frequency
用于存储每个字符的出现次数。- 遍历字符串,统计字符出现次数
- 再次遍历字符串,找到第一个唯一字符
自己用String写的
class Solution {public int firstUniqChar(String s) {int[] count = new int[256];// 统计每个字符出现的次数for (int i = 0; i < s.length(); ++i) {count[s.charAt(i)]++;}// 找第一个只出现一次的字符for (int i = 0; i < s.length(); ++i) {if (1 == count[s.charAt(i)]) {return i;}}return -1;}
}
- 首先创建一个整型数组。其实就是哈希表
- 遍历字符串中的每一个字符。并且每出现一次这个字母将字母对应的ascii码值在哈希表中+1
- 最后再次遍历这个字符串中的每一个字符。找到第一个在哈希表是1的字符。返回对应的下标。得到答案
3.最后一个单词的长度
public static void main(String[] args) {Scanner sc = new Scanner(System.in);// 获取一行单词String s = sc.nextLine();// 1. 找到最后一个空格// 2. 获取最后一个单词:从最后一个空格+1位置开始,一直截取到末尾// 3. 打印最后一个单词长度int len = s.substring(s.lastIndexOf(' ') + 1, s.length()).length();System.out.println(len);}
- 首先获取到这一个字符串。
- 使用
- s.substring(s.lastIndexOf(' ') + 1, s.length()).length()
- 截取字符串最后一个空格后面的字符串。
- 再计算这串字符串的长度。得到答案。
4.验证回文串
class Solution {public boolean isPalindrome(String s) {StringBuffer sgood = new StringBuffer();int length = s.length();for (int i = 0; i < length; i++) {char ch = s.charAt(i);if (Character.isLetterOrDigit(ch)) {sgood.append(Character.toLowerCase(ch));}}StringBuffer sgood_rev = new StringBuffer(sgood).reverse();return sgood.toString().equals(sgood_rev.toString());}
}
- 创建一个 StringBuffer 来存储经过处理后的字符串
- 用length获取字符串长度。
- 通过循环遍历每一个字符,如果这个字符是字母或者数字,则添加到sgood字符串中
- 最后通过StringBuffer翻转这个字符串。如果两者相等那么他就是回文串。
public static boolean isValidChar(char ch) {if ((ch >= 'a' && ch <= 'z') ||(ch >= '0' && ch <= '9')) {return true;}return false;}public boolean isPalindrome(String s) {s = s.toLowerCase();int left = 0, right = s.length() - 1;while (left < right) {// 1. 从左侧找到一个有效的字符while (left < right && !isValidChar(s.charAt(left))) {left++;}// 2. 从右侧找一个有效的字符while (left < right && !isValidChar(s.charAt(right))) {right--;}if (s.charAt(left) != s.charAt(right)) {return false;} else {left++;right--;}}return true;}
- 先判断是否是合法的字符,是否只包含字母和数字,如果是返回true,不是返回false
- 再验证是否是回文串。将所有字母全部转换成小写。
- 通过双指针循环,left从左边开始,right从右边开始循环。
- 从左边找到第一个有效字符,从右边找到第一个有效字符
- 再比较两个字符是否相等,如果不相等返回false,如果相等left++,right++。继续判断
- 直到left不小于right
- 最终返回true。