算法主體部分javascript
var OnlineUser = { //list : 待查找的數組 //key : 待插入的值 //order : 數組的順序 1:從小到大 0:從大到小 //start : 開始查找的起始下標位置 //end : 開始查找的結束下標位置 //例: //var arr1 = [17,15,15,14,14,13,13,8,8,7,7,6,5,4,3]; //var key = 8; //var index = OnlineUser.GetPosIndex(arr1, key,0, arr1.length, 0); GetPosIndex:function(list,key,order,start,end) { var index = -1; var halfIndex = Math.abs(parseInt((end+start)/2)); if(list[halfIndex] == key) //在中位 { index = halfIndex; } else if(list[halfIndex] < key && key < list[start])//在左半 { index = OnlineUser.GetPosIndex(list, key, order, start, halfIndex); } else if(list[end] < key && key < list[halfIndex])//在右半 { halfIndex ++ ; index = OnlineUser.GetPosIndex(list, key, order, halfIndex, end); } else if(list[start] <= key && list[halfIndex] < key) //在左邊 { index = start; } else if(list[end] >= key && list[halfIndex] > key) //在右邊 { index = end + 1; } else //一個值都沒有 { index = 0; } return index; }, Sort:function(array){ return array.sort(function(a, b){ return b - a; }); }, GetInsIndex:function(list,key) { var index = 0; //list = this.Sort(list); for(var k=0;k<list.length;k++) { if(list[k] <= key) { index = k; break; } else if(list[k] > key) index = k+1; else if(list[k] < key) break; } return index; } }
完整的程序和測試頁面html
<html> <head> <meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> </head> <body> <div id ="result"></div> </body> <script lang="javascript"> var OnlineUser = { //list : 待查找的數組 //key : 待插入的值 //order : 數組的順序 1:從小到大 0:從大到小 //start : 開始查找的起始下標位置 //end : 開始查找的結束下標位置 //例: //var arr1 = [17,15,15,14,14,13,13,8,8,7,7,6,5,4,3]; //var key = 8; //var index = OnlineUser.GetPosIndex(arr1, key,0, arr1.length, 0); GetPosIndex:function(list,key,order,start,end) { var index = -1; var halfIndex = Math.abs(parseInt((end+start)/2)); if(list[halfIndex] == key) //在中位 { index = halfIndex; } else if(list[halfIndex] < key && key < list[start])//在左半 { index = OnlineUser.GetPosIndex(list, key, order, start, halfIndex); } else if(list[end] < key && key < list[halfIndex])//在右半 { halfIndex ++ ; index = OnlineUser.GetPosIndex(list, key, order, halfIndex, end); } else if(list[start] <= key && list[halfIndex] < key) //在左邊 { index = start; } else if(list[end] >= key && list[halfIndex] > key) //在右邊 { index = end + 1; } else //一個值都沒有 { index = 0; } return index; }, Sort:function(array){ return array.sort(function(a, b){ return b - a; }); }, GetInsIndex:function(list,key) { var index = 0; //list = this.Sort(list); for(var k=0;k<list.length;k++) { if(list[k] <= key) { index = k; break; } else if(list[k] > key) index = k+1; else if(list[k] < key) break; } return index; } } var result = document.getElementById("result"); //測試用例,子數組第一個爲從小到大順序,第二個從大到小的順序 var testArray = [ [[],[]], [[1],[1]], [[1,2,3,4,5],[17,15,14,13]], [[11,12,13,14,15],[7,6,5,4,3,2,1]], [[1,2,3,4,5,7,7,7,8,8,9,9,10],[17,15,15,14,14,13,13,8,8,7,7,6,5,4,3]], [[1000003,1000003,1000003,1000003,1000003,11000269],[11000269,1000003,1000003,1000003,1000003,1000003]] ]; //指望值 var expectArray = [ [[0],[0]], [[1],[0]], [[5],[4]], [[0],[0]], [[10],[7]], [[5],[1]] ]; //各用例要查找的值 var testKey = [ [8], [8], [8], [8], [8], [1000003] ]; var start =0;//開始查找的起始下標位置 var end =0; //開始查找的結束下標位置 var index = 0; //待插入的位置 for(var i=0;i<testArray.length;i++) //組數 { result.innerHTML += "<hr />"; for(var j=0;j<testArray[i].length;j++) //組單元 { if(j==1) //大到小 { start = 0; end = testArray[i][j].length - 1; index = OnlineUser.GetPosIndex(testArray[i][j], testKey[i][0],1, start, end); result.innerHTML += "<br />small -> big: ["+testArray[i][j].toString()+"], key:"+testKey[i]+", index:" + index +", expect:"+expectArray[i][j]+ ((index == expectArray[i][j])?"":",<font color='red'>NotEq</font>"); } else //從小到大 { continue; start = testArray[i][j].length - 1; end = 0; index = OnlineUser.GetPosIndex(testArray[i][j], testKey[i][0],0, start, end); result.innerHTML += "<br />big <- small: ["+testArray[i][j].toString()+"], key:"+testKey[i]+", index:" + index+", expect:"+expectArray[i][j]+ ((index == expectArray[i][j])?"":",<font color='red'>NotEq</font>"); } } } /* */ //構造測試數據 var bufferArray = []; var newArray = [10002107,10002107,10000003,10002057,10000037,10000037,10000007,10000007,10000007,10000007,10000007,10000007,10000007,10000007,10002067,10000005,10000005,10000005,10000007,10000007,10000005,10000005,10000005,10000005,10000005,10000005,10000005,10000005,10000005,10000005,10000005,10000005]; newArray.reverse(); for(var n = 0;n<10000;n++) { newArray[n] = parseInt(10000000 + Math.random() * 1000000); } var key = parseInt(10000000 + Math.random() * 1000000); //IE彈出"是否中止運行此腳本"腳本超時時間設置 //1. 打開註冊表HKEY_CURRENT_USER\Software\Microsoft\InternetExplorer\Styles,若是 Styles 鍵不存在,建立調用 Styles 的一個新的項 //2. 建立新的 DWORD 值在此項下稱爲"MaxScriptStatements"並將值設置爲所需的腳本語句數,如100000000 // ***************** 如下數據準確性測試 ********************* //系統法排序 var bArray = OnlineUser.Sort(newArray); //二分法插入 for(var m=0; m < newArray.length;m++) { var index = OnlineUser.GetPosIndex(bufferArray, newArray[m], 1,0, bufferArray.length-1); //var index = OnlineUser.GetPosIndex(bufferArray, newArray[m], 0, bufferArray.length,0); bufferArray.splice(index, 0, newArray[m]); } //結果打印 var a = bufferArray.toString(); var b = bArray .toString(); result.innerHTML += "<br />Rnd:" + newArray.toString(); result.innerHTML += "<br />Dic:" + a; result.innerHTML += "<br />Sys:" + b; result.innerHTML += "<br /> Eq:" + (a == b); // ***************** 如下性能測試 ********************* //將數據分爲5段,作採樣 var keyArr = []; for(var ki=0;ki<5;ki++) { keyArr.push(newArray.length * ki * 0.2); } var cnt = 1000; var idx =0 ; var dateStart,dateEnd,timeSpan; //遍歷方法插入 result.innerHTML += "<br />"+keyArr.toString(); result.innerHTML += "<br />GetInsIndex: ckey key idx cnt t(ms)"; for(var ckey=0;ckey<keyArr.length;ckey++) { dateStart = new Date(); //@@@TEST for(var x=0;x<cnt;x++ ) { idx = OnlineUser.GetInsIndex(bArray, bArray[keyArr[ckey]]+1); } dateEnd = new Date();t = dateEnd.getTime() - dateStart.getTime(); result.innerHTML +="<br />"+ckey+" "+keyArr[ckey] + " "+idx+" "+cnt+" "+t; } cnt = 10000; //二分法插入 result.innerHTML += "<br />GetPosIndex: ckey key idx cnt t(ms)"; for(var ckey=0;ckey<keyArr.length;ckey++) { dateStart = new Date(); //@@@TEST for(var x=0;x<cnt;x++ ) { idx = OnlineUser.GetPosIndex(bArray, bArray[keyArr[ckey]]+1,1,0,bArray.length -1 ); } dateEnd = new Date();t = dateEnd.getTime() - dateStart.getTime(); result.innerHTML +="<br />"+ckey+" "+keyArr[ckey] + " "+idx+" "+cnt+" "+t; } </script> </html>