自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

OMG,12 個(gè)精致的 Java 字符串操作小技巧,學(xué)它

開發(fā) 后端
字符串可以說是 Java 中最具有代表性的類了,似乎沒有之一哈,這就好像直播界的李佳琪,脫口秀中的李誕,一等一的大哥地位。不得不承認(rèn),最近吐槽大會刷多了,腦子里全是那些段子,寫文章都有點(diǎn)不由自主,真的是,手不由己啊。

 [[339178]]

字符串可以說是 Java 中最具有代表性的類了,似乎沒有之一哈,這就好像直播界的李佳琪,脫口秀中的李誕,一等一的大哥地位。不得不承認(rèn),最近吐槽大會刷多了,腦子里全是那些段子,寫文章都有點(diǎn)不由自主,真的是,手不由己啊。

字符串既然最常用,那就意味著面試官好這一口,就喜歡問一些字符串方面的編碼技巧,來測試應(yīng)聘者是否技術(shù)過硬,底子扎實(shí),對吧?

那這次,我就來盤點(diǎn) 12 個(gè)精致的 Java 字符串操作小技巧,來幫助大家提高一下下。在查看我給出的答案之前,最好自己先動手嘗試一遍,寫不出來答案沒關(guān)系,先思考一遍,看看自己的知識庫里是不是已經(jīng)有解決方案,有的話,就當(dāng)是溫故復(fù)習(xí)了,沒有的話,也不要擔(dān)心,剛好學(xué)一遍。

01、如何在字符串中獲取不同的字符及其數(shù)量?

這道題可以拆解為兩個(gè)步驟,第一步,找出不同的字符,第二步,統(tǒng)計(jì)出它們的數(shù)量。好像有點(diǎn)廢話,是不是?那我先來一個(gè)答案吧。

  1. public class DistinctCharsCount { 
  2.     public static void main(String[] args) { 
  3.         printDistinctCharsWithCount("itwanger"); 
  4.         printDistinctCharsWithCount("chenmowanger"); 
  5.     } 
  6.  
  7.     private static void printDistinctCharsWithCount(String input) { 
  8.         Map<CharacterInteger> charsWithCountMap = new LinkedHashMap<>(); 
  9.  
  10.         for (char c : input.toCharArray()) { 
  11.             Integer oldValue = charsWithCountMap.get(c); 
  12.  
  13.             int newValue = (oldValue == null) ? 1 : 
  14.                     Integer.sum(oldValue, 1); 
  15.  
  16.             charsWithCountMap.put(c, newValue); 
  17.         } 
  18.         System.out.println(charsWithCountMap); 
  19.     } 

程序輸出的結(jié)果是:

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

說一下我的思路:

1)聲明一個(gè) LinkedHashMap,也可以用 HashMap,不過前者可以保持字符串拆分后的順序,結(jié)果看起來更一目了然。

為什么要用 Map 呢?因?yàn)?Map 的 key 是不允許重復(fù)的,剛好可以對重復(fù)的字符進(jìn)行數(shù)量的累加。

2)把字符串拆分成字符,進(jìn)行遍歷。

3)如果 key 為 null 的話,就表明它的數(shù)量要 +1;否則的話,就在之前的值上 +1,然后重新 put 到 Map 中,這樣就覆蓋了之前的字符數(shù)量。

思路很清晰,對不對?忍不住給自己鼓個(gè)掌。

那,JDK 8 之后,Map 新增了一個(gè)很厲害的方法 merge(),一次性為多個(gè)鍵賦值:

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

有沒有很厲害?一行代碼就搞定。第一個(gè)參數(shù)為鍵,第二個(gè)參數(shù)為值,第三個(gè)參數(shù)是一個(gè) BiFunction,意思是,如果鍵已經(jīng)存在了,就重新根據(jù) BiFunction 計(jì)算新的值。

如果字符是第一次出現(xiàn),就賦值為 1;否則,就把之前的值 sum 1。

02、如何反轉(zhuǎn)字符串?

