<kbd id="afajh"><form id="afajh"></form></kbd>
<strong id="afajh"><dl id="afajh"></dl></strong>
    <del id="afajh"><form id="afajh"></form></del>
        1. <th id="afajh"><progress id="afajh"></progress></th>
          <b id="afajh"><abbr id="afajh"></abbr></b>
          <th id="afajh"><progress id="afajh"></progress></th>

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

          共 4592字,需瀏覽 10分鐘

           ·

          2020-08-26 15:28

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

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

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

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

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

          public?class?DistinctCharsCount?{
          ????public?static?void?main(String[]?args)?{
          ????????printDistinctCharsWithCount("itwanger");
          ????????printDistinctCharsWithCount("chenmowanger");
          ????}

          ????private?static?void?printDistinctCharsWithCount(String?input)?{
          ????????Map?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);
          ????}
          }

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

          {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}

          說(shuō)一下我的思路:

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

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

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

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

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

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

          private?static?void?printDistinctCharsWithCountMerge(String?input)?{
          ????Map?charsWithCountMap?=?new?LinkedHashMap<>();

          ????for?(char?c?:?input.toCharArray())?{
          ????????charsWithCountMap.merge(c,?1,?Integer::sum);
          ????}
          ????System.out.println(charsWithCountMap);
          }

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

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

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

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

          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);
          ????}
          }

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

          二王默沉

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

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

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

          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?2;?i++)?{
          ????????????if?(input.charAt(i)?!=?input.charAt(length?-?i?-?1))?{
          ????????????????result?=?false;
          ????????????????break;
          ????????????}
          ????????}
          ????????System.out.println(input?+?"?對(duì)稱(chēng)嗎??"?+?result);

          ????}
          }

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

          沉默王二?對(duì)稱(chēng)嗎??false
          沉默王二?二王默沉?對(duì)稱(chēng)嗎??true

          說(shuō)一下我的思路:要判斷字符串對(duì)折后是否對(duì)稱(chēng),很簡(jiǎn)單,從中間劈開(kāi),第一個(gè)字符對(duì)照最后一個(gè)字符,一旦找到不等的那個(gè),就返回 false。

          注意三點(diǎn):

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

          2)下標(biāo) i 和 length-i-1 是對(duì)稱(chēng)的。

          3)一旦 false 就 break。

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

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

          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);
          ????}
          }

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

          沉默王
          chemowager

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

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

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

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

          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);
          ????}
          }

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

          true
          false
          沉默王二

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

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

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

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

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

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

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

          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);
          ????}
          }

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

          4
          4
          4

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

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

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

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

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

          public?class?CheckSameCharsInString?{
          ????public?static?void?main(String[]?args)?{
          ????????sameCharsStrings("沉默王二",?"沉王二默");
          ????????sameCharsStrings("沉默王二",?"沉默王三");
          ????}

          ????private?static?void?sameCharsStrings(String?s1,?String?s2)?{
          ????????Set?set1?=?s1.chars().mapToObj(c?->?(char)?c).collect(Collectors.toSet());
          ????????System.out.println(set1);
          ????????Set?set2?=?s2.chars().mapToObj(c?->?(char)?c).collect(Collectors.toSet());
          ????????System.out.println(set2);
          ????????System.out.println(set1.equals(set2));
          ????}
          }

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

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

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

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

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

          public?class?StringContainsSubstring?{
          ????public?static?void?main(String[]?args)?{
          ????????String?s1?=?"沉默王二";
          ????????String?s2?=?"沉默";

          ????????System.out.println(s1.contains(s2));
          ????}
          }

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

          true

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

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

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

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

          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);
          ????}
          }

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

          王二
          沉默

          說(shuō)一下我的思路:

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

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

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

          來(lái),上個(gè)例子來(lái)理解一下這道題。比如說(shuō)字符串“沉默王沉沉默二”,第一個(gè)不重復(fù)的字符是“王”,對(duì)吧?因?yàn)椤俺痢敝貜?fù)了,“默”重復(fù)了。

          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?discardedChars?=?new?ArrayList<>();

          ????????for?(int?i?=?0;?i?????????????char?c?=?chars[i];

          ????????????if?(discardedChars.contains(c))
          ????????????????continue;

          ????????????for?(int?j?=?i?+?1;?j?????????????????if?(c?==?chars[j])?{
          ????????????????????discardedChars.add(c);
          ????????????????????break;
          ????????????????}?else?if?(j?==?chars.length?-?1)?{
          ????????????????????return?c;
          ????????????????}
          ????????????}
          ????????}
          ????????return?null;
          ????}
          }

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



          null

          說(shuō)一下我的思路:

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

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

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

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

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

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

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

          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("只包含數(shù)字的字符串:"?+?string);
          ????????}
          ????}
          }

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

          只包含數(shù)字:123

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

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

          public?class?JavaStringCopy?{
          ????public?static?void?main(String?args[])?{
          ????????String?str?=?"沉默王二";
          ????????String?strCopy?=?str;

          ????????str?=?"沉默王三";
          ????????System.out.println(strCopy);
          ????}
          }

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

          沉默王二

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

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

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

          這次要說(shuō)不明白immutable類(lèi),我就怎么地

          最后

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


          ------------------

          公眾號(hào):沉默王二(ID:cmower)
          CSDN:沉默王二
          bilibili:沉默王二
          知乎:沉默王二

          這是一個(gè)有顏值卻假裝靠才華茍且的程序員,你知道,他的文章風(fēng)趣幽默,讀起來(lái)就好像花錢(qián)一樣爽快。

          長(zhǎng)按下圖二維碼關(guān)注,你將感受到一個(gè)有趣的靈魂,且每篇文章都有干貨。

          ------------------


          原創(chuàng)不易,莫要白票,如果覺(jué)得有點(diǎn)用的話,請(qǐng)毫不留情地素質(zhì)三連吧,分享、點(diǎn)贊、在看,我不挑,因?yàn)檫@將是我寫(xiě)作更多優(yōu)質(zhì)文章的最強(qiáng)動(dòng)力。

          PS:昨天結(jié)婚紀(jì)念日收到了眾多讀者的祝福,留言已經(jīng)爆了,只能公布 100 條,真的非常感謝各位;贊賞更是令我感動(dòng),心意都收到了,我會(huì)一一手動(dòng)回復(fù)。必須得一句,有個(gè)讀者直接贊賞了 66 元,我已經(jīng)牢牢記住他了!

          今天是七夕,我就不再秀了。但有一說(shuō)一,各位不僅是我的讀者,也是我放在心上的人!

          順帶公布一下上周日的中獎(jiǎng)名單!這已經(jīng)是第四周送書(shū)了,對(duì)不對(duì)?收到書(shū)籍的一些讀者也給我發(fā)來(lái)了感謝信,甚至有的在朋友圈都曬了,說(shuō)這輩子的三連都要給我——謝謝謝謝。

          第 1:青衫;第 50:雪行莽蒼?;第 100:Zhenqiang;第 150:磨蹭先生;第 200:龔思同。

          瀏覽 27
          點(diǎn)贊
          評(píng)論
          收藏
          分享

          手機(jī)掃一掃分享

          分享
          舉報(bào)
          評(píng)論
          圖片
          表情
          推薦
          點(diǎn)贊
          評(píng)論
          收藏
          分享

          手機(jī)掃一掃分享

          分享
          舉報(bào)
          <kbd id="afajh"><form id="afajh"></form></kbd>
          <strong id="afajh"><dl id="afajh"></dl></strong>
            <del id="afajh"><form id="afajh"></form></del>
                1. <th id="afajh"><progress id="afajh"></progress></th>
                  <b id="afajh"><abbr id="afajh"></abbr></b>
                  <th id="afajh"><progress id="afajh"></progress></th>
                  91精品国产99久久久久久红楼 | 日韩AV无码一区二区三区 | 在线看特及黄色片 | 日本成人电影在线观看 | 嫩穴av |