字符串操作的12个小技巧!

字符串可以说是 Java 中最具有代表性的类了,似乎没有之一哈,这就好像直播界的李佳琪,脱口秀中的李诞,一等一的大哥地位。不得不承认,最近吐槽大会刷多了,脑子里全是那些段子,写文章都有点不由自主,真的是,手不由己啊。

字符串既然最常用,那就意味着面试官好这一口,就喜欢问一些字符串方面的编码技巧,来测试应聘者是否技术过硬,底子扎实,对吧?

那这次,我就来盘点 12 个精致的 Java 字符串操作小技巧,来帮助大家提高一下下。在查看我给出的答案之前,最好自己先动手尝试一遍,写不出来答案没关系,先思考一遍,看看自己的知识库里是不是已经有解决方案,有的话,就当是温故复习了,没有的话,也不要担心,刚好学一遍。

01、如何在字符串中获取不同的字符及其数量?

这道题可以拆解为两个步骤,第一步,找出不同的字符,第二步,统计出它们的数量。好像有点废话,是不是?那我先来一个答案吧。

public class DistinctCharsCount {public static void main(String[] args) {printDistinctCharsWithCount("itwanger");printDistinctCharsWithCount("chenmowanger");}private static void printDistinctCharsWithCount(String input) {Map<Character, Integer> charsWithCountMap = new LinkedHashMap<>();for (char c : input.toCharArray()) {Integer oldValue = charsWithCountMap.get(c);int newValue = (oldValue == null) ? 1 :Integer.sum(oldValue, 1);charsWithCountMap.put(c, newValue);}System.out.println(charsWithCountMap);}
}

程序输出的结果是:

{i=1, t=1, w=1, a=1, n=1, g=1, e=1, r=1}
{c=1, h=1, e=2, n=2, m=1, o=1, w=1, a=1, g=1, r=1}

说一下我的思路:

1)声明一个 LinkedHashMap,也可以用 HashMap,不过前者可以保持字符串拆分后的顺序,结果看起来更一目了然。

为什么要用 Map 呢?因为 Map 的 key 是不允许重复的,刚好可以对重复的字符进行数量的累加。

2)把字符串拆分成字符,进行遍历。

3)如果 key 为 null 的话,就表明它的数量要 +1;否则的话,就在之前的值上 +1,然后重新 put 到 Map 中,这样就覆盖了之前的字符数量。

思路很清晰,对不对?忍不住给自己鼓个掌。

那,JDK 8 之后,Map 新增了一个很厉害的方法 merge(),一次性为多个键赋值:

private static void printDistinctCharsWithCountMerge(String input) {Map<Character, Integer> charsWithCountMap = new LinkedHashMap<>();for (char c : input.toCharArray()) {charsWithCountMap.merge(c, 1, Integer::sum);}System.out.println(charsWithCountMap);
}

有没有很厉害?一行代码就搞定。第一个参数为键,第二个参数为值,第三个参数是一个 BiFunction,意思是,如果键已经存在了,就重新根据 BiFunction 计算新的值。

如果字符是第一次出现,就赋值为 1;否则,就把之前的值 sum 1。

02、如何反转字符串?

如果同学们对 StringBuilder 和 StringBuffer 很熟悉的话,这道题就很简单,直接 reverse() 就完事,对不对?

public class ReverseAString {public static void main(String[] args) {reverseInputString("沉默王二");}private static void reverseInputString(String input) {StringBuilder sb = new StringBuilder(input);String result = sb.reverse().toString();System.out.println(result);}
}

输出结果如下所示:

二王默沉

多说一句,StringBuffer 和 StringBuilder 很相似,前者是同步的,所有 public 方法都加了 synchronized 关键字,可以在多线程中使用;后者是不同步的,没有 synchronized 关键字,所以性能更佳,没有并发要求的话,就用 StringBuilder。

03、如何判断一个字符串是前后对称的?

什么意思呢?就好像一个字符串,前后一折,是对称的。就像你站在镜子前,看到了一个玉树临风、闭月羞花的自己。

