字符串能够说是 Java 中最具备表明性的类了,彷佛没有之一哈,这就好像直播界的李佳琪,脱口秀中的李诞,一等一的大哥地位。不得不认可,最近吐槽大会刷多了,脑子里全是那些段子,写文章都有点情不自禁,真的是,手不禁己啊。git
字符串既然最经常使用,那就意味着面试官好这一口,就喜欢问一些字符串方面的编码技巧,来测试应聘者是否技术过硬,底子扎实,对吧?面试
那此次,我就来盘点 12 个精致的 Java 字符串操做小技巧,来帮助你们提升一下下。在查看我给出的答案以前,最好本身先动手尝试一遍,写不出来答案不要紧,先思考一遍,看看本身的知识库里是否是已经有解决方案,有的话,就当是温故复习了,没有的话,也不要担忧,恰好学一遍。正则表达式
这道题能够拆解为两个步骤,第一步,找出不一样的字符,第二步,统计出它们的数量。好像有点废话,是否是?那我先来一个答案吧。编程
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。
若是同窗们对 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。
什么意思呢?就好像一个字符串,先后一折,是对称的。就像你站在镜子前,看到了一个玉树临风、闭月羞花的本身。
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。
字符串类没有提供 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
字符串不可变的这个事我曾写过两篇文章,写到最后我都要吐了。可是仍然会有一些同窗弄不明白,隔段时间就有人私信我,我就不得不把以前的文章放到收藏夹,问的时候我就把连接发给他。
之因此形成这个混乱,有不少因素,好比说,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 的输出结果为“沉默王二”就证实了字符串是不可变的。
这道题呢?主要针对的是英文字符串的状况。虽然中文字符串中也能够有空白字符,但不存在单词这一说。
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() 方法很感兴趣的话,能够查看我以前写的一篇文章,很饱满,很丰富。
咦,拆分个字符串都这么讲究
如何理解这道题呢?好比说,字符串“沉默王二”和“沉王二默”就用了一样的字符,对吧?好比说,字符串“沉默王二”和“沉默王三”用的字符就不一样,理解了吧?
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 是一个不容许有重复元素的集合,因此就把字符串中的不一样字符收集起来了。
这道题有点简单,对吧?上一道还用 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; }
这道题就有点意思了,对吧?尤为是前提条件,不使用第三个变量。
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() 方法分别取出第二个字符串和第一个字符串。
十、如何从字符串中找出第一个不重复的字符?
来,上个例子来理解一下这道题。好比说字符串“沉默王沉沉默二”,第一个不重复的字符是“王”,对吧?由于“沉”重复了,“默”重复了。
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)也没有找到,就是第一个不重复的字符,对吧?
有一种很傻的解法,就是用 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
因为字符串是不可变的,因此能够直接使用“=”操做符将一个字符串拷贝到另一个字符串,而且互不影响。
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 个精致的字符串操做小技巧能够帮助你们巩固一波基础,反正我本身已经从新巩固了一波,颇有收获的样子,感受就像是“一群小精灵在我脑子里跳舞同样”,学它就对了!
本文转载自微信公众号「沉默王二」,能够经过如下二维码关注。转载本文请联系沉默王二公众号。
【编辑推荐】
【责任编辑:武晓燕 TEL:(010)68476606】