查找定义:根据给定的某个值,在查找表中确定一个其关键字等于给定值的数据元素(或记录)。
查找算法分类:
1)静态查找和动态查找;
注:静态或者动态都是针对查找表而言的。动态表指查找表中有删除和插入操作的表。
2)无序查找和有序查找。
无序查找:被查找数列有序无序均可;
有序查找:被查找数列必须为有序数列。
平均查找长度(Average Search Length,ASL):需和指定key进行比较的关键字的个数的期望值,称为查找算法在查找成功时的平均查找长度。
对于含有n个数据元素的查找表,查找成功的平均查找长度为:ASL = Pi*Ci的和。
Pi:查找表中第i个数据元素的概率。
Ci:找到第i个数据元素时已经比较过的次数。
顺序查找
说明:顺序查找适合于存储结构为顺序存储或链接存储的线性表。
基本思想:顺序查找也称为线形查找,属于无序查找算法。从数据结构线形表的一端开始,顺序扫描,依次将扫描到的结点关键字与给定值k相比较,若相等则表示查找成功;若扫描结束仍没有找到关键字等于k的结点,表示查找失败。
复杂度分析:
查找成功时的平均查找长度为:(假设每个数据元素的概率相等) ASL = 1/n(1+2+3+…+n) = (n+1)/2 ;
当查找不成功时,需要n+1次比较,时间复杂度为O(n);
所以,顺序查找的时间复杂度为O(n)。
<script>
function SequenceSearch(arr, value) {
for (let i = 0; i < arr.length; i++) {
if (arr[i] == value) {
return i;
}
}
return -1;
}
var arr = [1,3,4,5,2,4,2]
console.log(SequenceSearch(arr,2))
</script>
二分查找
说明:元素必须是有序的,如果是无序的则要先进行排序操作。
基本思想:也称为是折半查找,属于有序查找算法。用给定值k先与中间结点的关键字比较,中间结点把线形表分成两个子表,若相等则查找成功;若不相等,再根据k与该中间结点关键字的比较结果确定下一步查找哪个子表,这样递归进行,直到查找到或查找结束发现表中没有这样的结点。
复杂度分析:最坏情况下,关键词比较次数为log2(n+1),且期望时间复杂度为O(log2n);
注:折半查找的前提条件是需要有序表顺序存储,对于静态查找表,一次排序后不再变化,折半查找能得到不错的效率。但对于需要频繁执行插入或删除操作的数据集来说,维护有序的排序会带来不小的工作量,那就不建议使用。——《大话数据结构》
<script>
// 递归
function binarySearch(data, dest, start, end) {
if (start > end) { // 新增否则找不到进入死循环了
return false;
}
var end = end || data.length - 1;
var start = start || 0;
var mid = Math.floor((start + end) / 2);
//var mid = parseInt(start+(end-start)/2);
//直接命中
if (data[mid] == dest) {
return mid;
}
if (data[mid] > dest) { // 放左
end = mid - 1;
return binarySearch(data, dest, start, end);
} else { // 放右
start = mid + 1;
return binarySearch(data, dest, start, end);
}
return false;
</script>
<script>
// 非递归 用while
//代码中的判断条件必须是while (left <= right),
//否则的话判断条件不完整,比如:array[3] = {1, 3, 5};
//待查找的键为5,此时在(low < high)条件下就会找不到,因为low和high相等时,指向元素5,但是此时条件不成立,没有进入while()中
function binarySearch2(data, dest) {
var end = data.length - 1;
var start = 0;
while (start <= end) {
var m = Math.floor((end + 1) / 2);
if (data[m] == dest) {
return m;
}
if (data[m] > dest) {
end = m - 1;
} else {
start = m + 1;
}
}
return falsex
</script>
插值查找
基本思想:基于二分查找算法,将查找点的选择改进为自适应选择,可以提高查找效率。当然,差值查找也属于有序查找。
mid=low+1/2*(high-low);将查找的点改进为如下:
mid=low+(key-a[low])/(a[high]-a[low])*(high-low),
注:对于表长较大,而关键字分布又比较均匀的查找表来说,插值查找算法的平均性能比折半查找要好的多。反之,数组中如果分布非常不均匀,那么插值查找未必是很合适的选择。
复杂度分析:查找成功或者失败的时间复杂度均为O(log2(log2n))。
<script>
function InsertionSearch(arr, val, start, end) {
var end = end || data.length - 1;
var start = start || 0;
var mid = start + (val - arr[low]) / (arr[end] - arr[start]) * (end - start);
if (arr[mid] == val) {
return mid;
}
if (arr[mid] > val) {
return InsertionSearch(arr, val, start, mid - 1);
}
else {
return InsertionSearch(arr, val, mid + 1, end);
}
}
</script>
斐波那契查找
斐波那契数列:1, 1, 2, 3, 5, 8, 13, 21, 34, 55, 89…….(从第三个数开始,后边每一个数都是前两个数的和)。斐波那契查找也属于一种有序查找算法。
斐波那契数组的实现:
<script>
function getNum1(index) {
if (index == 1 || index == 2) {
return 1;
} else {
return getNum(index - 1) + getNum(index - 2);
}
}
function getNum2(index) {
if (index == 1 || index == 2) {
return 1;
} else {
var one = 1;
var two = 1;
for (var i = 3; i <= index; i++) {
if (i == 3) {
one = 1;
two = 1;
}
else {
var temp = one;
one = two;
two = temp + two;
}
}
return one + two
}
}
function getNum3(index) {
var F = [];
F[0] = 0;
F[1] = 1;
for (var i = 2; i < index - 1; i++) {
F[i] = F[i - 1] + F[i - 2];
}
return F[index];
}
</script>
基本思路:
相对于折半查找,一般将待比较的key值与第mid=(low+high)/2位置的元素比较,比较结果分三种情况:
1)相等,mid位置的元素即为所求
2)>,low=mid+1;
3)<,high=mid-1。
斐波那契查找与折半查找很相似,他是根据斐波那契序列的特点对有序表进行分割的。他要求开始表中记录的个数为某个斐波那契数小1,及n=F(k)-1;
开始将k值与第F(k-1)位置的记录进行比较(及mid=low+F(k-1)-1),比较结果也分为三种
1)相等,mid位置的元素即为所求
2)>,low=mid+1,k-=2;
说明:low=mid+1说明待查找的元素在[mid+1,high]范围内,k-=2 说明范围[mid+1,high]内的元素个数为n-(F(k-1))= Fk-1-F(k-1)=Fk-F(k-1)-1=F(k-2)-1个,所以可以递归的应用斐波那契查找。
3)<,high=mid-1,k-=1。
说明:low=mid+1说明待查找的元素在[low,mid-1]范围内,k-=1 说明范围[low,mid-1]内的元素个数为F(k-1)-1个,所以可以递归 的应用斐波那契查找。
<script>
function search(array, value) {
let low = 0, high = array.length - 1, n = array.length - 1;
let mid, k = 0;
//构建一个长度大于array数组的斐波那契数组
var F = [];
F[0] = 0;
F[1] = 1;
for (var i = 2; i < high + 5; i++) {
F[i] = F[i - 1] + F[i - 2];
}
while (high > F[k] - 1) { //寻找第k项
k++;
}
for (let i = high; i < F[k] - 1; i++) { //补全有序数组
array[i] = array[high];
}
while (low <= high) {
mid = low + F[k - 1] - 1;
if (array[mid] > value) {
high = mid - 1;
k = k - 1; //长度缩减为F[k-1]-1
} else if (array[mid] < value) {
low = mid + 1;
k = k - 2; //长度缩减为F[k-2]-1
} else {
if (m <= n) //相等则找到位置
return mid;
else {
return n; //大于原始长度,则说明等于数组最后一项
}
}
return -1;
}
}
</script>
树表查找
最简单的树表查找算法——二叉树查找算法。
基本思想:二叉查找树是先对待查找的数据进行生成树,确保树的左分支的值小于右分支的值,然后在就行和每个节点的父节点比较大小,查找最适合的范围。 这个算法的查找效率很高,但是如果使用这种查找方法要首先创建树。
二叉查找树(BinarySearch Tree,也叫二叉搜索树,或称二叉排序树Binary Sort Tree)或者是一棵空树,或者是具有下列性质的二叉树:
1)若任意节点的左子树不空,则左子树上所有结点的值均小于它的根结点的值;
2)若任意节点的右子树不空,则右子树上所有结点的值均大于它的根结点的值;
3)任意节点的左、右子树也分别为二叉查找树。
二叉查找树性质:对二叉查找树进行中序遍历,即可得到有序的数列。
索引查找(分块查找)
分块查找又称索引顺序查找,它是顺序查找的一种改进方法。
算法思想:将n个数据元素"按块有序"划分为m块(m ≤ n)。每一块中的结点不必有序,但块与块之间必须"按块有序";即第1块中任一元素的关键字都必须小于第2块中任一元素的关键字;而第2块中任一元素又都必须小于第3块中的任一元素,……
算法流程:
step1 先选取各块中的最大关键字构成一个索引表;
step2 查找分两个部分:先对索引表进行二分查找或顺序查找,以确定待查记录在哪一块中;然后,在已确定的块中用顺序法进行查找。
贪心算法
遵循一种近似解决问题的技术,期盼通过每个阶段的局部最优选择(当前最好的解),从而达到全局的最优(全局最优解)。贪心得到结果是一个可以接受的解,不一定总是得到最优的解。
最少硬币找零问题:最少硬币找零是给出要找零的钱数,以及可以用硬币的额度数量,找出有多少种找零方法。
如:美国面额硬币有:1,5,10,25
我们给36美分的零钱,看能得怎样的结果?
<script>
function MinCoinChange(coins) {
var coins = coins;
var cache = {};
this.makeChange = function (amount) {
var change = [], total = 0;
for (var i = coins.length; i >= 0; i--) {
var coin = coins[i];
while (total + coin <= amount) {
change.push(coin);
total += coin;
}
}
return change;
}
}
var minCoinChange = new MinCoinChange([1, 5, 10, 25]);
minCoinChange.makeChange(36);
//一个25, 一个10, 一个1
</script>