public class PalindromeString {public static void main(String[] args) {checkPalindromeString("沉默王二");checkPalindromeString("沉默王二 二王默沉");}private static void checkPalindromeString(String input) {boolean result = true;int length = input.length();for (int i = 0; i < length / 2; i++) {if (input.charAt(i) != input.charAt(length - i - 1)) {result = false;break;}}System.out.println(input + " 对称吗? " + result);}
}

输出结果如下所示:

沉默王二 对称吗? false
沉默王二 二王默沉 对称吗? true

说一下我的思路:要判断字符串对折后是否对称,很简单,从中间劈开,第一个字符对照最后一个字符,一旦找到不等的那个,就返回 false。

注意三点:

1)for 循环的下标从 0 开始,到 length/2 结束。

2)下标 i 和 length-i-1 是对称的。

3)一旦 false 就 break。

04、如何删除所有出现的指定字符?

字符串类没有提供 remove() 方法,但提供了 replaceAll() 方法,通过将指定的字符替换成空白字符就可以办得到,对吧?

public class RemoveCharFromString {public static void main(String[] args) {removeCharFromString("沉默王二", '二');removeCharFromString("chenmowanger", 'n');}private static void removeCharFromString(String input, char c) {String result = input.replaceAll(String.valueOf(c), "");System.out.println(result);}
}

输出结果如下所示:

沉默王
chemowager

05、如何证明字符串是不可变的?

字符串不可变的这个事我曾写过两篇文章,写到最后我都要吐了。但是仍然会有一些同学弄不明白,隔段时间就有人私信我,我就不得不把之前的文章放到收藏夹,问的时候我就把链接发给他。

之所以造成这个混乱,有很多因素,比如说,Java 到底是值传递还是引用传递?字符串常量池是个什么玩意?

这次又不得不谈,虽然烦透了,但仍然要证明啊!

public class StringImmutabilityTest {public static void main(String[] args) {String s1 = "沉默王二";String s2 = s1;System.out.println(s1 == s2);s1 = "沉默王三";System.out.println(s1 == s2);System.out.println(s2);}
}

输出结果如下所示:

true
false
沉默王二

1)String s1 = "沉默王二",Java 在字符串常量池中创建“沉默王二”这串字符的对象,并且把地址引用赋值给 s1

2)String s2 = s1,s2 和 s1 指向了同一个地址引用——常量池中的那个“沉默王二”。

所以,此时 s1 == s2 为 true。

3)s1 = "沉默王三",Java 在字符串常量池中创建“沉默王三”这串字符的对象,并且把地址引用赋值给 s1,但 s2 仍然指向的是“沉默王二”那串字符对象的地址引用。

所以,此时 s1 == s2 为 false,s2 的输出结果为“沉默王二”就证明了字符串是不可变的。

06、如何统计字符串中的单词数?

这道题呢?主要针对的是英文字符串的情况。虽然中文字符串中也可以有空白字符,但不存在单词这一说。

public class CountNumberOfWordsInString {public static void main(String[] args) {countNumberOfWords("My name is Wanger");countNumberOfWords("I Love Java Programming");countNumberOfWords(" Java    is  very   important ");}private static void countNumberOfWords(String line) {String trimmedLine = line.trim();int count = trimmedLine.isEmpty() ? 0 : trimmedLine.split("\\s+").length;System.out.println(count);}
}

输出结果如下所示:

4
4
4

split() 方法可以对字符串进行拆分,参数不仅可以是空格,也可以使正则表达式代替的空白字符(多个空格、制表符);返回的是一个数组,通过 length 就可以获得单词的个数了。

如果对 split() 方法很感兴趣的话,可以查看我之前写的一篇文章,很饱满,很丰富。

咦,拆分个字符串都这么讲究

07、如何检查两个字符串中的字符是相同的?

如何理解这道题呢?比如说,字符串“沉默王二”和“沉王二默”就用了同样的字符,对吧?比如说,字符串“沉默王二”和“沉默王三”用的字符就不同,理解了吧?

public class CheckSameCharsInString {public static void main(String[] args) {sameCharsStrings("沉默王二", "沉王二默");sameCharsStrings("沉默王二", "沉默王三");}private static void sameCharsStrings(String s1, String s2) {Set<Character> set1 = s1.chars().mapToObj(c -> (char) c).collect(Collectors.toSet());System.out.println(set1);Set<Character> set2 = s2.chars().mapToObj(c -> (char) c).collect(Collectors.toSet());System.out.println(set2);System.out.println(set1.equals(set2));}
}

