contributed by < jim12312321
>
首先我們先將測驗 1 中的題目程式碼,依照函式拆程若干個部份並分別解釋。
hlist_node
結構中可以看到有提供兩子結構,分別為 next
和 pprev
。其中 next
是 hlist_node
的指標,所以他應該指向下一個 hlist_node
;而 pprev
是指標的指標,所以他應該指向前一個 hlist_node *
,也就是前一個 hlist_node
的指標。hlist_head
就比較簡單了,裡面只有一個子結構 first
用以指向第一個節點。bits
用來初始化 hash table 的大小power of 2 的翻譯是「2 的冪」,不是「冪次」
架構示意圖
根據 bits
的大小,將 hash table 初始化成長度為 的 hash table 。
這段程式用來描述一個 hash_key
的結構,其中 key
用來儲存對應的 hash key
, data
用來儲存資料(在這題中就是準備用來找兩數之和的數字的索引值), hlist_node
則是用以提供各式操作的結構。
container_of
用以透過子結構提取上層結構
hash
利用 val * GOLDEN_RATIO_32
製造雜湊值,又因如老師在註解提到的高位較具隨機性,因此 shift 32 bits 後回傳成雜湊值。
無聊去查了一下 GOLDEN_RATIO_32
為什麼要設定成那個值,明明 1640531527 (0x61C88647 轉成 10 進位後的值) 看起來跟黃金比例沒什麼關係呀?
原來這是 的近似值 ( 就是黃金比例),而且在 linux kernel 中也是用他來實作 hash 呢。
Link: linux code
TODO: 以 語法重新排版數學表示式
jservImage Not Showing Possible ReasonsLearn More →
- The image file may be corrupted
- The server hosting the image is unavailable
- The image path is incorrect
- The image format is not supported
若 map
中存在對應的 key
,則將 hash_key
中的 data
回傳,否則回傳 NULL 。
首先要先找到這個 key 所對應的 hash table 位置,並設定該位置中的第一個節點
接著先做 AAA
,然後如果 first
不是 NULL ,則將該 hash
值的 linked list
做成雙向鏈結串列,之後將原先的 first
替換成 n
,最後做 BBB
。
因此可以知道我們還缺乏設定新節點的 pprev
和 next
所以可知
AAA : n->next = first
BBB : n->pprev = &h->first
將 hash table 中所有佔用的記憶體空間釋放
步驟解析:
target - nums[i]
已經存在於 hash table ,則將當前數字與存於 hash table 中的索引值取出,並且離開 for-loopTODO: 延伸題目2
本測驗為實做 LeetCode 82 的程式碼填空
LeetCode 82 中要答題者將重複的節點全部移除
該架構裡面存有一個屬性( val
用來儲存該結點的值)和另一個架構( next
用來指向下一個 ListNode
)
什麼叫做「子架構」?用語要精準
步驟解析:
COND1
有沒有發生COND2
存在則 head 繼續往後移動,然後對 head->next 繼續進行一次 deleteDuplicates()
deleteDuplicates()
,然後回傳 headCOND1
應該要可以判斷是否要發生重複的情況(當然重複的前提就是下個節點還存在),因此可得知COND1 = head->next && head->val == head->next->val
COND2
終須判斷是否下下一個仍然存在且為同一個值,因此可推斷COND2 = head->next->next && head->next->val == head->next->next->val
COND1
的解法取代掉原本我認為的 COND2
的解法,因此可知COND2 = head->next && head->val == head->next->val
廢話不多說,先貼程式碼
步驟解析:
*indirect
(就是 head
) 以及 *indrect->next
是否存在,若有其中有一者不存在,則跳出迴圈,回傳 head
;否則繼續步驟 2離開步驟 2 時有兩種情況,但在步驟 3 中沒有先對這兩種情況判斷
I. 下一個結點中的值與目前結點中的值不相同
II. 下一個結點不存在
dup
為當前結點,並判斷後續 list 中有多少與之重複的值(當然也還是要判斷下一個結點是否存在)*indirect
指向 dup->next
*indirect != dup
(這樣的情況就代表最後一個結點與前面有相同值,需移除, e.g. [1,2,3,3]) ,則將 *indirect
指向 NULL雜記:
這算是真正第一次自己利用a pointer of a pointer
做出題目。
還蠻開心的,不過最後還是躲不掉要進行步驟 4. 的判斷,不知道還能不能更簡化,但是目前想不到了,先這樣。
TODO: 延伸問題2
本測驗為結合 lab0-c 所提供的 list.h 實作 linux 風格的 LRU Cache
這題希望建構一個實作一個 LRU Cache ,建構結構體 LRUCache 並提供以下函式
capacity
的 LRU Cachekey
,若有則回傳該值,沒有則回傳 -1key
將 value
插入 LRU Cache 中,若 key
已存在則更新 LRU Cache 中同樣 key
中的值,若沒有則將最少使用的 key
移除,再將這對 key-value
pair 加入 LRU Cache 。hhead
的儲存空間hlink
dhead
後的 linked list 結點,排序排在越前面代表越近期被使用hhead
後的 linked list 結點,會依照 key 值所分配到的 hash 值加在對應的 hhead
後。e.g.
初始化新的 LRU Cache 並為其配置記憶體空間,配置好的新結點 (list_head
) 也要進行初始化,將其變成雙向鏈結 linked list 。
釋放架構空間時要走訪所有結點並逐一釋放,最終再釋放 Cache 本身。
因為要走訪 linked list 中所有結點且在這個過程中要釋放該結點,因此我們可以知道必須使用 safe 模式的 linked list 走訪
MMM1 = list_for_each_entry_safe
依照特定的 hash 值走訪 Cache 中相對應的 hheads[hash]
並將該結點移到 dhead
的第一個結點。
e.g. 找 key2
將 Node2 的 dlink
移到 dhead
的最前面。
因為要走訪 linked list 中所有結點但不需要在這個過程中釋放該結點,因此可以不用使用 safe 模式的 linked list 走訪
MMM2 = list_for_each_entry
依照特定的 key 放進 Cache 中。若該 key 已經存在,則更新 key 中的值並返回。若不存在則將最沒被使用的值移除 (dhead
中的最後一個結點)並將新結點放進對應的 hheads[hash]
中與 dhead
中的第一個。
e.g. put(key3,value3)
將最沒有被使用的 Node 移除。
在 MMM3
所在的那一段中,需要走訪對應 hash 的 linked list(hheads[hash]) 並尋找 key
是否已經存在。
因此我們可以知道需要走訪 linked list 並且不需要在這個過程中移除結點,因此可以使用非 safe 的走訪版本
MMM3 = list_for_each_entry
在 MMM4
所在的那一段中,需要走訪對應 hash 的 linked list(hheads[hash]) 的最後一個結點並將其移除。
因此我們可以知道需要走訪 linked list 中的最後一個結點
MMM4 = list_last_entry
TODO: 延伸問題 1 中的改進與測試、2
本測驗為結合 lab0-c 所提供的 list.h 實作 linux 風格的 Find Longest Consecutive Sequence
在一個未被排序過的 array 中尋找最長的連續序列( e.g. [1,2,3] 連續序列)
在 linked list 中尋找特定的 num
若存在就回傳他所在的結點,若無則回傳 NULL。
步驟解析:
n_size
設立一個 heads
並初始化其中所有 head
nums
中的所有值,並且如果該值並未在 heads
中的任一 head
找到,就將該值新增進對應的 head
nums
中的所有值,並且尋找該值在 heads
中有沒有形成連續序列,首先先往當前的 head
的左側(較小側)走訪,若遇到不連續則終止。接著對右邊也進行相同的事情,最後存取當前找到最長的長度我們可以知道在 LLL
中,應該要往當前數值的左側(比當前數值還小的那側)走訪,且應該先 -1 (因為 left = num
)
LLL = --left
我們可以知道在 RRR
中,應該要往當前數值的右側(比當前數值還大的那側)走訪,且應該先 +1 (因為 right = num
)
RRR = ++right
TODO: 延伸問題 1 中的改進與測試、2