如果同學(xué)們對 StringBuilder 和 StringBuffer 很熟悉的話,這道題就很簡單,直接 reverse() 就完事,對不對?

  1. public class ReverseAString { 
  2.     public static void main(String[] args) { 
  3.         reverseInputString("沉默王二"); 
  4.     } 
  5.     private static void reverseInputString(String input) { 
  6.         StringBuilder sb = new StringBuilder(input); 
  7.         String result = sb.reverse().toString(); 
  8.         System.out.println(result); 
  9.     } 

輸出結(jié)果如下所示:

二王默沉

多說一句,StringBuffer 和 StringBuilder 很相似,前者是同步的,所有 public 方法都加了 synchronized 關(guān)鍵字,可以在多線程中使用;后者是不同步的,沒有 synchronized 關(guān)鍵字,所以性能更佳,沒有并發(fā)要求的話,就用 StringBuilder。

03、如何判斷一個(gè)字符串是前后對稱的?

什么意思呢?就好像一個(gè)字符串,前后一折,是對稱的。就像你站在鏡子前,看到了一個(gè)玉樹臨風(fēng)、閉月羞花的自己。

  1. public class PalindromeString { 
  2.     public static void main(String[] args) { 
  3.  
  4.         checkPalindromeString("沉默王二"); 
  5.         checkPalindromeString("沉默王二 二王默沉"); 
  6.     } 
  7.  
  8.     private static void checkPalindromeString(String input) { 
  9.         boolean result = true
  10.         int length = input.length(); 
  11.         for (int i = 0; i < length / 2; i++) { 
  12.             if (input.charAt(i) != input.charAt(length - i - 1)) { 
  13.                 result = false
  14.                 break; 
  15.             } 
  16.         } 
  17.         System.out.println(input + " 對稱嗎? " + result); 
  18.  
  19.     } 

輸出結(jié)果如下所示:

  1. 沉默王二 對稱嗎? false 
  2. 沉默王二 二王默沉 對稱嗎? true 

說一下我的思路:要判斷字符串對折后是否對稱,很簡單,從中間劈開,第一個(gè)字符對照最后一個(gè)字符,一旦找到不等的那個(gè),就返回 false。

注意三點(diǎn):

1)for 循環(huán)的下標(biāo)從 0 開始,到 length/2 結(jié)束。

2)下標(biāo) i 和 length-i-1 是對稱的。

3)一旦 false 就 break。

04、如何刪除所有出現(xiàn)的指定字符?

字符串類沒有提供 remove() 方法,但提供了 replaceAll() 方法,通過將指定的字符替換成空白字符就可以辦得到,對吧?

  1. public class RemoveCharFromString { 
  2.     public static void main(String[] args) { 
  3.         removeCharFromString("沉默王二"'二'); 
  4.         removeCharFromString("chenmowanger"'n'); 
  5.  
  6.     } 
  7.  
  8.     private static void removeCharFromString(String input, char c) { 
  9.         String result = input.replaceAll(String.valueOf(c), ""); 
  10.         System.out.println(result); 
  11.     } 

輸出結(jié)果如下所示:

  1. 沉默王 
  2.  
  3. chemowager 

05、如何證明字符串是不可變的?

字符串不可變的這個(gè)事我曾寫過兩篇文章,寫到最后我都要吐了。但是仍然會有一些同學(xué)弄不明白,隔段時(shí)間就有人私信我,我就不得不把之前的文章放到收藏夾,問的時(shí)候我就把鏈接發(fā)給他。

之所以造成這個(gè)混亂,有很多因素,比如說,Java 到底是值傳遞還是引用傳遞?字符串常量池是個(gè)什么玩意?

這次又不得不談,雖然煩透了,但仍然要證明啊!

  1. public class StringImmutabilityTest { 
  2.     public static void main(String[] args) { 
  3.         String s1 = "沉默王二"
  4.         String s2 = s1; 
  5.         System.out.println(s1 == s2); 
  6.  
  7.         s1 = "沉默王三"
  8.         System.out.println(s1 == s2); 
  9.  
  10.         System.out.println(s2); 
  11.     } 

輸出結(jié)果如下所示:

  1. true 
  2. false 
  3. 沉默王二 

1)String s1 = "沉默王二",Java 在字符串常量池中創(chuàng)建“沉默王二”這串字符的對象,并且把地址引用賦值給 s1

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

所以,此時(shí) s1 == s2 為 true。

3)s1 = "沉默王三",Java 在字符串常量池中創(chuàng)建“沉默王三”這串字符的對象,并且把地址引用賦值給 s1,但 s2 仍然指向的是“沉默王二”那串字符對象的地址引用。

所以,此時(shí) s1 == s2 為 false,s2 的輸出結(jié)果為“沉默王二”就證明了字符串是不可變的。

06、如何統(tǒng)計(jì)字符串中的單詞數(shù)?