输出结果如下所示:

[默, 沉, 王, 二]
[默, 沉, 王, 二]
true
[默, 沉, 王, 二]
[默, 沉, 三, 王]
false

上面的代码用到了 Stream 流,看起来很陌生,但很好理解,就是把字符串拆成字符,然后收集到 Set 中,Set 是一个不允许有重复元素的集合,所以就把字符串中的不同字符收集起来了。

08、如何判断一个字符串包含了另外一个字符串?

这道题有点简单,对吧?上一道还用 Stream 流,这道题就直接送分了?不用怀疑自己,就用字符串类的 contains() 方法。

public class StringContainsSubstring {public static void main(String[] args) {String s1 = "沉默王二";String s2 = "沉默";System.out.println(s1.contains(s2));}
}

输出结果如下所示:

true

contains() 方法内部其实调用的是 indexOf() 方法:

public boolean contains(CharSequence s) {return indexOf(s.toString()) >= 0;
}

09、如何在不用第三个变量的情况下交换两个字符串?

这道题就有点意思了,对吧?尤其是前提条件,不使用第三个变量。

public class SwapTwoStrings {public static void main(String[] args) {String s1 = "沉默";String s2 = "王二";s1 = s1.concat(s2);s2 = s1.substring(0,s1.length()-s2.length());s1 = s1.substring(s2.length());System.out.println(s1);System.out.println(s2);}
}

输出结果如下所示:

王二
沉默

说一下我的思路:

1)通过 concat() 方法把两个字符串拼接到一块。

2)然后通过 substring() 方法分别取出第二个字符串和第一个字符串。

10、如何从字符串中找出第一个不重复的字符?

来,上个例子来理解一下这道题。比如说字符串“沉默王沉沉默二”,第一个不重复的字符是“王”,对吧?因为“沉”重复了,“默”重复了。

public class FindNonRepeatingChar {public static void main(String[] args) {System.out.println(printFirstNonRepeatingChar("沉默王沉沉默二"));System.out.println(printFirstNonRepeatingChar("沉默王沉"));System.out.println(printFirstNonRepeatingChar("沉沉沉"));}private static Character printFirstNonRepeatingChar(String string) {char[] chars = string.toCharArray();List<Character> discardedChars = new ArrayList<>();for (int i = 0; i < chars.length; i++) {char c = chars[i];if (discardedChars.contains(c))continue;for (int j = i + 1; j < chars.length; j++) {if (c == chars[j]) {discardedChars.add(c);break;} else if (j == chars.length - 1) {return c;}}}return null;}
}

输出结果如下所示:

王
默
null

说一下我的思路:

1)把字符串拆分成字符数组。

2)声明一个 List,把重复的字符放进去。

3)外层的 for 循环,从第一个字符开始,如果已经在 List 中,继续下一轮。

4)嵌套的 for 循环,从第一个字符的下一个字符(j = i + 1)开始遍历,如果找到和之前字符重复的,就加入到 List 中,跳出内层的循环;如果找到最后(j == chars.length - 1)也没有找到,就是第一个不重复的字符,对吧?

11、如何检查字符串中只包含数字?

有一种很傻的解法,就是用 Long.parseLong(string) 对字符串强转,如果转不成整形,那肯定不是只包含数字,对吧?

但这种方法也太不可取了,所以还得换一种巧妙的,就是使用正则表达式。

public class CheckIfStringContainsDigitsOnly {public static void main(String[] args) {digitsOnlyString("123 沉默王二");digitsOnlyString("123");}private static void digitsOnlyString(String string) {if (string.matches("\\d+")) {System.out.println("只包含数字的字符串:" + string);}}
}

输出结果如下所示:

只包含数字:123

12、如何实现字符串的深度拷贝?

由于字符串是不可变的,所以可以直接使用“=”操作符将一个字符串拷贝到另外一个字符串,并且互不影响。

public class JavaStringCopy {public static void main(String args[]) {String str = "沉默王二";String strCopy = str;str = "沉默王三";System.out.println(strCopy);}
}

输出结果如下所示:

沉默王二

