摘要:題目要求要求從單鏈表中,隨機(jī)返回一個(gè)節(jié)點(diǎn)的值,要求每個(gè)節(jié)點(diǎn)被選中的概率是相等的。假如一共有個(gè)物品,需要從其中挑選出個(gè)物品,要求確保個(gè)物品中每個(gè)物品都能夠被等概率選中。對于這種等概率問題,簡答的做法是通過隨機(jī)數(shù)獲取選中物品的下標(biāo)。
題目要求
Given a singly linked list, return a random node"s value from the linked list. Each node must have the same probability of being chosen. Follow up: What if the linked list is extremely large and its length is unknown to you? Could you solve this efficiently without using extra space? Example: // Init a singly linked list [1,2,3]. ListNode head = new ListNode(1); head.next = new ListNode(2); head.next.next = new ListNode(3); Solution solution = new Solution(head); // getRandom() should return either 1, 2, or 3 randomly. Each element should have equal probability of returning. solution.getRandom();
要求從單鏈表中,隨機(jī)返回一個(gè)節(jié)點(diǎn)的值,要求每個(gè)節(jié)點(diǎn)被選中的概率是相等的。
思路和代碼在等概率隨機(jī)選擇算法中,最經(jīng)典的算法就是蓄水池算法??梢詤⒖纪愋皖}目398 random pick index。這里再次整理一下蓄水池算法的思路和簡單證明。
假如一共有N個(gè)物品,需要從其中挑選出K個(gè)物品,要求確保N個(gè)物品中每個(gè)物品都能夠被等概率選中。對于這種等概率問題,簡答的做法是通過隨機(jī)數(shù)獲取選中物品的下標(biāo)。但是蓄水池算法允許我們從數(shù)據(jù)流的角度來隨機(jī)獲得K個(gè)物品,即在并不知道總體的樣本數(shù)有多少的情況下,隨機(jī)抽取K個(gè)物品。
蓄水池算法的思路如下:
選中前K個(gè)物品放入蓄水池
對于第K+1個(gè)物品,其被選中并替換蓄水池中任意一個(gè)物品的概率為K/(K+1)
對于第K+i個(gè)物品,其被選中并替換蓄水池中任意一個(gè)物品的概率為K/(K+i)
重復(fù)這個(gè)步驟直到K+i=N
對于這個(gè)算法,我們可以采用歸納法進(jìn)行簡單證明。已知對于前K個(gè)物品,每個(gè)物品的被選中的概率為1,滿足了K/K=1的概率。
對于K+i-1個(gè)物品,假設(shè)每個(gè)物品被選中的概率為K/(K+i-1)。證明對于前K+i個(gè)物品,每個(gè)物品被放入蓄水池中的概率為K/(K+i)
對于第K+i個(gè)物品,其被選中并替換蓄水池中任意一個(gè)物品的概率為K/(K+i)
對于之前在蓄水池中的物品,其仍在蓄水池中的概率為之前被選中在蓄水池中概率乘以這一次未被換出蓄水池的概率,即P = P(上一輪在蓄水池中) * P(這一輪沒有被替換掉)。對此進(jìn)行計(jì)算,P(上一輪在蓄水池中) * P(這一輪沒有被替換掉) = P(上一輪在蓄水池中) * (1-P(這一輪被替換掉)) = (K / (K+i-1)) * (1 - (P * 1/K)),算出P = K/(K+i)
證明對于前K+i個(gè)物品,每個(gè)物品被放入蓄水池中的概率為K/(K+i),當(dāng)K+i等于N時(shí),每個(gè)物品被選中的概率為K/N
在本題中,使用蓄水池算法的N為單鏈表的長度,K為1。
代碼如下:
private ListNode head; private Random r; /** @param head The linked list"s head. Note that the head is guaranteed to be not null, so it contains at least one node. */ public Solution(ListNode head) { this.head = head; this.r = new Random(); } /** Returns a random node"s value. */ public int getRandom() { ListNode tmp = this.head; int result = 0; int index = 1; do{ if(r.nextInt(index) == 0) { result = tmp.val; } tmp = tmp.next; index++; }while(tmp != null); return result; }
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/74862.html
Problem Given a singly linked list, return a random nodes value from the linked list. Each node must have the same probability of being chosen. Follow up:What if the linked list is extremely large and...
LeetCode[138] Copy List with Random Pointer A linked list is given such that each node contains an additional random pointer which could point to any node in the list or null. Return a deep copy of t...
摘要:大體意思就是,先復(fù)制到,順便將所有的放在再復(fù)制所有的到,順便將所有的放在最后令,令,將和分離,返回的頭結(jié)點(diǎn) Problem A linked list is given such that each node contains an additional random pointer which could point to any node in the list or null. ...
摘要:棧迭代復(fù)雜度時(shí)間空間如果不算新鏈表的空間則是思路由于隨機(jī)指針有可能產(chǎn)生環(huán)路,我們不能直接沿著隨機(jī)指針的方向一個(gè)一個(gè)復(fù)制。同時(shí)我們又不能沿著指針直接復(fù)制,因?yàn)槲覀儾恢离S機(jī)指針?biāo)赶虻男鹿?jié)點(diǎn)是哪個(gè)。 Copy List with Random Pointer A linked list is given such that each node contains an additiona...
摘要:給定一個(gè)鏈表,每個(gè)節(jié)點(diǎn)包含一個(gè)額外增加的隨機(jī)指針,該指針可以指向鏈表中的任何節(jié)點(diǎn)或空節(jié)點(diǎn)。要求返回這個(gè)鏈表的深拷貝。提示你必須返回給定頭的拷貝作為對克隆列表的引用。確定隨機(jī)節(jié)點(diǎn)的關(guān)系之后再拆分鏈表。其時(shí)間復(fù)雜度為,空間復(fù)雜度為。 給定一個(gè)鏈表,每個(gè)節(jié)點(diǎn)包含一個(gè)額外增加的隨機(jī)指針,該指針可以指向鏈表中的任何節(jié)點(diǎn)或空節(jié)點(diǎn)。 要求返回這個(gè)鏈表的深拷貝。 A linked list is g...
閱讀 3106·2021-08-03 14:05
閱讀 2152·2019-08-29 15:35
閱讀 688·2019-08-29 13:30
閱讀 3176·2019-08-29 13:20
閱讀 2541·2019-08-23 18:15
閱讀 1807·2019-08-23 14:57
閱讀 2224·2019-08-23 13:57
閱讀 1320·2019-08-23 12:10