這道題呢?主要針對的是英文字符串的情況。雖然中文字符串中也可以有空白字符,但不存在單詞這一說。

  1. public class CountNumberOfWordsInString { 
  2.     public static void main(String[] args) { 
  3.         countNumberOfWords("My name is Wanger"); 
  4.         countNumberOfWords("I Love Java Programming"); 
  5.         countNumberOfWords(" Java    is  very   important "); 
  6.     } 
  7.  
  8.     private static void countNumberOfWords(String line) { 
  9.         String trimmedLine = line.trim(); 
  10.         int count = trimmedLine.isEmpty() ? 0 : trimmedLine.split("\\s+").length; 
  11.  
  12.         System.out.println(count); 
  13.     } 

輸出結(jié)果如下所示:

split() 方法可以對字符串進(jìn)行拆分,參數(shù)不僅可以是空格,也可以使正則表達(dá)式代替的空白字符(多個(gè)空格、制表符);返回的是一個(gè)數(shù)組,通過 length 就可以獲得單詞的個(gè)數(shù)了。

如果對 split() 方法很感興趣的話,可以查看我之前寫的一篇文章,很飽滿,很豐富。

咦,拆分個(gè)字符串都這么講究

07、如何檢查兩個(gè)字符串中的字符是相同的?

如何理解這道題呢?比如說,字符串“沉默王二”和“沉王二默”就用了同樣的字符,對吧?比如說,字符串“沉默王二”和“沉默王三”用的字符就不同,理解了吧?

  1. public class CheckSameCharsInString { 
  2.     public static void main(String[] args) { 
  3.         sameCharsStrings("沉默王二""沉王二默"); 
  4.         sameCharsStrings("沉默王二""沉默王三"); 
  5.     } 
  6.  
  7.     private static void sameCharsStrings(String s1, String s2) { 
  8.         Set<Character> set1 = s1.chars().mapToObj(c -> (char) c).collect(Collectors.toSet()); 
  9.         System.out.println(set1); 
  10.         Set<Character> set2 = s2.chars().mapToObj(c -> (char) c).collect(Collectors.toSet()); 
  11.         System.out.println(set2); 
  12.         System.out.println(set1.equals(set2)); 
  13.     } 

輸出結(jié)果如下所示:

  1. [默, 沉, 王, 二] 
  2.  
  3. [默, 沉, 王, 二] 
  4.  
  5. true 
  6.  
  7. [默, 沉, 王, 二] 
  8.  
  9. [默, 沉, 三, 王] 
  10.  
  11. false 

上面的代碼用到了 Stream 流,看起來很陌生,但很好理解,就是把字符串拆成字符,然后收集到 Set 中,Set 是一個(gè)不允許有重復(fù)元素的集合,所以就把字符串中的不同字符收集起來了。

08、如何判斷一個(gè)字符串包含了另外一個(gè)字符串?

這道題有點(diǎn)簡單,對吧?上一道還用 Stream 流,這道題就直接送分了?不用懷疑自己,就用字符串類的 contains() 方法。

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

輸出結(jié)果如下所示:

  1. true 

contains() 方法內(nèi)部其實(shí)調(diào)用的是 indexOf() 方法:

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

09、如何在不用第三個(gè)變量的情況下交換兩個(gè)字符串?

這道題就有點(diǎn)意思了,對吧?尤其是前提條件,不使用第三個(gè)變量。

 

  1. public class SwapTwoStrings { 
  2.     public static void main(String[] args) { 
  3.         String s1 = "沉默"
  4.         String s2 = "王二"
  5.  
  6.         s1 = s1.concat(s2); 
  7.         s2 = s1.substring(0,s1.length()-s2.length()); 
  8.         s1 = s1.substring(s2.length()); 
  9.  
  10.         System.out.println(s1); 
  11.         System.out.println(s2); 
  12.     } 

輸出結(jié)果如下所示:

  1. 王二 
  2. 沉默 

說一下我的思路:

1)通過 concat() 方法把兩個(gè)字符串拼接到一塊。

2)然后通過 substring() 方法分別取出第二個(gè)字符串和第一個(gè)字符串。

10、如何從字符串中找出第一個(gè)不重復(fù)的字符?

來,上個(gè)例子來理解一下這道題。比如說字符串“沉默王沉沉默二”,第一個(gè)不重復(fù)的字符是“王”,對吧?因?yàn)?ldquo;沉”重復(fù)了,“默”重復(fù)了。

  1. public class FindNonRepeatingChar { 
  2.     public static void main(String[] args) { 
  3.         System.out.println(printFirstNonRepeatingChar("沉默王沉沉默二")); 
  4.         System.out.println(printFirstNonRepeatingChar("沉默王沉")); 
  5.         System.out.println(printFirstNonRepeatingChar("沉沉沉")); 
  6.     } 
  7.  
  8.     private static Character printFirstNonRepeatingChar(String string) { 
  9.         char[] chars = string.toCharArray(); 
  10.  
  11.         List<Character> discardedChars = new ArrayList<>(); 
  12.  
  13.         for (int i = 0; i < chars.length; i++) { 
  14.             char c = chars[i]; 
  15.  
  16.             if (discardedChars.contains(c)) 
  17.                 continue
  18.  
  19.             for (int j = i + 1; j < chars.length; j++) { 
  20.                 if (c == chars[j]) { 
  21.                     discardedChars.add(c); 
  22.                     break; 
  23.                 } else if (j == chars.length - 1) { 
  24.                     return c; 
  25.                 } 
  26.             } 
  27.         } 
  28.         return null
  29.     } 

輸出結(jié)果如下所示:

  1. 王 
  2. 默 
  3. null 

說一下我的思路:

1)把字符串拆分成字符數(shù)組。

