摘要:冒泡排序原理冒泡排序的過程就是將數(shù)組中相鄰的兩個(gè)元素進(jìn)行比較,如果前面的元素比后面的元素要大交換位置,否則位置不變舉個(gè)栗子有數(shù)組第一輪循環(huán)和比較,小于兩者位置不變,接下來和比較,大于,兩者交換位置,接著和比較,兩者交換位置,繼續(xù)和比較兩者交
1.冒泡排序
原理:冒泡排序的過程就是將數(shù)組中相鄰的兩個(gè)元素進(jìn)行比較,如果前面的元素比后面的元素要大交換位置,否則位置不變;舉個(gè)栗子:有數(shù)組 arr = [3,5,4,2,1];
第一輪循環(huán):3和5比較,3小于5兩者位置不變,接下來5和4比較,5大于4,兩者交換位置,接著5和2比較,5>2兩者交換位置,繼續(xù)5和1 比較 5>1兩者交換位置,一輪后得到的數(shù)組是[3,4,2,1,5];把大的元素放到數(shù)組的最末尾,這種就像水泡樣一層一層的像后移動(dòng),就是冒泡排序了;
代碼實(shí)現(xiàn):
// 記錄循環(huán)次數(shù) let count = 0 // 位置交換函數(shù) const change = function (arr, n1, n2) { // 用es6的實(shí)現(xiàn)交換 [arr[n1], arr[n2]] = [arr[n2], arr[n1]] //let temp = arr[n1] //arr[n1] = arr[n2] //arr[n2] = temp } // 冒泡排序 const bubbleSort = function (soucre) { let len = soucre.length for (let i = 0;i < len - 1; i++) { for (let j = 0; j < len - 1 - i;j++) { count ++ if (soucre[j] > soucre[j+1]) { change(soucre, j, j+1) } } } return soucre } //驗(yàn)證 console.log(bubbleSort([3,6,2,4,9,1,8])) // [1,2,3,4,6,8,9] console.log(count) // 212.選擇排序
選擇排序和冒泡排序類似也是依次對相鄰的數(shù)進(jìn)行兩兩比較。不同之處在于,他不是每次兩兩相鄰比較后交換位置,他是先找出最大(最?。⑺诺秸_的位置,然后再尋找次最大(最?。┓旁谡_的位置;
舉個(gè)栗子:有數(shù)組 arr = [3,5,4,2,1];
先假設(shè)第一個(gè)元素是最小值,并定義一個(gè)minidx=0變量記錄最?。ㄗ畲螅┲档奈恢?,for循環(huán)和其他元素進(jìn)行比較,3和5進(jìn)行比較,5>3此時(shí)不做處理,4也是一樣處理,當(dāng)3和2比較時(shí),3>2,此時(shí)將minidx賦值為2的位置,接下來用arr[minidx]和剩余的元素比較遇到比他小的就用minidx記錄小值的位置;最后將最小的位置值和初始給的值位置進(jìn)行互換(當(dāng)然是初始的值和一輪循環(huán)下來的minidx位置不一樣才互換);所以一輪循環(huán)下來結(jié)果是arr = [1,5,4,2,3]
代碼實(shí)現(xiàn):
// 記錄循環(huán)次數(shù) let count = 0 // 選擇排序 const selectSort = function (soucre) { let len = soucre.length let minidx; for (let i = 0; i < len; i ++) { minidx = i for (let j = i + 1; j < len; j++) { count ++ if (soucre[minidx] > soucre[j]) { minidx = j } } if (minidx !== i) { change(soucre,i,minidx) } } return soucre } console.log(selectSort([3,6,2,4,9,1,8,23,45,16,14])) // [1, 2, 3, 4, 6, 8, 9, 14, 16, 23, 45] console.log(count) // 553.插入排序
原理:將數(shù)組分為已排序和未排序,將第一個(gè)元素看作是已排序的元素,而其他是未排序的,從未排序的里面取出一元素和已排序元素進(jìn)行比較,并插入到正確位置,這樣已排序部分增加一個(gè)元素,而未排序的部分減少一個(gè)元素。直到排序完成
舉個(gè)栗子:有數(shù)組 arr = [1,5,4,2,3],第一次假設(shè)元素1 是已排序部分,5,4,2,3為未排序,取出元素5加入已排序部分,5>1,已排序部分為1,5;而未排序部分為4,2,3;如此往復(fù)完成排序;
代碼實(shí)現(xiàn):
const insertSort = function (source) { let len = source.length let value let j let i for (i = 0; i < len; i++) { value = source[i] // 已排序部分進(jìn)行元素的右移一位,并把目標(biāo)值value插入到對應(yīng)的位置 for (j = i -1 ;j > -1 && source[j] > value; j--) { source[j+1] = source[j] } source[j+1] = value } return source } console.log(insertSort([3,6,2,4,9,1,8])) // [1,2,3,4,6,8,9]4.歸并排序
原理: 將兩個(gè)已經(jīng)排序的數(shù)組合并,要比從頭開始排序所有元素來得快。因此,可以將數(shù)組拆開,分成n個(gè)只有一個(gè)元素的數(shù)組,然后不斷地兩兩合并,直到全部排序完成
代碼實(shí)現(xiàn):
const mergeSort = function mergeSort(source) { let len = source.length if (len < 2) { return source } let mid = Math.floor(len/2) let left = source.slice(0,mid) let right = source.slice(mid) return merge(mergeSort(left), mergeSort(right)) } function merge(left, right) { let result = [] while (left.length && right.length) { if (left[0] <= right[0]) { result.push(left.shift()) } else { result.push(right.shift()) } } while (left.length){ result.push(left.shift()) } while (right.length){ result.push(right.shift()) } return result } console.log(mergeSort([4,8,1,3,5,9,6])) // [1,3,4,5,6,8,9]5.快速排序
原理:快速排序是目前公認(rèn)的速度快和高效的排序方式,時(shí)間復(fù)雜度O(nlogn)是比較理想的狀態(tài),他的實(shí)現(xiàn)過程是,先在數(shù)組找到一個(gè)基點(diǎn),把大于這個(gè)基點(diǎn)的值放到右側(cè),小于基點(diǎn)的值放到左側(cè),再將右側(cè)的和左側(cè)的也按照這種方式再次分配,直到完成排序
舉個(gè)栗子:有一個(gè)數(shù)組 arr = [1,5,4,2,3];假設(shè)我們找數(shù)組的中間點(diǎn)作為基點(diǎn)也就是4,那一輪循環(huán)后結(jié)果就是[1,2,3,4,5] ->_->怎么這么巧,一輪就OK,果然是快速排序,就是快 哈哈,當(dāng)然程序不會(huì)這么做,他是嚴(yán)謹(jǐn)?shù)?,他還會(huì)去拆分[1,2,3]只是這個(gè)實(shí)際上已經(jīng)是排好了的;
代碼實(shí)現(xiàn):粗糙一點(diǎn)的
const quire = function quire(source) { if(source.length < 2) return source let left = [] let right = [] let len = source.length let key = source[Math.floor(len/2 -1)] for (let i = 0;ikey){ right.push(source[i]) } } return [].concat(quire(left),key,quire(right)) }
上面這種方法缺點(diǎn)就是空間浪費(fèi),他會(huì)創(chuàng)建很多個(gè)left 和 right 這樣的數(shù)組,造成空間的浪費(fèi),當(dāng)數(shù)據(jù)量一大的話還是很恐怖的,所以我們要改進(jìn)的就是,不新建中間數(shù)組,而是直接修改位移目標(biāo)數(shù)組;
改進(jìn)原理: 選取一個(gè)基點(diǎn),從數(shù)組的兩頭兩個(gè)指針分別向基點(diǎn)位移,位移的原則是,基點(diǎn)的左邊的元素如果小于基點(diǎn),那就像基點(diǎn)位置靠攏一位,i++,如果大于基點(diǎn)就原地不動(dòng),基點(diǎn)右邊的元素反過來,如果大于基點(diǎn)就像基點(diǎn)靠攏一位,j--;如果小于就原地不動(dòng);這時(shí)再比較兩個(gè)原地不動(dòng)的點(diǎn),如果右邊的不動(dòng)點(diǎn)小于左邊的值,就互換他們的位置;
代碼實(shí)現(xiàn):
// 位置交換 const change = function (arr, n1, n2) { // let temp = arr[n1] // arr[n1] = arr[n2] // arr[n2] = temp // 用es6的實(shí)現(xiàn)交換 [arr[n1], arr[n2]] = [arr[n2], arr[n1]] } const quiregai = function quiregai(source, start, end) { let pivot = source[Math.floor((start + end)/2)] let i = start // 左邊指針初始位置 let j = end // 右邊指針初始位置 while(i<=j) { while (source[i] < pivot) { i ++ // 左指針右移 } while (source[j] > pivot) { j -- // 右指針左移 } if (i <= j){ change(source,i,j) // 交換兩個(gè)位置的值 i++ j-- } } return i // 返回一輪循環(huán)后左指針的位置,為下一輪循環(huán)初始位置確定 } const quiregaiSort = function quiregaiSort(source, start, end) { if (source.length < 2) return source var start = start || 0 var end = end || source.length - 1 var nextStart = quiregai(source, start, end) // debugger if (start < nextStart -1) { quiregaiSort(source, start, nextStart -1 ) // 上個(gè)循環(huán)結(jié)束的左指針作為左邊區(qū)塊循環(huán)的右指針 } if (nextStart < end) { quiregaiSort(source, nextStart, end) // 上個(gè)循環(huán)結(jié)束的左指針作為右邊區(qū)塊循環(huán)的左指針 } return source } console.log(quiregaiSort([4,1,9,3,7,5,76,21,12,53,24]))
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/95757.html
摘要:探討判斷橫豎屏的最佳實(shí)現(xiàn)前端掘金在移動(dòng)端,判斷橫豎屏的場景并不少見,比如根據(jù)橫豎屏以不同的樣式來適配,抑或是提醒用戶切換為豎屏以保持良好的用戶體驗(yàn)。 探討判斷橫豎屏的最佳實(shí)現(xiàn) - 前端 - 掘金在移動(dòng)端,判斷橫豎屏的場景并不少見,比如根據(jù)橫豎屏以不同的樣式來適配,抑或是提醒用戶切換為豎屏以保持良好的用戶體驗(yàn)。 判斷橫豎屏的實(shí)現(xiàn)方法多種多樣,本文就此來探討下目前有哪些實(shí)現(xiàn)方法以及其中的優(yōu)...
摘要:常見的內(nèi)部排序算法有插入排序希爾排序選擇排序冒泡排序歸并排序快速排序堆排序基數(shù)排序等。這里主要介紹快速排序??焖倥判蚴且环N既不浪費(fèi)空間又可以快一點(diǎn)的排序算法。 常見的內(nèi)部排序算法有:插入排序、希爾排序、選擇排序、冒泡排序、歸并排序、快速排序、堆排序、基數(shù)排序等。這里主要介紹快速排序。 一圖勝千言: showImg(https://segmentfault.com/img/bVNIpc?...
摘要:在上家公司裸辭之后,經(jīng)過一段時(shí)間休整,月中下旬面試了一些公司,由于本人框架使用的是,所以面試題涉及到框架的都是,現(xiàn)將面試題整理一下列舉常用的特性。事件冒泡以及事件捕獲。其他前端分頁和后端分頁優(yōu)缺點(diǎn)。包含多個(gè)子節(jié)點(diǎn)及孫節(jié)點(diǎn),遍歷。 在上家公司裸辭之后,經(jīng)過一段時(shí)間休整,5月中下旬面試了一些公司,由于本人框架使用的是vue,所以面試題涉及到框架的都是vue,現(xiàn)將面試題整理一下: es6 ...
摘要:在上家公司裸辭之后,經(jīng)過一段時(shí)間休整,月中下旬面試了一些公司,由于本人框架使用的是,所以面試題涉及到框架的都是,現(xiàn)將面試題整理一下列舉常用的特性。事件冒泡以及事件捕獲。其他前端分頁和后端分頁優(yōu)缺點(diǎn)。包含多個(gè)子節(jié)點(diǎn)及孫節(jié)點(diǎn),遍歷。 在上家公司裸辭之后,經(jīng)過一段時(shí)間休整,5月中下旬面試了一些公司,由于本人框架使用的是vue,所以面試題涉及到框架的都是vue,現(xiàn)將面試題整理一下: es6 ...
摘要:場景當(dāng)數(shù)據(jù)兩足夠大的時(shí)候,一頁展示不完的時(shí)候,我們經(jīng)常會(huì)需要分頁的功能。方案三,數(shù)據(jù)比較大,排序需要排序當(dāng)數(shù)據(jù)量比較大的時(shí)候,并且需要排序的時(shí)候,可以使用這種情況。 場景 當(dāng)數(shù)據(jù)兩足夠大的時(shí)候,一頁展示不完的時(shí)候,我們經(jīng)常會(huì)需要分頁的功能。 方案 方案一,數(shù)據(jù)不是很大 需要排序 s := globalS.Copy() c := s.DB(db).C(collection...
閱讀 2978·2023-04-26 02:04
閱讀 1286·2021-11-04 16:07
閱讀 3712·2021-09-22 15:09
閱讀 685·2019-08-30 15:54
閱讀 1906·2019-08-29 14:11
閱讀 2534·2019-08-26 12:19
閱讀 2261·2019-08-26 12:00
閱讀 764·2019-08-26 10:27