这个例子和之前证明字符串是不可变的例子几乎没什么差别,对吧?这的确是因为字符串是不可变的,如果是可变对象的话,深度拷贝就要注意了,最好使用 new 关键字返回新的对象。

public Book getBook() {Book clone = new Book();clone.setPrice(this.book.getPrice());clone.setName(this.book.getName());return clone;
}

关于不可变对象,请点击下面的链接查看我之前写了一篇文章。

这次要说不明白immutable类,我就怎么地

最后

希望这 12 个精致的字符串操作小技巧可以帮助大家巩固一波基础,反正我自己已经重新巩固了一波,很有收获的样子,感觉就像是“一群小精灵在我脑子里跳舞一样”,学它就对了!


往期推荐

URL 去重的 6 种方案!(附详细代码)


多图证明,Java到底是值传递还是引用传递?


磊哥工作十几年了,竟没有用过do-while!(文末送书)

关注下方二维码,收获更多干货!

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

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

相关文章

关于二维数组取地址加以或减一解引用问题

int main() { int aa[2][5] { 1, 2, 3, 4, 5, 6, 7, 8, 9, 10 }; int *ptr1 (int *)(&aa 1); int *ptr2 (int *)(*(aa 1)); printf("%d,%d", *(ptr1 - 1), *(ptr2 - 1));system("pause");return 0; }很显然aa是一个二维数组&#xff0c;很多…

rand和srand的用法

首先我们要对rand&#xff06;srand有个总体的看法:srand初始化随机种子,rand产生随机数&#xff0c;下面将详细说明。rand&#xff08;产生随机数&#xff09;表头文件: #include定义函数 :int rand(void)函数说明 :因为rand的内部实现是用线性同余法做的&#xff0c;他不是真…

repeating 路由_CSS中带有示例的repeating-linear-gradient()函数

repeating 路由Introduction: 介绍&#xff1a; So far, we have learned so many functions but learning never gets enough, therefore as a good developer, we must learn as many functions as we can and know their behavior with the help of practical implementati…

万字详解|手撕 9大排序算法!

0. 前言大家好&#xff0c;我是多选参数的程序锅&#xff0c;一个正在捣鼓操作系统、学数据结构和算法以及 Java 的失业人员。数据结构和算法我已经学了有一段日子了&#xff0c;最近也开始在刷 LeetCode 上面的题目了&#xff0c;但是自己感觉在算法上还是 0 &#xff0c;还得…

INF文件修改注册表

INF是一个用于安装驱动程序的文件&#xff0c;是文本格式的。里面记录着驱动程序的所有相关资料&#xff0c;随便打开一个INF文件度可看到很多奇奇怪怪的东西&#xff0c;其实这一点也不奇怪&#xff0c;因为每一项都有特定的含义&#xff0c;比如从VERSION一项就可以看出支持的…

.Net判断一个对象是否为数值类型探讨总结(高营养含量,含最终代码及跑分)...

前一篇发出来后引发了积极的探讨&#xff0c;起到了抛砖引玉效果&#xff0c;感谢大家参与。 吐槽一下&#xff1a;这个问题比其看起来要难得多得多啊。 大家的讨论最终还是没有一个完全正确的答案&#xff0c;不过我根据讨论结果总结了一个差不多算是最终版的代码&#xff0c;…

ai怎么约束每个字的大小_人工智能的约束满意问题

ai怎么约束每个字的大小Constraint Satisfactory problems, as the name suggests are the problems which have some constraints which need to be satisfied while solving any problem. In simpler words, we can say that while solving any problem or changing any stat…

一个多月的时间,终于把这件事做完了!

作者 | 王磊来源 | Java中文社群&#xff08;ID&#xff1a;javacn666&#xff09;转载请联系授权&#xff08;微信ID&#xff1a;GG_Stone&#xff09;关注我的小伙伴都知道&#xff0c;前段时间磊哥搞了一个免费的模拟面试&#xff0c;但因为工作和&#xff08;面试&#xff…

简易电子密码锁制作

看到电影里面的保险箱用到的密码锁&#xff0c;于是心血来潮动手做了一个简单的密码锁&#xff0c;其有3次输入的机会&#xff0c;全错的话便进入死循环&#xff0c;一直警报&#xff0c;任何操作都无效&#xff0c;除了复位操作哈。所需素材&#xff1a;51单片机、1602液晶、蜂…