2)聲明一個(gè) List,把重復(fù)的字符放進(jìn)去。

3)外層的 for 循環(huán),從第一個(gè)字符開始,如果已經(jīng)在 List 中,繼續(xù)下一輪。

4)嵌套的 for 循環(huán),從第一個(gè)字符的下一個(gè)字符(j = i + 1)開始遍歷,如果找到和之前字符重復(fù)的,就加入到 List 中,跳出內(nèi)層的循環(huán);如果找到最后(j == chars.length - 1)也沒有找到,就是第一個(gè)不重復(fù)的字符,對吧?

11、如何檢查字符串中只包含數(shù)字?

有一種很傻的解法,就是用 Long.parseLong(string) 對字符串強(qiáng)轉(zhuǎn),如果轉(zhuǎn)不成整形,那肯定不是只包含數(shù)字,對吧?

但這種方法也太不可取了,所以還得換一種巧妙的,就是使用正則表達(dá)式。

  1. public class CheckIfStringContainsDigitsOnly { 
  2.     public static void main(String[] args) { 
  3.         digitsOnlyString("123 沉默王二"); 
  4.         digitsOnlyString("123"); 
  5.  
  6.     } 
  7.  
  8.     private static void digitsOnlyString(String string) { 
  9.         if (string.matches("\\d+")) { 
  10.             System.out.println("只包含數(shù)字的字符串:" + string); 
  11.         } 
  12.     } 

輸出結(jié)果如下所示:

  1. 只包含數(shù)字:123 

12、如何實(shí)現(xiàn)字符串的深度拷貝?

由于字符串是不可變的,所以可以直接使用“=”操作符將一個(gè)字符串拷貝到另外一個(gè)字符串,并且互不影響。

  1. public class JavaStringCopy { 
  2.     public static void main(String args[]) { 
  3.         String str = "沉默王二"
  4.         String strCopy = str; 
  5.  
  6.         str = "沉默王三"
  7.         System.out.println(strCopy); 
  8.     } 

輸出結(jié)果如下所示:

  1. 沉默王二 

這個(gè)例子和之前證明字符串是不可變的例子幾乎沒什么差別,對吧?這的確是因?yàn)樽址遣豢勺兊?,如果是可變對象的話,深度拷貝就要注意了,最好使?new 關(guān)鍵字返回新的對象。

  1. public Book getBook() { 
  2.     Book clone = new Book(); 
  3.     clone.setPrice(this.book.getPrice()); 
  4.     clone.setName(this.book.getName()); 
  5.     return clone; 

關(guān)于不可變對象,請點(diǎn)擊下面的鏈接查看我之前寫了一篇文章。

這次要說不明白immutable類,我就怎么地

最后

希望這 12 個(gè)精致的字符串操作小技巧可以幫助大家鞏固一波基礎(chǔ),反正我自己已經(jīng)重新鞏固了一波,很有收獲的樣子,感覺就像是“一群小精靈在我腦子里跳舞一樣”,學(xué)它就對了!

本文轉(zhuǎn)載自微信公眾號「沉默王二」,可以通過以下二維碼關(guān)注。轉(zhuǎn)載本文請聯(lián)系沉默王二公眾號。

 

責(zé)任編輯:武曉燕 來源: 沉默王二
相關(guān)推薦

2023-11-27 16:01:59

JavaScrip技巧

2025-02-21 12:30:00

字符串前端JavaScript

2020-12-31 07:56:02

JavaScript 字符串技巧

2025-02-20 09:00:00

字符串JavaScript代碼

2024-05-16 11:09:40

Python字符串代碼

2023-02-09 16:15:27

JavaScript編程語言字符串

2023-04-17 16:19:32

編程語言JavaScript開發(fā)

2021-06-11 18:08:00

Java字符串拼接

2024-09-06 17:32:55

字符串Python

2022-03-10 08:01:06

CSS技巧選擇器

2022-11-24 10:34:05

CSS前端

2024-06-11 00:09:00

JavaScript模式變量

2016-05-10 10:16:13

JavaScript技巧

2024-05-10 09:26:26

Python字符串

2022-05-18 10:56:58

Java字符串編碼

2020-08-12 22:03:17

JavaScript開發(fā)技術(shù)

2022-07-18 08:18:11

字符JavaJDK

2009-12-11 13:16:04

PHP查詢字符串

2009-07-15 17:20:45

Jython字符串

2009-11-27 10:24:25

PHP字符串操作
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號