循序搜尋法(使用衛兵)


說明

搜尋的目的,是在「已排序的資料」中尋找指定的資料,而當中循序搜尋是最基本的搜尋法,只要從資料開頭尋找到最後,看看是否找到資料即可。

解法

初學者看到循序搜尋,多數都會使用以下的方式來進行搜尋:
while(i < MAX) {
    if(number[i] == k) {
        printf("找到指定值");
        break;
    }
    i++;
}
 

這個方法基本上沒有錯,但是可以加以改善,可以利用設定衛兵的方式,省去if判斷式,衛兵通常設定在數列最後或是最前方,假設設定在列前方好了(索引0的 位置),我們從數列後方向前找,如果找到指定的資料時,其索引值不是0,表示在數列走訪完之前就找到了,在程式的撰寫上,只要使用一個while迴圈就可 以了。

下面的程式為了配合衛兵的設置,自行使用快速排序法先將產生的數列排序,然後才進行搜尋,若只是數字的話,通常您可以使用程式語言函式庫所提供的搜尋函式。

實作:C    Java    Python    Scala    Ruby

  • C
#include <stdio.h> 
#include <stdlib.h>
#include <time.h>
#define MAX 10
#define SWAP(x,y) {int t; t = x; x = y; y = t;}

int search(int[]);
int partition(int[], int, int);
void quickSort(int[], int, int);

int main(void) {
srand(time(NULL));

int number[MAX+1] = {0};

int i;
for(i = 1; i <= MAX; i++)
number[i] = rand() % 100;

quickSort(number, 1, MAX);

printf("數列:");
for(i = 1; i <= MAX; i++)
printf("%d ", number[i]);

printf("\n輸入搜尋值:");
scanf("%d", &number[0]);

int find = search(number);
if(find >= 0)
printf("\n找到數值於索引 %d ", find);
else
printf("\n找不到數值");

printf("\n");

return 0;
}

int search(int number[]) {
int i = MAX;
while(number[i] != number[0]) i--;
return i;
}

int partition(int number[], int left, int right) {
int s = number[right];
int i = left - 1;

int j;
for(j = left; j < right; j++) {
if(number[j] <= s) {
i++;
SWAP(number[i], number[j]);
}
}

SWAP(number[i+1], number[right]);
return i+1;
}

void quickSort(int number[], int left, int right) {
if(left < right) {
int q = partition(number, left, right);
quickSort(number, left, q-1);
quickSort(number, q+1, right);
}
}

  • Java
public class Search {
public static int linear(int[] number, int des) {
int[] tmp = new int[number.length + 1];
for(int i = 1; i < tmp.length; i++) {
tmp[i] = number[i-1];
}

tmp[0] = des;

int i = number.length;
while(tmp[i] != tmp[0])
i--;

return i - 1;
}

public static void main(String[] args) {
int[] number = {1, 2, 3, 4, 6, 7, 8};
int find = Search.linear(number, 3);
System.out.println(find >= 0 ? "找到數值於索引" + find : "找不到數值");
}
}

  • Python
def search(number, des):
tmp = [0] * (len(number) + 1)
for i in range(1, len(number)):
tmp[i] = number[i - 1]
tmp[0] = des
i = len(number)
while tmp[i] != tmp[0]:
i -= 1
return i - 1

number = [1, 4, 2, 6, 7, 3, 9, 8]
number.sort()
find = search(number, 2)
print("找到數值於索引 " + str(find) if find >= 0 else "找不到數值")

  • Scala
object Search {
def linear(number: Array[Int], des: Int): Int = {
val tmp = new Array[Int](number.length + 1)
for(i <- 1 until tmp.length) {
tmp(i) = number(i - 1)
}
tmp(0) = des
for(i <- number.length until (0, -1) if tmp(i) == des) {
return i -1
}
return -1
}
}

val number = Array(1, 2, 3, 4, 6, 7, 8)
val find = Search.linear(number, 3)
println(if(find >= 0) "找到數值於索引 " + find else "找不到數值")

  • Ruby
# encoding: Big5
def search(number, des)
tmp = Array.new(number.length + 1, 0)
1.upto(number.length - 1) { |i|
tmp[i] = number[i - 1]
}
tmp[0] = des
i = number.length
while tmp[i] != tmp[0]
i -= 1
end
i - 1
end

number = [1, 4, 2, 6, 7, 3, 9, 8]
number.sort!
find = search(number, 2)
print find >= 0 ? "找到數值於索引 " + find.to_s : "找不到數值", "\n"