摘要:實(shí)驗(yàn)及數(shù)據(jù)可視化本節(jié),我將使用不同的數(shù)字作為乘子,對(duì)超過萬個(gè)英文單詞進(jìn)行哈希運(yùn)算,并計(jì)算哈希算法的沖突率。同時(shí)我們注意觀察質(zhì)數(shù)作為乘子時(shí),哈希值的分布情況。這也就不難解釋為什么數(shù)字作為乘子時(shí),算出哈希值的沖突率如此之高的原因了。
1. 背景
某天,我在寫代碼的時(shí)候,無意中點(diǎn)開了 String hashCode 方法。然后大致看了一下 hashCode 的實(shí)現(xiàn),發(fā)現(xiàn)并不是很復(fù)雜。但是我從源碼中發(fā)現(xiàn)了一個(gè)奇怪的數(shù)字,也就是本文的主角31。這個(gè)數(shù)字居然不是用常量聲明的,所以沒法從字面意思上推斷這個(gè)數(shù)字的用途。后來帶著疑問和好奇心,到網(wǎng)上去找資料查詢一下。在看完資料后,默默的感嘆了一句,原來是這樣啊。那么到底是哪樣呢?在接下來章節(jié)里,請(qǐng)大家?guī)е闷嫘暮臀医议_數(shù)字31的用途之謎。
2. 選擇數(shù)字31的原因在詳細(xì)說明 String hashCode 方法選擇數(shù)字31的作為乘子的原因之前,我們先來看看 String hashCode 方法是怎樣實(shí)現(xiàn)的,如下:
public int hashCode() { int h = hash; if (h == 0 && value.length > 0) { char val[] = value; for (int i = 0; i < value.length; i++) { h = 31 * h + val[i]; } hash = h; } return h; }
上面的代碼就是 String hashCode 方法的實(shí)現(xiàn),是不是很簡單。實(shí)際上 hashCode 方法核心的計(jì)算邏輯只有三行,也就是代碼中的 for 循環(huán)。我們可以由上面的 for 循環(huán)推導(dǎo)出一個(gè)計(jì)算公式,hashCode 方法注釋中已經(jīng)給出。如下:
s[0]*31^(n-1) + s[1]*31^(n-2) + ... + s[n-1]
這里說明一下,上面的 s 數(shù)組即源碼中的 val 數(shù)組,是 String 內(nèi)部維護(hù)的一個(gè) char 類型數(shù)組。這里我來簡單推導(dǎo)一下這個(gè)公式:
假設(shè) n=3 i=0 -> h = 31 * 0 + val[0] i=1 -> h = 31 * (31 * 0 + val[0]) + val[1] i=2 -> h = 31 * (31 * (31 * 0 + val[0]) + val[1]) + val[2] h = 31*31*31*0 + 31*31*val[0] + 31*val[1] + val[2] h = 31^(n-1)*val[0] + 31^(n-2)*val[1] + val[2]
上面的公式包括公式的推導(dǎo)并不是本文的重點(diǎn),大家了解了解即可。接下來來說說本文的重點(diǎn),即選擇31的理由。從網(wǎng)上的資料來看,一般有如下兩個(gè)原因:
第一,31是一個(gè)不大不小的質(zhì)數(shù),是作為 hashCode 乘子的優(yōu)選質(zhì)數(shù)之一。另外一些相近的質(zhì)數(shù),比如37、41、43等等,也都是不錯(cuò)的選擇。那么為啥偏偏選中了31呢?請(qǐng)看第二個(gè)原因。
第二、31可以被 JVM 優(yōu)化,31 * i = (i << 5) - i。
上面兩個(gè)原因中,第一個(gè)需要解釋一下,第二個(gè)比較簡單,就不說了。下面我來解釋第一個(gè)理由。一般在設(shè)計(jì)哈希算法時(shí),會(huì)選擇一個(gè)特殊的質(zhì)數(shù)。至于為啥選擇質(zhì)數(shù),我想應(yīng)該是可以降低哈希算法的沖突率。至于原因,這個(gè)就要問數(shù)學(xué)家了,我?guī)缀蹩梢院雎缘臄?shù)學(xué)水平解釋不了這個(gè)原因。上面說到,31是一個(gè)不大不小的質(zhì)數(shù),是優(yōu)選乘子。那為啥同是質(zhì)數(shù)的2和101(或者更大的質(zhì)數(shù))就不是優(yōu)選乘子呢,分析如下。
這里先分析質(zhì)數(shù)2。首先,假設(shè) n = 6,然后把質(zhì)數(shù)2和 n 帶入上面的計(jì)算公式。并僅計(jì)算公式中次數(shù)最高的那一項(xiàng),結(jié)果是2^5 = 32,是不是很小。所以這里可以斷定,當(dāng)字符串長度不是很長時(shí),用質(zhì)數(shù)2做為乘子算出的哈希值,數(shù)值不會(huì)很大。也就是說,哈希值會(huì)分布在一個(gè)較小的數(shù)值區(qū)間內(nèi),分布性不佳,最終可能會(huì)導(dǎo)致沖突率上升。
上面說了,質(zhì)數(shù)2做為乘子會(huì)導(dǎo)致哈希值分布在一個(gè)較小區(qū)間內(nèi),那么如果用一個(gè)較大的大質(zhì)數(shù)101會(huì)產(chǎn)生什么樣的結(jié)果呢?根據(jù)上面的分析,我想大家應(yīng)該可以猜出結(jié)果了。就是不用再擔(dān)心哈希值會(huì)分布在一個(gè)小的區(qū)間內(nèi)了,因?yàn)?b>101^5 = 10,510,100,501。但是要注意的是,這個(gè)計(jì)算結(jié)果太大了。如果用 int 類型表示哈希值,結(jié)果會(huì)溢出,最終導(dǎo)致數(shù)值信息丟失。盡管數(shù)值信息丟失并不一定會(huì)導(dǎo)致沖突率上升,但是我們暫且先認(rèn)為質(zhì)數(shù)101(或者更大的質(zhì)數(shù))也不是很好的選擇。最后,我們?cè)賮砜纯促|(zhì)數(shù)31的計(jì)算結(jié)果: 31^5 = 28629151,結(jié)果值相對(duì)于32和10,510,100,501來說。是不是很nice,不大不小。
上面用了比較簡陋的數(shù)學(xué)手段證明了數(shù)字31是一個(gè)不大不小的質(zhì)數(shù),是作為 hashCode 乘子的優(yōu)選質(zhì)數(shù)之一。接下來我會(huì)用詳細(xì)的實(shí)驗(yàn)來驗(yàn)證上面的結(jié)論,不過在驗(yàn)證前,我們先看看 Stack Overflow 上關(guān)于這個(gè)問題的討論,Why does Java"s hashCode() in String use 31 as a multiplier?。其中排名第一的答案引用了《Effective Java》中的一段話,這里也引用一下:
The value 31 was chosen because it is an odd prime. If it were even and the multiplication overflowed, information would be lost, as multiplication by 2 is equivalent to shifting. The advantage of using a prime is less clear, but it is traditional. A nice property of 31 is that the multiplication can be replaced by a shift and a subtraction for better performance: 31 * i == (i << 5) - i`. Modern VMs do this sort of optimization automatically.
簡單翻譯一下:
選擇數(shù)字31是因?yàn)樗且粋€(gè)奇質(zhì)數(shù),如果選擇一個(gè)偶數(shù)會(huì)在乘法運(yùn)算中產(chǎn)生溢出,導(dǎo)致數(shù)值信息丟失,因?yàn)槌硕喈?dāng)于移位運(yùn)算。選擇質(zhì)數(shù)的優(yōu)勢并不是特別的明顯,但這是一個(gè)傳統(tǒng)。同時(shí),數(shù)字31有一個(gè)很好的特性,即乘法運(yùn)算可以被移位和減法運(yùn)算取代,來獲取更好的性能:31 * i == (i << 5) - i,現(xiàn)代的 Java 虛擬機(jī)可以自動(dòng)的完成這個(gè)優(yōu)化。
排名第二的答案設(shè)這樣說的:
As Goodrich and Tamassia point out, If you take over 50,000 English words (formed as the union of the word lists provided in two variants of Unix), using the constants 31, 33, 37, 39, and 41 will produce less than 7 collisions in each case. Knowing this, it should come as no surprise that many Java implementations choose one of these constants.
這段話也翻譯一下:
正如 Goodrich 和 Tamassia 指出的那樣,如果你對(duì)超過 50,000 個(gè)英文單詞(由兩個(gè)不同版本的 Unix 字典合并而成)進(jìn)行 hash code 運(yùn)算,并使用常數(shù) 31, 33, 37, 39 和 41 作為乘子,每個(gè)常數(shù)算出的哈希值沖突數(shù)都小于7個(gè),所以在上面幾個(gè)常數(shù)中,常數(shù) 31 被 Java 實(shí)現(xiàn)所選用也就不足為奇了。
上面的兩個(gè)答案完美的解釋了 Java 源碼中選用數(shù)字 31 的原因。接下來,我將針對(duì)第二個(gè)答案就行驗(yàn)證,請(qǐng)大家繼續(xù)往下看。
3. 實(shí)驗(yàn)及數(shù)據(jù)可視化本節(jié),我將使用不同的數(shù)字作為乘子,對(duì)超過23萬個(gè)英文單詞進(jìn)行哈希運(yùn)算,并計(jì)算哈希算法的沖突率。同時(shí),我也將針對(duì)不同乘子算出的哈希值分布情況進(jìn)行可視化處理,讓大家可以直觀的看到數(shù)據(jù)分布情況。本次實(shí)驗(yàn)所使用的數(shù)據(jù)是 Unix/Linux 平臺(tái)中的英文字典文件,文件路徑為 /usr/share/dict/words。
3.1 哈希值沖突率計(jì)算計(jì)算哈希算法沖突率并不難,比如可以一次性將所有單詞的 hash code 算出,并放入 Set 中去除重復(fù)值。之后拿單詞數(shù)減去 set.size() 即可得出沖突數(shù),有了沖突數(shù),沖突率就可以算出來了。當(dāng)然,如果使用 JDK8 提供的流式計(jì)算 API,則可更方便算出,代碼片段如下:
public static Integer hashCode(String str, Integer multiplier) { int hash = 0; for (int i = 0; i < str.length(); i++) { hash = multiplier * hash + str.charAt(i); } return hash; } /** * 計(jì)算 hash code 沖突率,順便分析一下 hash code 最大值和最小值,并輸出 * @param multiplier * @param hashs */ public static void calculateConflictRate(Integer multiplier, Listhashs) { Comparator cp = (x, y) -> x > y ? 1 : (x < y ? -1 : 0); int maxHash = hashs.stream().max(cp).get(); int minHash = hashs.stream().min(cp).get(); // 計(jì)算沖突數(shù)及沖突率 int uniqueHashNum = (int) hashs.stream().distinct().count(); int conflictNum = hashs.size() - uniqueHashNum; double conflictRate = (conflictNum * 1.0) / hashs.size(); System.out.println(String.format("multiplier=%4d, minHash=%11d, maxHash=%10d, conflictNum=%6d, conflictRate=%.4f%%", multiplier, minHash, maxHash, conflictNum, conflictRate * 100)); }
結(jié)果如下:
從上圖可以看出,使用較小的質(zhì)數(shù)做為乘子時(shí),沖突率會(huì)很高。尤其是質(zhì)數(shù)2,沖突率達(dá)到了 55.14%。同時(shí)我們注意觀察質(zhì)數(shù)2作為乘子時(shí),哈希值的分布情況。可以看得出來,哈希值分布并不是很廣,僅僅分布在了整個(gè)哈??臻g的正半軸部分,即 0 ~ 231-1。而負(fù)半軸 -231 ~ -1,則無分布。這也證明了我們上面斷言,即質(zhì)數(shù)2作為乘子時(shí),對(duì)于短字符串,生成的哈希值分布性不佳。然后再來看看我們之前所說的 31、37、41 這三個(gè)不大不小的質(zhì)數(shù),表現(xiàn)都不錯(cuò),沖突數(shù)都低于7個(gè)。而質(zhì)數(shù) 101 和 199 表現(xiàn)的也很不錯(cuò),沖突率很低,這也說明哈希值溢出并不一定會(huì)導(dǎo)致沖突率上升。但是這兩個(gè)家伙一言不合就溢出,我們認(rèn)為他們不是哈希算法的優(yōu)選乘子。最后我們?cè)賮砜纯?32 和 36 這兩個(gè)偶數(shù)的表現(xiàn),結(jié)果并不好,尤其是 32,沖突率超過了了50%。盡管 36 表現(xiàn)的要好一點(diǎn),不過和 31,37相比,沖突率還是比較高的。當(dāng)然并非所有的偶數(shù)作為乘子時(shí),沖突率都會(huì)比較高,大家有興趣可以自己驗(yàn)證。
3.2 哈希值分布可視化上一節(jié)分析了不同數(shù)字作為乘子時(shí)的沖突率情況,這一節(jié)來分析一下不同數(shù)字作為乘子時(shí),哈希值的分布情況。在詳細(xì)分析之前,我先說說哈希值可視化的過程。我原本是打算將所有的哈希值用一維散點(diǎn)圖進(jìn)行可視化,但是后來找了一圈,也沒找到合適的畫圖工具。加之后來想了想,一維散點(diǎn)圖可能不合適做哈希值可視化,因?yàn)檫@里有超過23萬個(gè)哈希值。也就意味著會(huì)在圖上顯示超過23萬個(gè)散點(diǎn),如果不出意外的話,這23萬個(gè)散點(diǎn)會(huì)聚集的很密,有可能會(huì)變成一個(gè)大黑塊,就失去了可視化的意義了。所以這里選擇了另一種可視化效果更好的圖表,也就是 excel 中的平滑曲線的二維散點(diǎn)圖(下面簡稱散點(diǎn)曲線圖)。當(dāng)然這里同樣沒有把23萬散點(diǎn)都顯示在圖表上,太多了。所以在實(shí)際繪圖過程中,我將哈??臻g等分成了64個(gè)子區(qū)間,并統(tǒng)計(jì)每個(gè)區(qū)間內(nèi)的哈希值數(shù)量。最后將分區(qū)編號(hào)做為X軸,哈希值數(shù)量為Y軸,就繪制出了我想要的二維散點(diǎn)曲線圖了。這里舉個(gè)例子說明一下吧,以第0分區(qū)為例。第0分區(qū)數(shù)值區(qū)間是[-2147483648, -2080374784),我們統(tǒng)計(jì)落在該數(shù)值區(qū)間內(nèi)哈希值的數(shù)量,得到 <分區(qū)編號(hào), 哈希值數(shù)量> 數(shù)值對(duì),這樣就可以繪圖了。分區(qū)代碼如下:
/** * 將整個(gè)哈??臻g等分成64份,統(tǒng)計(jì)每個(gè)空間內(nèi)的哈希值數(shù)量 * @param hashs */ public static Mappartition(List hashs) { // step = 2^32 / 64 = 2^26 final int step = 67108864; List nums = new ArrayList<>(); Map statistics = new LinkedHashMap<>(); int start = 0; for (long i = Integer.MIN_VALUE; i <= Integer.MAX_VALUE; i += step) { final long min = i; final long max = min + step; int num = (int) hashs.parallelStream() .filter(x -> x >= min && x < max).count(); statistics.put(start++, num); nums.add(num); } // 為了防止計(jì)算出錯(cuò),這里驗(yàn)證一下 int hashNum = nums.stream().reduce((x, y) -> x + y).get(); assert hashNum == hashs.size(); return statistics; }
本文中的哈希值是用整形表示的,整形的數(shù)值區(qū)間是 [-2147483648, 2147483647],區(qū)間大小為 2^32。所以這里可以將區(qū)間等分成64個(gè)子區(qū)間,每個(gè)自子區(qū)間大小為 2^26。詳細(xì)的分區(qū)對(duì)照表如下:
分區(qū)編號(hào) | 分區(qū)下限 | 分區(qū)上限 | 分區(qū)編號(hào) | 分區(qū)下限 | 分區(qū)上限 |
---|---|---|---|---|---|
0 | -2147483648 | -2080374784 | 32 | 0 | 67108864 |
1 | -2080374784 | -2013265920 | 33 | 67108864 | 134217728 |
2 | -2013265920 | -1946157056 | 34 | 134217728 | 201326592 |
3 | -1946157056 | -1879048192 | 35 | 201326592 | 268435456 |
4 | -1879048192 | -1811939328 | 36 | 268435456 | 335544320 |
5 | -1811939328 | -1744830464 | 37 | 335544320 | 402653184 |
6 | -1744830464 | -1677721600 | 38 | 402653184 | 469762048 |
7 | -1677721600 | -1610612736 | 39 | 469762048 | 536870912 |
8 | -1610612736 | -1543503872 | 40 | 536870912 | 603979776 |
9 | -1543503872 | -1476395008 | 41 | 603979776 | 671088640 |
10 | -1476395008 | -1409286144 | 42 | 671088640 | 738197504 |
11 | -1409286144 | -1342177280 | 43 | 738197504 | 805306368 |
12 | -1342177280 | -1275068416 | 44 | 805306368 | 872415232 |
13 | -1275068416 | -1207959552 | 45 | 872415232 | 939524096 |
14 | -1207959552 | -1140850688 | 46 | 939524096 | 1006632960 |
15 | -1140850688 | -1073741824 | 47 | 1006632960 | 1073741824 |
16 | -1073741824 | -1006632960 | 48 | 1073741824 | 1140850688 |
17 | -1006632960 | -939524096 | 49 | 1140850688 | 1207959552 |
18 | -939524096 | -872415232 | 50 | 1207959552 | 1275068416 |
19 | -872415232 | -805306368 | 51 | 1275068416 | 1342177280 |
20 | -805306368 | -738197504 | 52 | 1342177280 | 1409286144 |
21 | -738197504 | -671088640 | 53 | 1409286144 | 1476395008 |
22 | -671088640 | -603979776 | 54 | 1476395008 | 1543503872 |
23 | -603979776 | -536870912 | 55 | 1543503872 | 1610612736 |
24 | -536870912 | -469762048 | 56 | 1610612736 | 1677721600 |
25 | -469762048 | -402653184 | 57 | 1677721600 | 1744830464 |
26 | -402653184 | -335544320 | 58 | 1744830464 | 1811939328 |
27 | -335544320 | -268435456 | 59 | 1811939328 | 1879048192 |
28 | -268435456 | -201326592 | 60 | 1879048192 | 1946157056 |
29 | -201326592 | -134217728 | 61 | 1946157056 | 2013265920 |
30 | -134217728 | -67108864 | 62 | 2013265920 | 2080374784 |
31 | -67108864 | 0 | 63 | 2080374784 | 2147483648 |
接下來,讓我們對(duì)照上面的分區(qū)表,對(duì)數(shù)字2、3、17、31、101的散點(diǎn)曲線圖進(jìn)行簡單的分析。先從數(shù)字2開始,數(shù)字2對(duì)于的散點(diǎn)曲線圖如下:
上面的圖還是很一幕了然的,乘子2算出的哈希值幾乎全部落在第32分區(qū),也就是 [0, 67108864)數(shù)值區(qū)間內(nèi),落在其他區(qū)間內(nèi)的哈希值數(shù)量幾乎可以忽略不計(jì)。這也就不難解釋為什么數(shù)字2作為乘子時(shí),算出哈希值的沖突率如此之高的原因了。所以這樣的哈希算法要它有何用啊,拖出去斬了吧。接下來看看數(shù)字3作為乘子時(shí)的表現(xiàn):
3作為乘子時(shí),算出的哈希值分布情況和2很像,只不過稍微好了那么一點(diǎn)點(diǎn)。從圖中可以看出絕大部分的哈希值最終都落在了第32分區(qū)里,哈希值的分布性很差。這個(gè)也沒啥用,拖出去槍斃5分鐘吧。在看看數(shù)字17的情況怎么樣:
數(shù)字17作為乘子時(shí)的表現(xiàn),明顯比上面兩個(gè)數(shù)字好點(diǎn)了。雖然哈希值在第32分區(qū)和第34分區(qū)有一定的聚集,但是相比較上面2和3,情況明顯好好了很多。除此之外,17作為乘子算出的哈希值在其他區(qū)也均有分布,且較為均勻,還算是一個(gè)不錯(cuò)的乘子吧。
接下來來看看我們本文的主角31了,31作為乘子算出的哈希值在第33分區(qū)有一定的小聚集。不過相比于數(shù)字17,主角31的表現(xiàn)又好了一些。首先是哈希值的聚集程度沒有17那么嚴(yán)重,其次哈希值在其他區(qū)分布的情況也要好于17??傊?,選31,準(zhǔn)沒錯(cuò)啊。
最后再來看看大質(zhì)數(shù)101的表現(xiàn),不難看出,質(zhì)數(shù)101作為乘子時(shí),算出的哈希值分布情況要好于主角31,有點(diǎn)喧賓奪主的意思。不過不可否認(rèn)的是,質(zhì)數(shù)101的作為乘子時(shí),哈希值的分布性確實(shí)更加均勻。所以如果不在意質(zhì)數(shù)101容易導(dǎo)致數(shù)據(jù)信息丟失問題,或許其是一個(gè)更好的選擇。
經(jīng)過上面的分析與實(shí)踐,我想大家應(yīng)該明白了 String hashCode 方法中選擇使用數(shù)字31作為乘子的原因了。本文本質(zhì)是一篇簡單的科普文而已,并沒有銀彈
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/70231.html
摘要:接下來分析幾個(gè)常見的實(shí)現(xiàn)方式。再哈希法再哈希法,就是出現(xiàn)沖突后采用其他的哈希函數(shù)計(jì)算,直到不再?zèng)_突為止。,其中為不同的哈希函數(shù)。 由表及里,循序漸進(jìn),請(qǐng)往下看。隨手點(diǎn)贊是對(duì)作者最大的鼓勵(lì)!^0^。 什么是哈希表 引用:嚴(yán)蔚敏 《數(shù)據(jù)結(jié)構(gòu)(C語言版)》中的內(nèi)容 showImg(https://segmentfault.com/img/bVZlmE?w=800&h=364); 哈希表就是 ...
摘要:相關(guān)的文章網(wǎng)上很多了寫這個(gè)主要是按自己的思路進(jìn)行記錄是什么中的實(shí)現(xiàn)是一個(gè)本地方法生成一個(gè)表征當(dāng)前對(duì)象實(shí)例的特征值具體的實(shí)現(xiàn)根據(jù)的實(shí)現(xiàn)可能會(huì)不同中實(shí)際計(jì)算的函數(shù)的實(shí)現(xiàn)如下為時(shí)是直接使用的內(nèi)存地址但默認(rèn)使用的是的隨 hashcode相關(guān)的文章網(wǎng)上很多了, 寫這個(gè)主要是按自己的思路進(jìn)行記錄 hashCode是什么 Object中的hashCode實(shí)現(xiàn)是一個(gè)本地方法, 生成一個(gè)表征當(dāng)前對(duì)象實(shí)例...
摘要:類提供方法和方法來通知系統(tǒng)進(jìn)行垃圾回收清理系統(tǒng)資源,并提供了和方法來加載文件和動(dòng)態(tài)鏈接庫。返回按照標(biāo)準(zhǔn)的規(guī)定,對(duì)兩個(gè)參數(shù)進(jìn)行余數(shù)運(yùn)算。 與用戶互動(dòng) 運(yùn)行Java程序的參數(shù) //Java程序入口:main()方法 public static void main(String[] args){...} public修飾符:Java類由JVM調(diào)用,為了讓JVM可以自由調(diào)用這個(gè)main()方...
數(shù)組 數(shù)組是我們比較熟悉的一種數(shù)據(jù)結(jié)構(gòu):固定大小,索引(下標(biāo))對(duì)應(yīng)的槽位用以存儲(chǔ)數(shù)據(jù): showImg(https://segmentfault.com/img/bV7rJ8?w=480&h=119); 我們要在數(shù)組中查找一個(gè)值,比如紅框圈中的 元素5 ,可以通過遍歷或者排序后二分的方式達(dá)到目的。沒有更快捷的查找方式了嗎?顯然是有的,比如Map。我們對(duì)存 / 取動(dòng)一動(dòng)腦筋,還是上圖的那些元素,假如...
摘要:注意這里我說的是一般情況下,因?yàn)楣K惴ㄐ枰骖櫺阅芘c準(zhǔn)確性,是有一定概率出現(xiàn)重復(fù)的情況的。哈希算法實(shí)際上是數(shù)學(xué)家和計(jì)算機(jī)基礎(chǔ)科學(xué)家研究的領(lǐng)域。 背景 做了幾年 CRUD 工程師,深感自己的計(jì)算機(jī)基礎(chǔ)薄弱,在看了幾篇大牛的分享文章之后,發(fā)現(xiàn)很多人都是通過刷 LeetCode 來提高自己的算法水平。的確,通過分析解決實(shí)際的問題,比自己潛心研究書本效率還是要高一些。 一直以來遇到底層自己無...
閱讀 1946·2021-11-23 09:51
閱讀 1252·2019-08-30 15:55
閱讀 1625·2019-08-30 15:44
閱讀 771·2019-08-30 14:11
閱讀 1152·2019-08-30 14:10
閱讀 922·2019-08-30 13:52
閱讀 2641·2019-08-30 12:50
閱讀 625·2019-08-29 15:04