版權(quán)說明:本文檔由用戶提供并上傳,收益歸屬內(nèi)容提供方,若內(nèi)容存在侵權(quán),請進(jìn)行舉報或認(rèn)領(lǐng)
文檔簡介
鏈表(LinkedList)是一種常見的基礎(chǔ)數(shù)據(jù)結(jié)構(gòu),是一種線性表,但是并不會按線性的順序數(shù)據(jù),而是在每一個節(jié)點里存到下一個節(jié)點的指針(Pointer由于不必須按順序,鏈表在插入的時候可以達(dá)到O(1)的復(fù)雜度,比另一種線性——順序表快得多,但是查找一個節(jié)點或者特定編號的節(jié)點則需要O(n)的時間,而順序表相應(yīng)的時間復(fù)雜度分別是O(n)和O(1)。示例輸入輸入1->2->3->4->5-輸出5->4->3->2->1-進(jìn)階next指向之前的節(jié)點就可以了。1→2→3→nullnull←1←2←3next指針改為指向publicpublicclassReverseLinkedListpublicListNodereverseList(ListNodehead){ListNodecurr=head;ListNodeprev=while(curr!=ListNodetempNext=curr.next;curr.next=prev;prev=curr;curr=tempNext;}return}}時間復(fù)雜度:O(n)nO(n)假設(shè)鏈表為( n1→n2→…→nk?1→nk→nk+1→…→nm→ nknk+1nm n1→n2→…→nk?1→nk→nk+1←…←nk+1nk,所以,應(yīng)該有nk+1.next=nkpublicpublicListNodereverseList(ListNodehead)if(head==null||head.next==null){returnhead;}ListNoderestHead=ListNodereversedRest restHead.next=head;head.next=null;returnreversedRest;}時間復(fù)雜度:時間復(fù)雜度:O(n),假設(shè)n是鏈表的長度,那么時間復(fù)雜度為O(n)。輸入:輸入:1->2->41->3-list1list2。只要它們都不為空,就取出當(dāng)前它們各自的頭節(jié)點就(sentinelpublicpublicclassMergeTwoSortedListspublicListNodemergeTwoLists(ListNodel1,ListNodel2)ListNoderesultPrev=newListNode(-ListNodeListNodeprev=while(l1!=null&&l2!=nullif(l1.val<=l2.val){prev.next=l1;prev=l1;l1=l1.next;}elseprev.next=l2;prev=l2;l2=}}prev.next=(l1==null)?l2:return}}時間復(fù)雜度:O(n+m)n和ml1和l2while循環(huán)的次數(shù)不會超過兩個鏈表的長度之和。所有其他操作的時間復(fù)雜度都是常數(shù)級別的,因此總的時間復(fù)雜度為O(n+m)publicpublicListNodemergeTwoLists(ListNodel1,ListNodel2)if(l1==nullreturnelseif(l2==nullreturnif(l1.val<=l2.vall1.next=mergeTwoLists(l1.next,return}elsel2.next=mergeTwoLists(l1,return}}時間復(fù)雜度:O(n+m)nnm分別為兩個鏈表的長度。因為每次調(diào)用遞歸都l1l2的頭節(jié)點(直到至少有一個鏈表為空mergeTwoList至多只會遞歸調(diào)用每個節(jié)點一次。因此,時間復(fù)雜度取決于合并后的鏈表長度,即O(n+m)。空間復(fù)雜度:O(nm)nmmergeTwoLists函數(shù)時需要消耗??臻g,??臻g的大小取決于遞歸調(diào)用的深度。結(jié)束遞歸調(diào)用時mergeTwoLists函數(shù)最多調(diào)用n+m次,因此空間復(fù)雜度為O(n+m)。n給定一個鏈表給定一個鏈表1->2->3->4->5,n1->2->3-n保證是有效的。javaJVMGC最簡單的想法是,我們首先從頭節(jié)點開始對鏈表進(jìn)行一次遍歷,得到鏈表的長度L。然后,我們再從頭節(jié)點開始對鏈表進(jìn)行一次遍歷,當(dāng)遍歷到第L-N+1個節(jié)點時,它就NpublicpublicclassRemoveNthNodeFromEndpublicListNoderemoveNthFromEnd(ListNodehead,intn)intl=ListNodesentinel=newListNode(-1);sentinel.next=head;ListNodecurr=for(inti=0;i<l-n;i++){curr=curr.next;}curr.next=return}publicstaticintgetLength(ListNodeintlength=while(head!=null){length++;head=}return}}時間復(fù)雜度:O(L)L是鏈表的長度。只用了兩次遍歷,是線性時間復(fù)雜度。n個節(jié)點就是需要刪除的節(jié)點,并且目前棧頂?shù)墓?jié)點就是待刪除節(jié)publicpublicListNoderemoveNthFromEnd(ListNodehead,intn){ListNodesentinel=newListNode(-1);sentinel.next=Stack<ListNode>stack=newStack<>();ListNodecurr=sentinel;while(curr!=null){curr=curr.next;}for(inti=0;i<n;i++){}stack.peek().next=return}<=空間復(fù)雜度:O(L)Lfirstsecondfirstsecond超前N個節(jié)點。Nfirst遍歷到鏈表的末尾(null)時,second就恰L-N+1,也就是倒數(shù)第N個節(jié)點了。publicpublicListNoderemoveNthFromEnd(ListNodehead,intn){ListNodesentinel=newListNode(-1);sentinel.next=ListNodefirst=sentinel,second=for(inti=0;i<n+1;i++){first=first.next;}while(first!=null){first=first.next;second=second.next;}second.next=return}時間復(fù)雜度:O(L)L是鏈表的長度。這次真正實現(xiàn)了一次遍歷。據(jù)結(jié)構(gòu)也就是說它通過把關(guān)鍵字值映射到表中一個位置來記錄以加快查找的速度。pair的key快速value。哈希表在不考慮的情況下,插入、刪除和操作時間復(fù)雜度均為O(1)設(shè)計一個哈希表,有兩個問題需要去解決如何設(shè)計哈希方法(哈希函數(shù)哈希方法(hashmethod,也叫哈希函數(shù))會將鍵值映射到某塊空間映射到同一個空間,這種情況稱為“哈希碰撞”(HashCollision,也叫“哈?!,F(xiàn)(Java中就是用鏈表來實現(xiàn)的示例1:輸入輸入輸出輸入輸入輸出publicintpublicintsingleNumber(int[]nums){List<Integer>singleList=newArrayList<>();for(Integernum:nums){if(singleList.contains(num))}return}時間復(fù)雜度:O(n^2)numsO(n)判斷是否存在這個數(shù)字,再花費O(n)的時間,所以總循環(huán)時間為O(n^2)。空間復(fù)雜度:O(n)nnumsHashMap中,這樣查詢的時候不就不用再遍歷一次了。publicpublicintsingleNumber(int[]nums)Map<Integer,Integer>singleMap=newfor(Integernum:nums)if(singleMap.get(num)!=null)singleMap.put(num,}return}時間復(fù)雜度:O(n)。forO(n)HashMapget操作時間復(fù)雜O(1)。空間復(fù)雜度:O(n)。HashMapnumssetset2,publicintpublicintsingleNumber3(int[]nums){Set<Integer>set=newHashSet<>();intarraySum=0;IntegersetSum=for(intnum:nums){arraySum+=num;}}for(Integernum:set)setSum+=num;returnsetSum*2-})空間復(fù)雜度:O(n)。HashSetnums0XORXOR0XOR滿換律和結(jié)合XORpublicpublicintsingleNumber(int[]nums)intresult=for(intnum:nums)result^=num;return}時間復(fù)雜度:O(n)n是數(shù)組長度。只需要對數(shù)組遍歷一次。給定一個未排序的整數(shù)數(shù)組nums,找出數(shù)字連續(xù)的最長序列(不要求序列元素在原O(n)示例輸入:輸入:nums[1,2,3,4]4。示例2:輸入:輸入:nums0<=nums.length<=-109<=nums[i]<=(publicpublicclassLongestConsecutiveSequencepublicintlongestConsecutive(int[]nums)intmaxLength=for(inti=0;i<nums.length;intcurrNum=intcurrLength=while(contains(nums,currNum+1)){currLength++;currNum}if(currLength>maxLength)maxLength=currLength;}return}publicstaticbooleancontains(int[]nums,intfor(intnum:numsif(num==xreturn}return}}}O(N^3)。publicpublicintlongestConsecutive(int[]nums)intmaxLength=HashSet<Integer>hashSet=newfor(intnum:nums)for(inti=0;i<nums.length;intcurrNum=intcurrLength=while(hashSet.contains(currNum+1)){currLength++;currNum}if(currLength>maxLength)maxLength=currLength;}}return}時間復(fù)雜度:O(N^2)HashSet需要。后面由于簡化了內(nèi)層循環(huán)中判斷后繼的過程只耗費O(1)時間所以最終是內(nèi)外兩重循環(huán)情況下時間復(fù)雜度為O(N^2)。O(N)的內(nèi)存空間。xx,x+1,x+2,?,x+y的連續(xù)序列。并且,我們可以確定,這種情況得到的結(jié)果(連續(xù)序列的長度x為publicpublicintlongestConsecutive(int[]nums)intmaxLength=HashSet<Integer>hashSet=newfor(intnum:nums)for(inti=0;i<nums.length;i++)intcurrNum=intcurrLength=ifif(!hashSet.contains(currNum-1))while(hashSet.contains(currNum+1)){}if(currLength>maxLength)maxLength=}}return}時間復(fù)雜度:O(N)O(n)的時間復(fù)雜度,只有當(dāng)一個數(shù)是連續(xù)序列的第LRU緩存機制運用你所掌握的數(shù)據(jù)結(jié)構(gòu),設(shè)計一個LRU(最近最少使用)緩存機制。實現(xiàn)LRUCache類:LRUCache(intcapacity)以正整數(shù)作為容量capacityLRUintget(intkey)key-1O(1)["LRUCache","put","put","get","put","get","put","get","get",[[2],[1,1],[2,2],[1],[3,3],[2],[4,4],[1],[3],[null,null,null,1,null,-1,null,-1,3,LRUCacheLRUCachelRUCachenewLRUCache(2);lRUCache.put(1,1);//緩存是{1=1}lRUCache.put(22);{1=1, lRUCache.put(3,3);//該操作會使得關(guān)鍵字2作廢,緩存是{1=1,3=3} //返回-1(未找到)lRUCache.put(4,4);//該操作會使得關(guān)鍵字1作廢,緩存是{4=4,3=3} //返回-1(未找到) 1<=capacity<=0<=key<=0<=value<=3*104get和LRU(Leastrecentlyused,最近最少使用)果數(shù)據(jù)最近被過,那么將來被的幾率也更高。可以用一個HashMap來作為緩存的數(shù)據(jù)結(jié)構(gòu)。這樣,我們的和插入,就都可以以HashMap要有一個容量限制;而且當(dāng)達(dá)LRU的策略刪除最近最少使用的那個數(shù)據(jù)。這就要求須把數(shù)據(jù),按照一定的線性結(jié)構(gòu)排列起來,的數(shù)據(jù)放在后面,新數(shù)據(jù)的插入可“頂?shù)糇钋懊娴牟怀5臄?shù)據(jù)這種數(shù)據(jù)結(jié)構(gòu)其實可以用鏈表來實現(xiàn)。所以,我們最終可以使用一個哈希表+LRU哈希表”——LinkedHashMap。它本身繼承了HashMap,而它的節(jié)點Entry除了繼承自HashMap.Nodebeforeafter兩個指針,從而實現(xiàn)了雙向鏈表。publicpublicclassLRUCacheextendsLinkedHashMap<Integer,privateintpublicLRUCache(intcapacity){super(capacity,0.75f,true);this.capacity=capacity;}publicintget(intkey)return}publicvoidput(intkey,intvalue)super.put(key,}protectedbooleanremoveEldestEntry(Map.Entry<Integer,Integer>eldest){returnreturnsize()>}}publicpublicclassLRUCacheclassNodeintkey;intvalue;Nodeprev;NodepublicNode()publicNode(intkey,intvalue)this.key=this.value=}}privateintcapacity;privateintsize;privateNodehead,tail;publicLRUCache(intcapacity)this.capacity=this.size=head=newtail=newhead.next=tail.prev=}publicintget(intkey)Nodenode=if(node==null)return-}return}publicvoidput(intkey,intvalue)Nodenode=if(node!=null)node.value=}elseNodenewNode=newNode(key,hashMap.put(key,sizeif(size>capacity)Nodetail=size--}}}privatevoidmoveToTail(Nodenode)}privatevoidremoveNode(Nodenode.prev.next=node.next.prev=}privatevoidaddToTail(Nodenode)node.next node.prev=tail.prev.next tail.prev }privateNoderemoveHead()NoderealHead=return}}時間復(fù)雜度:O(1)HashMapputget空間復(fù)雜度:O(capacity),因為哈希表和雙向鏈表最多capacity+1個元素(超出緩capacity+1。(LIFO)在具體應(yīng)用中通常用鏈表或者數(shù)組來實現(xiàn)。隊列只允許在后端(稱為rear)進(jìn)行插入操作,端(稱為front)進(jìn)行刪除操作。(Deque:doubleended8首先出隊:有元素,時間復(fù)雜度O(n,并不是最理想的方式。因此,一般是用大頂堆(MaxHeap,有時也叫最大堆)來實現(xiàn)最大優(yōu)先隊列,每一次5119O(logn)。O(logn)。push(x)--xpop()--top()--empty()--注意你只能使用隊列的基本操作--pushtoback,peek/popfromfront,size,isempty這些操作是合法的。listdeque(雙端隊列)來模,只要是標(biāo)準(zhǔn)的隊列操作即可。你可以假設(shè)所有操作都是有效的(例如,poptop操式;區(qū)別在于,元素進(jìn)出的方式不同??梢栽黾右粋€隊列來做輔助。我們記原始負(fù)責(zé)數(shù)據(jù)的隊列為queue1,新增的輔助queue2。queue2中本沒有數(shù)據(jù),所以當(dāng)前元素一定在隊首。queue1:ab接下來,就讓queue1queue2中來。這樣,queue2queue2:xab最后queue2的內(nèi)容給queue1做然后清空queue2在代碼上,queue1queue2指向的內(nèi)容即可。queue1:xabqueue1publicclasspublicclassMyStack{Queue<Integer>queue1;Queue<Integer>queue2;publicMyStack(){queue1=newqueue2=new}publicvoidpush(intx)whilequeue2.offer(queue1.poll()}Queue<Integer>temp=queue1;queue1=queue2;queue2=}publicintpop()return}publicinttop()return}publicbooleanempty()return}}O(n)O(1)pushqueue1nn+1queue2,總計2n+1次操作。每次出隊和入隊操作的時間復(fù)雜度都是O(1),因此入棧操作的時間復(fù)雜度是O(n)。popqueue1的隊首元素出隊,時間復(fù)雜度是O(1)。topqueue1O(1)。isEmptyqueue1O(1)空間復(fù)雜度:O(n),其中n是棧內(nèi)的元素。需要使用兩個隊列棧內(nèi)的元素xqueue1,x移動到隊首了。publicpublicclassMyStack2Queue<Integer>publicMyStack2()queue=new}publicvoidpush(intx)intl=for(inti=0;i<l;queue.offer(queue.poll()}}publicintpop()return}publicinttop()return}publicbooleanempty()return}}O(n)O(1)pushqueue1nn+1queue2,總計2n+1次操作。每次出隊和入隊操作的時間復(fù)雜度都是O(1),因此入棧操作的時間復(fù)雜度是O(n)。popqueue1的隊首元素出隊,時間復(fù)雜度是O(1)。topqueue1O(1)。isEmptyqueue1O(1)空間復(fù)雜度:O(n),其中n是棧內(nèi)的元素。需要使用兩個隊列棧內(nèi)的元素emptyMyQueuevoidpush(intx)xintpop()intpeek()booleanempty()true你只能使用標(biāo)準(zhǔn)的棧操作——pushtotop,peek/popfromtop,isempty你所使用的語言也許不支持棧。你可以使用list或者deque(雙端隊列)來模擬你能否實現(xiàn)每個操作均攤時間復(fù)雜度為O(1)的隊列?換句話說,執(zhí)行n個操作的總時間復(fù)雜度為O(n),即使其中一個操作可能花費較長時間。["MyQueue","push","push","peek","pop",[[],[1],[2],[],[],[null,null,null,1,1,MyQueuemyQueue=newMyQueue();myQueue.push(1);MyQueuemyQueue=newMyQueue();myQueue.push(1);//queueis:myQueue.push(2);//queueis:[1,2](leftmostisfrontofthequeue)myQueue.peek();//return1myQueue.pop();//return1,queueismyQueue.empty();//return1<=x<=100push、pop、peek(poppeek操作FIFO的特性,我們需要將入棧的元素次序進(jìn)行反轉(zhuǎn),這樣在出隊時就可以按照入隊順序依元素是最先入隊的元素,而入隊的元素要壓入棧底。我們可以用一個棧來元素的最終順序(隊列順序,記作stack1;用另一個進(jìn)行輔stack2。stack2pushstack2stack1,進(jìn)行反轉(zhuǎn)。publicclasspublicclassMyQueue{Stack<Integer>stack1;Stack<Integer>stack2;publicMyQueue(){stack1=newstack2=new}publicvoidpush(intx)while}while}}publicintpop()return}publicintpeek()return}publicbooleanempty()return}}除新元外,所有元素都會被壓入兩次,彈出兩次。新元素被壓入兩次,彈出一次(stack1清空之后把新元素直接壓入,就只壓入一次了)4n+3n是隊列的大小。由于入棧操作和彈出操作的時O(1)O(n)需要額外的內(nèi)存來隊列中的元素stack1stack1中所有元stack2stack2的棧頂元素。stack1stack2中的棧頂元素就可以了。publicclassMyQueue2Stack<Integer>Stack<Integer>publicMyQueue2()stack1=newstack2=new}publicvoidpush(intx)}publicintpop()ifwhile}}return}publicintpeek()ifwhile}}return}publicbooleanempty()returnstack1.isEmpty()&&}}}入隊空間復(fù)雜度:O(n)。需要額外的內(nèi)存(stack1和stack2共同)來隊列元素出隊時間復(fù)雜度:攤還復(fù)雜度O(1),情況下的時間復(fù)雜度在情況下,stack2為空,算法需要執(zhí)行while循環(huán)進(jìn)行反轉(zhuǎn)。具體過程是從nnstack2n代表隊列的大小。這個過程產(chǎn)生了2n步操作,時間復(fù)雜度為O(n)。但當(dāng)stack2非空時,只需要直接彈棧,算法就只有O(1)的時間復(fù)雜度。均攤下來,O(1)。攤還分析(Amortizedysis,均攤法,用來評價某個數(shù)據(jù)結(jié)構(gòu)的一系列操作的平均,攤還分析的在于情況下的操作一旦發(fā)生了一次,那么在未來很長一段時間都,是平均操作。在攤還分析中,不涉及概率,并且保證在情況下每一個操作的平均性能'(',')','{','}','[',示例輸入輸入輸出輸入輸入輸出輸入輸入輸出輸入輸入輸出輸入輸入 輸出: 由于給定字符串中只包含'(',')','{','}','[',']',所以我們不需要額外考慮字符false。publicpublicclassValidParenthesespublicbooleanisValid(Strings){Deque<Character>stack=newLinkedList<>();for(inti=0;i<s.length();i++){charc=if(c=='('}elseif(c=='['}elseif(c=='{'}elseifif(stack.isEmpty())returnfalse;charright=stack.pop();if(c!=right)return}}return}}時間復(fù)雜度:O(n)ns的長度。只需要遍歷一次字符串。n個非負(fù)整數(shù),用來表示柱狀圖中各個柱子的高度。每個柱子彼此相鄰,且寬度為1。1[2,1,5,6,2,3]10示例輸入輸入輸出(即矩形面積計算中的長和寬publicpublicclassLargestRectangleInHistogrampublicintlargestRectangleArea1(int[]heights)intlargestArea=forfor(intleft=0;left<heights.length;left++intcurrHeight=for(intright=left;right<heights.length;right++){currHeight=(heights[right]<currHeight)?heights[right]:intcurrArea=(right-left+1)*currHeight;largestArea=(currArea>largestArea)?currArea:}}return}}publicpublicintlargestRectangleArea(int[]heights)intlargestArea=for(inti=0;i<heights.length;i++intintheight=intleft=i,right=while(left>=0if(heights[left]<height)break;left--;}while(right<heights.lengthif(heights[right]<height)break;right++;}intwidth=right-left-intcurrArea=height*largestArea=currArea>largestArea?currArea:}return}O(N^2)。publicpublicintlargestRectangleArea(int[]heights)intn=heights.length;int[]lefts=newint[n];int[]rights=newint[n];intlargestArea=0;for(inti=0;i<n;i++){intheight=heights[i];intleft=i-1;while(left>=0if(heights[left]<height)break;left=lefts[left];}lefts[i]=}for(inti=n-1;i>=0;i--intheight=intright=i+while(right<nif(heights[right]<height)break;right=rights[right];}rights[i]=}for(inti=0;i<n;i++intintcurrArea=(rights[i]-lefts[i]-1)*heights[i];largestArea=currArea>largestArea?currArea:largestArea;}return}時間復(fù)雜度:O(N)。我們發(fā)現(xiàn),while循環(huán)內(nèi)的判斷比對總體數(shù)量其實是有限的。每次O(N)??臻g復(fù)雜度:O(N)n[6,7,5,2,4,5,9,3]66-1。隨后我們將6入棧。我們枚舉7,由于6<7,因此不會移除棧頂元素,所以7左側(cè)的柱子是6置為0。隨后7入棧。棧:[6(0),7(1)]57≥576≥5,再移除棧頂元素6。此時棧為空,所以5左側(cè)的柱子是「哨兵,位置為?1。隨后5入棧。接下來的枚舉過程也大同小異。我們枚舉2,移除棧頂元素5,得到2左側(cè)的?1。將2入棧。453,45。將它們?nèi)霔?。棧:[2(3),4(4),5(5),9(6)]39,54323。將3入棧。棧:[2(3[?1,0,?1,?1,3,4,5,3]用相同的方法,我們從右向左進(jìn)行遍歷,也可以得到它們右側(cè)的柱子編號分別為[2,2,3,8,7,7,7,8],這里位置8看作右側(cè)的“哨兵publicpublicintlargestRectangleArea(int[]heights)intn=heights.length;int[]lefts=newint[n];int[]rights=newint[n];intlargestArea=0;Stack<Integer>stack=newfor(inti=0;i<n;i++while(!stack.isEmpty()&&heights[stack.peek()]>=heights[i]){}lefts[i]=stack.isEmpty()?-1:stack.peek();}for(inti=n-1;i>=0;i--while(!stack.isEmpty()&&heights[stack.peek()]>=heights[i]){}rights[i]=stack.isEmpty()?n:stack.peek();}for(inti=0;i<n;i++intcurrArea=(rights[i]-lefts[i]-1)*heights[i];largestArea=currArea>largestArea?currArea:largestArea;}return}的總時間復(fù)雜度為O(N)。publicpublicintlargestRectangleArea(int[]heights)intn=heights.length;int[]lefts=newint[n];int[]rights=newfor(inti=0;i<n;i++)rights[i]=intlargestArea=Stack<Integer>stack=newfor(inti=0;i<n;i++while(!stack.isEmpty()&&heights[stack.peek()]>=heights[i]){rights[stack.peek()]=i;}lefts[i]=stack.isEmpty()?-1:stack.peek();}for(inti=0;i<n;i++intcurrArea=(rights[i]-lefts[i]-1)*heights[i];largestArea=currArea>largestArea?currArea:largestArea;}return}空間復(fù)雜度:O(N)O(N)。將數(shù)值以哈希(hash)或分桶(bucket)的形式直接映射到空間來實現(xiàn)的。選擇排序(Selection冒泡排序(Bubble插入排序(InsertionO(1)。排序(S。1959年由S發(fā)明,是第一個突破O(n2)的排序算法,是簡單插入排序的改進(jìn)版。它與插入排序的不同之處在于,它會優(yōu)先比較距離較遠(yuǎn)的元素排序又叫縮小增量。1。的增量序列如Hibbard經(jīng)過復(fù)雜證明可使得時間復(fù)雜度為O(n^3/2)。歸并排序是建立在歸并操作上的一種有效的排序算法。該算法是采用分治法andConquer)2-路歸并。O(nlognlists“基準(zhǔn)”(pivot,中心,支點;publicpublicclassQuickSortpublicstaticvoidqSort(int[]nums,intstart,intendif(start>=endintmid=partition(nums,start,qSort(nums,start,mid-1qSortqSort(nums,mid+1,end}privatestaticintpartition(int[]nums,intstart,intendintpivot=intleft=intright=while(left<rightwhile(left<right&&nums[right]>=pivot)right--;nums[left]=while(left<right&&nums[left]<=pivot)left++;nums[right]=}nums[left]=return}}堆排序(Heap(MaxHeapO(nlogn)。k不是很大并且序列比較集中時,計數(shù)排序是一個很有效的排序算法。桶排序(Bucketsort)的工作原理:假設(shè)輸入數(shù)據(jù)服從均勻分布,將數(shù)據(jù)分到有限數(shù)基數(shù)排序(Radix到最。O(n+k)kn>>k,因此額外n個左右。k個最大的元素。請注意,你需要找的是數(shù)組排序后的第k示例輸入輸入3,2,1,5,6,4]k輸出輸入輸入3,2,3,1,2,4,5,5,6]k輸出說明k總是有效的,且1kK大的元素,首先能想到的,當(dāng)然就是直接排序。只要數(shù)組是有序的,K個元素就可以了。publicpublicclass ementpublicintfindKthLargest(int[]nums,intk){returnnums[nums.length-}}我們知道,javaArrays.sort()O(nlogn)。對子數(shù)組進(jìn)行劃分,如果劃分得到的位置q正好就是我們需要的下標(biāo),就直接返回a[q];否則,如果q比目標(biāo)下標(biāo)小,就遞歸右子區(qū)間,否則遞歸左子區(qū)間。這樣就可以把原來遞隨機化來加速這個過程,它的時間代價的期望是O(n)。publicpublicintfindKthLargest(int[]nums,intk)returnquickSelect(nums,0,nums.length-1,nums.length-k}publicintquickSelect(int[]nums,intstart,intend,intindexintq=randomPatition(nums,start,endif(q==return}elsereturnq>index?quickSelect(nums,start,q-1,index):quickSelect(nums,q+1,end,index);}}publicintrandomPatition(int[]nums,intstart,intend){Randomrandom=newRandom();intrandIndex=start+random.nextInt(end-start+1);swap(nums,start,randIndex);returnpartition(nums,start,}publicintpartition(int[]nums,intstart,intendintpivot=intleft=intright=while(left<rightwhile(left<right&&nums[right]>=pivot)right--;nums[left]=while(left<right&&nums[left]<=pivot)left++;nums[right]=}nums[left]=return}publicvoidswap(int[]nums,inti,intjinttemp=nums[i];nums[i]=nums[j];nums[j]=temp;}空間復(fù)雜度:O(logn)O(logn)。k?1次刪除操作后堆頂元素就是我們要找的答案。publicpublicintfindKthLargest(int[]nums,intk)intn=intheapSize=buildMaxHeap(nums,heapSizefor(inti=n-1;i>n-k;i--){swap(nums,0,i);heapSize--maxHeapify(nums,0,heapSize}return}publicvoidbuildMaxHeap(int[]nums,intheapSizefor(inti=heapSize/2-1;i>=0;i--){maxHeapify(nums,i,heapSize);}}publicvoidmaxHeapify(int[]nums,inttop,intheapSizeintleft=top*2+1;intright=top*2+2;intlargest=top;ifif(right<heapSize&&nums[right]>nums[largest]){largest=right;}if(left<heapSize&&nums[left]>nums[largest]){largest=left;}if(largest!=topswap(nums,top,largest);maxHeapify(nums,largest,heapSize);}}時間復(fù)雜度:O(nlogn)O(n),k-1O(klogn),因為k<n,故漸進(jìn)時間復(fù)雜為O(n+klogn)=O(nlogn)。n個元素的數(shù)組,原地對它們進(jìn)行排序,使得01和2示例1:輸入:輸入:nums示例輸入:輸入:nums示例輸入:輸入:nums示例輸入:輸入:numsn==1<=n<=nums[i]0、1EdsgerWDijkstra首先提出。3種顏色的條紋拼接而成,如下圖所示:JavapublicpublicvoidsortColors(int[]nums){}如果用選擇排序的思路,我們可以通過遍歷數(shù)組,找到當(dāng)前最?。ɑ蜃畲蟮臄?shù)012publicpublicvoidsortColors(int[]nums)intcurr=for(inti=0;i<nums.length;ifif(nums[i]==0swap(nums,curr++,i}}for(inti=0;i<nums.length;if(nums[i]==1swap(nums,curr++,i}}}publicvoidswap(int[]nums,inti,intjinttemp=nums[i];nums[i]=nums[j];nums[j]=temp;}時間復(fù)雜度:O(n),nnums的長度。需要遍歷兩次數(shù)組。0,1,2publicpublicclassSortColorspublicvoidsortColors(int[]nums)intcount0=0,count1=0,count2=for(intnum:numsif(num==0)count0++;elseif(num==1)count1++;count2}for(inti=0;i<nums.length;i++if(i<count0)nums[i]=0;elseif(i<count0+count1)nums[i]=1;nums[i]=}}}時間復(fù)雜度:O(n),nnums的長度。需要遍歷兩次數(shù)組。02移到數(shù)組尾,1保持不變就可publicpublicvoidsortColors(int[]nums)intleft=0,right=nums.length-inti=while(left<right&&i<=rightwhile(i<=right&&nums[i]==2)swap(nums,i,right--);if(nums[i]==0swap(nums,i,left++);}}時間復(fù)雜度:O(n),nnums的長度。雙指針法只虛遍歷一次數(shù)組。給出一個區(qū)間的集合,請合并所有的區(qū)間示例輸入輸入intervals輸出解釋區(qū)間[1,3]輸入輸入intervals輸出解釋區(qū)間[1,4][4,5]intervals[i][0]<=a2a1<=b2。也就是說,如果某個子區(qū)間的左邊界在另一子區(qū)間內(nèi),那么它們可以合很明顯,這樣的算法,時間復(fù)雜度不會低于O(n^2)。有沒有更好的方式呢publicpublicclassMergeIntervalspublicint[][]merge(int[][]intervals){List<int[]>result=newArrayList<int[]>();Arrays.sort(intervals,newComparator<int[]>()publicintcompare(int[]o1,int[]o2)returno1[0]-}for(int[]interval:intervalsintleft=interval[0],right=intlength=if(length==0||left>result.get(length-1)[1]){}elseintmergedLeft=result.get(length-intmergedRight=Math.max(result.get(length-right}}}returnresult.toArray(new}}時間復(fù)雜度:O(nlogn),其中n為區(qū)間的數(shù)量。除去排序的開銷,我們只需要一次線性掃描,所以主要的時間開銷是排序的O(nlogn)??臻g復(fù)雜度:O(logn),其中n為區(qū)間的數(shù)量。O(logn)即為快速排序所需要的空間復(fù)樹是一種非線性n(n>=0)(rootTm-1,每個集合都是一棵樹,稱為根結(jié)點的(subtree。節(jié)點的度:節(jié)點擁有的個(leaf:樹的高度:樹點的最大層數(shù),也叫做樹的深二叉樹(Binary的兩個,一般叫做節(jié)點的左和右。0i2^i個結(jié)點k2^(k+1)1個結(jié)點(k>=-1)(空樹的高度為-(0)數(shù)為m,2n,則m+(Recursion一個簡單示例就是計算階乘:01;nn-npublicstaticintfactorial(intif(n==0)returnreturnfactorial(n-1)*}publicstaticintfact(intacc,intif(n==0)returnreturnfact(acc*n,n-1}return語句之前,這種形式的iipublicstaticvoidprintTreePreOrder(TreeNoderoot){if(root==null)return;System.out.print(root.val+"\t");printTreePreOrder(root.left);printTreePreOrder(root.right}publicstaticvoidprintTreeInOrder(TreeNoderootif(root==null)return;printTreeInOrder(root.left);System.out.print(root.val+"\t");printTreeInOrder(root.right);}publicstaticvoidprintTreePostOrder(TreeNoderootif(root==null)return;printTreePostOrder(root.left);printTreePostOrder(root.right);System.out.print(root.val+"\t");}publicstaticvoidprintTreeLevelOrder(TreeNodepublicstaticvoidprintTreeLevelOrder(TreeNoderoot){Queue<TreeNode>queue=newLinkedList<>();while(!queue.isEmpty()TreeNodecurNode=System.System.out.print(curNode.val+"\t");if(curNode.left!=null)if(curNode.right!=null)}}二叉搜索樹(BinarySearch任意節(jié)點左如果不為空,則左點的值均小于根節(jié)點的任意節(jié)點右如果不為空,則右點的值均大于根節(jié)點的任意節(jié)點的左右,也分別是二叉搜索nO(logN)。n個節(jié)點的線性鏈表。如下圖平衡二叉搜索樹:簡稱平衡二叉樹。由前的數(shù)學(xué)家Adelse-Velskil和Landis在年高度平衡的二叉樹,根據(jù)科學(xué)家的英文名也稱為AVL樹。1棵樹的左右的高度之差超過1,如左的樹高為2,右的樹高為0,樹高差2就打破了這個平衡。AVL樹是帶有平衡條件的二叉搜索樹,它是嚴(yán)格的平衡二叉樹,平衡條件必須滿足(所有節(jié)點的左右高度差不超過1)。不管我們是執(zhí)行插入還是刪除操作,只要不滿足上面AVLWindows進(jìn)程地址空間樹(Red-Black。樹是一種特殊的二叉查找樹樹的每個節(jié)點上都有位表示節(jié)點的顏色,可。每個葉子節(jié)點都是黑色的空節(jié)點(NIL節(jié)點旋轉(zhuǎn)等操作,讓新的樹符合所有規(guī)則。AVL樹多用于搜索,插入,刪除操作多的情況下。樹應(yīng)用比較廣泛C++STL中,mapset都是用紅黑樹實現(xiàn)的。Java中的TreeSet,TreeMap也都是用樹實現(xiàn)的。著名的linux進(jìn)程調(diào)度 yFairScheduler,用樹管理進(jìn)程控制塊epoll在內(nèi)核中的實現(xiàn),用樹管理nginx中,用樹管理timerB樹(B-B樹(B-Tree)是一種自平衡的樹,它是一種多路搜索樹(并不是二叉的,能夠保證數(shù)據(jù)有序。同時,BO(logn),為大塊數(shù)據(jù)的讀寫操作做了優(yōu)化,同時它也可以用來描述外部。M根結(jié)點的兒子數(shù)為[2,除根結(jié)點以外的非葉子結(jié)點的兒子數(shù)為[M/2M/2-1(取上整)M-12非葉子結(jié)點的關(guān)鍵字個數(shù)=指向兒子的指針個數(shù)–非葉子結(jié)點的關(guān)鍵字:K[1],K[2],K[M-1]K[i]指向關(guān)鍵字大于K[M-1]的,其它P[i]指向關(guān)鍵字屬于(K[i-1],K[i])的M3BB+B+B-B+樹只有達(dá)到葉子結(jié)點才命中(B-樹可以在非葉非葉子結(jié)點相當(dāng)于是葉子結(jié)點的索引(稀疏索引,葉子結(jié)點相當(dāng)于是(關(guān)鍵B+層級更低,IOB+樹的主要原因。4 // / 3 4 72/ / 6 publicpublicTreeNodeinvertTree(TreeNoderoot){if(root==null)returnnull;TreeNodetemp=root.left;root.left=root.right;root.right=temp;invertTree(root.left);invertTree(root.rightreturnreturn}的高度。在平均情況下,二叉樹的高度與節(jié)點個數(shù)為對數(shù)關(guān)系,即O(logN)。而在情況O(N)。publicpublicTreeNodeinvertTree(TreeNoderoot){if(root==null)returnnull;TreeNodeleft=invertTree(root.left);TreeNoderight=invertTree(root.right);root.left=right;root.right=return}一個二叉樹每個節(jié)點的左右兩個的高度差的絕對值不超過1示例輸入:root輸入:root示例輸入:root輸入:root示例輸入:輸入:root(類似先序遍歷)或者自底向上(類似后序遍歷容易想到的一個方法是,從根節(jié)點開始,自頂向下遞歸地判斷左右是否平衡publicpublicclassBalancedBinaryTreepublicbooleanisBalanced(TreeNoderoot)if(root==null)returnreturnMath.abs(height(root.left)-height(root.right))<=1&&isBalanced(root.left)&&
溫馨提示
- 1. 本站所有資源如無特殊說明,都需要本地電腦安裝OFFICE2007和PDF閱讀器。圖紙軟件為CAD,CAXA,PROE,UG,SolidWorks等.壓縮文件請下載最新的WinRAR軟件解壓。
- 2. 本站的文檔不包含任何第三方提供的附件圖紙等,如果需要附件,請聯(lián)系上傳者。文件的所有權(quán)益歸上傳用戶所有。
- 3. 本站RAR壓縮包中若帶圖紙,網(wǎng)頁內(nèi)容里面會有圖紙預(yù)覽,若沒有圖紙預(yù)覽就沒有圖紙。
- 4. 未經(jīng)權(quán)益所有人同意不得將文件中的內(nèi)容挪作商業(yè)或盈利用途。
- 5. 人人文庫網(wǎng)僅提供信息存儲空間,僅對用戶上傳內(nèi)容的表現(xiàn)方式做保護(hù)處理,對用戶上傳分享的文檔內(nèi)容本身不做任何修改或編輯,并不能對任何下載內(nèi)容負(fù)責(zé)。
- 6. 下載文件中如有侵權(quán)或不適當(dāng)內(nèi)容,請與我們聯(lián)系,我們立即糾正。
- 7. 本站不保證下載資源的準(zhǔn)確性、安全性和完整性, 同時也不承擔(dān)用戶因使用這些下載資源對自己和他人造成任何形式的傷害或損失。
最新文檔
- 《廉潔社會共建》課件
- 2025排煙設(shè)備合作合同
- 二零二五年機場保安服務(wù)合同規(guī)范范本3篇
- 2025訂貨合同書范本
- 2025關(guān)于解除勞動合同的協(xié)議書
- 二零二五年高端定制家具買賣合同2篇
- 2025房地產(chǎn)委托代理合同范文
- 2025裝飾工程合同樣
- 農(nóng)業(yè)設(shè)施腳手架安裝合同模板
- 水利設(shè)施安全保障合同范本
- 2024-2025學(xué)年成都高新區(qū)七上數(shù)學(xué)期末考試試卷【含答案】
- 定額〔2025〕1號文-關(guān)于發(fā)布2018版電力建設(shè)工程概預(yù)算定額2024年度價格水平調(diào)整的通知
- 2025年浙江杭州市西湖區(qū)專職社區(qū)招聘85人歷年高頻重點提升(共500題)附帶答案詳解
- 《數(shù)學(xué)廣角-優(yōu)化》說課稿-2024-2025學(xué)年四年級上冊數(shù)學(xué)人教版
- “懂你”(原題+解題+范文+話題+技巧+閱讀類素材)-2025年中考語文一輪復(fù)習(xí)之寫作
- 2025年景觀照明項目可行性分析報告
- 2025年江蘇南京地鐵集團(tuán)招聘筆試參考題庫含答案解析
- 2025年度愛讀書學(xué)長參與的讀書項目投資合同
- 電力系統(tǒng)分析答案(吳俊勇)(已修訂)
- 一種基于STM32的智能門鎖系統(tǒng)的設(shè)計-畢業(yè)論文
- 華為經(jīng)營管理-華為經(jīng)營管理華為的IPD(6版)
評論
0/150
提交評論