漫画:什么是红黑树?(整合版)

前段时间&#xff0c;小灰发布了红黑树相关的文章&#xff0c;分成上下篇来讲解。这一次&#xff0c;小灰把两篇文章做了整合&#xff0c;并且修正了红黑树删除部分的图片错误&#xff0c;感谢大家的指正。————— 第二天 —————————————————二叉查找树&a…

PHP高并发高负载系统架构

2019独角兽企业重金招聘Python工程师标准>>> 一、高并发和高负载的约束条件 硬件部署操作系统Web 服务器PHPMySQL测试二、解决之道——硬件篇 处理能力的提升&#xff1a;部署多颗CPU&#xff0c;选择多核心、具备更高运算频率、更大高速缓存的CPU&#xff1b; 处理…

Java ObjectInputStream registerValidation()方法与示例

ObjectInputStream类registerValidation()方法 (ObjectInputStream Class registerValidation() method) registerValidation() method is available in java.io package. registerValidation()方法在java.io包中可用。 registerValidation() method is used to register an ob…

电脑系统越来越慢,怎么删除临时文件

1.关闭"休眠"方法:打开[控制面板]→[电源选项]→[休眠],把"启用休眠"前面的勾去掉说明:休眠是系统长时间一种待机状态,使您在长时间离开电脑时保存操作状态,如果您不是经常开着电脑到别处去的话,那就把它关了吧!☆立即节省:256M2.关闭"系统还原"…

线性方程组的矩阵表示_用矩阵表示线性方程组

线性方程组的矩阵表示A Linear Equation can be represented in matrix form using a: 线性方程可以使用以下形式以矩阵形式表示 &#xff1a; Coefficient Matrix 系数矩阵 Variable Matrix and 可变矩阵和 Constant Matrix 常数矩阵 The System of linear equation in three…

为easyui添加多条件验证

easyui的验证框架&#xff0c;validatebox不能有效的支持多个条件的验证&#xff0c;比如中文用户名&#xff0c;既要验证其是中文&#xff0c;又要验证其长度不超过6位时便显得很繁琐&#xff0c;需要反复的为easyui添加验证规则。 在此实现一个多个条件验证的验证规则&#x…

图解|查找数组中最大值的5种方法!

作者 | 王磊来源 | Java中文社群&#xff08;ID&#xff1a;javacn666&#xff09;转载请联系授权&#xff08;微信ID&#xff1a;GG_Stone&#xff09;我们在一些特定场景下&#xff0c;例如查询公司员工的最高薪资&#xff0c;以及班级的最高成绩又或者是面试中都会遇到查找最…

学习C语言,要从入门到精通

1、学习C语言&#xff0c;要从入门到精通&#xff0c;需要读哪些书&#xff08;从简单的到难的排序&#xff0c;越详细越好&#xff0c;最好都能注释下选择这本书的理由&#xff09;&#xff1f; 入门阶段&#xff1a;还是老谭那本。 、、理由&#xff1a;虽然不能说它写得有多…

程序设计爬楼梯问题_楼梯案例:解决楼梯问题的C ++程序

程序设计爬楼梯问题A child is running up a staircase with N steps, and can hop 1 step, 2 steps or 3 steps at a time. Implement a method to count how many possible ways the child can run up to the stairs? You need to return number of possible ways W. 一个孩…

JDK15正式发布,新增功能预览!

作者 | 王磊来源 | Java中文社群&#xff08;ID&#xff1a;javacn666&#xff09;转载请联系授权&#xff08;微信ID&#xff1a;GG_Stone&#xff09;JDK 15 在 2020 年 9 月 15 号正式发布了&#xff0c;这次发布的主要功能有&#xff1a;JEP 339&#xff1a;EdDSA 数字签名…

[LeetCode] Longest Consecutive Sequence 求解

为什么80%的码农都做不了架构师&#xff1f;>>> 题目 Given an unsorted array of integers, find the length of the longest consecutive elements sequence. For example, Given [100, 4, 200, 1, 3, 2], The longest consecutive elements sequence is [1, 2, …