C語言排序方法(冒泡,選擇,插入,歸并,快速)
1.冒泡排序
它重復地走訪過要排序的元素列,依次比較兩個相鄰的元素,如果順序錯誤就把他們交換過來。走訪元素的工作是重復地進行直到沒有相鄰元素需要交換,也就是說該元素列已經排序完成。
算法步驟 比較相鄰的元素。如果第一個比第二個大,就交換他們兩個。 對每一對相鄰元素作同樣的工作,從開始第一對到結尾的最后一對。這步做完后,最后的元素會是最大的數。
針對所有的元素重復以上的步驟,除了最后一個。 持續(xù)每次對越來越少的元素重復上面的步驟,直到沒有任何一對數字需要比較。
執(zhí)行過程:
# include <stdio.h> # include <string.h> int main(void) { int arr[] = {5, 2, 3, -8, 34, 76, 32, 43, 0, -70, 35, 543, 6}; int len= sizeof(arr) / sizeof(arr[0]);; int i; //比較的輪數 int j; //每輪比較的次數 int temp; //交換數據時用于存放中間數據 for (i=0; i<len-1; ++i) //比較n-1輪 { for (j=0; j<len-1-i; ++j) //每輪比較n-1-i次, { if (arr[j] < arr[j+1]) { temp = arr[j]; arr[j] = arr[j+1]; arr[j+1] = temp; } } } printf("排序后:\n"); for (i=0; i<len; ++i) { printf("%d\x20", arr[i]); } printf("\n"); return 0; }
2.選擇排序
選擇排序(Selection sort)是一種簡單直觀的排序算法。它的工作原理是:第一次從待排序的數據元素中選出最?。ɑ蜃畲螅┑囊粋€元素,存放在序列的起始位置,然后再從剩余的未排序元素中尋找到最小(大)元素,然后放到已排序的序列的末尾。以此類推,直到全部待排序的數據元素的個數為零。選擇排序是不穩(wěn)定的排序方法。
算法步驟 首先在未排序序列中找到最?。ù螅┰兀娣诺脚判蛐蛄械钠鹗嘉恢?。
再從剩余未排序元素中繼續(xù)
尋找最小(大)元素,然后放到已排序序列的末尾。
重復第二步,直到所有元素均排序完畢。
代碼:
#include <stdio.h> # include <string.h> int main() { int arr[] = { 5, 2, 3, -8, 34, 76, 32, 43, 0, -70, 35, 543, 6}; int len = (int) sizeof(arr) / sizeof(*arr); int i, j, temp; for (i = 0; i < len - 1; i++) for (j = 0; j < len - 1 - i; j++) if (arr[j] > arr[j + 1]) { temp = arr[j]; arr[j] = arr[j + 1]; arr[j + 1] = temp; } for (i = 0; i < len; i++) printf("%d ", arr[i]); return 0; }
3.插入排序
插入排序,一般也被稱為直接插入排序。對于少量元素的排序,它是一個有效的算法 [1] 。插入排序是一種最簡單的排序方法,它的基本思想是將一個記錄插入到已經排好序的有序表中,從而一個新的、記錄數增1的有序表。在其實現(xiàn)過程使用雙層循環(huán),外層循環(huán)對除了第一個元素之外的所有元素,內層循環(huán)對當前元素前面有序表進行待插入位置查找,并進行移動 [2] 。
算法步驟 將第一待排序序列第一個元素看做一個有序序列,把第二個元素到最后一個元素當成是未排序序列。
從頭到尾依次掃描未排序序列,將掃描到的每個元素插入有序序列的適當位置。(如果待插入的元素與有序序列中的某個元素相等,則將待插入元素插入到相等元素的后面。)
#include <stdio.h> # include <string.h> int main(){ int arr[] = { 5, 2, 3, -8, 34, 76, 32, 43, 0, -70, 35, 543, 6}; int len = (int) sizeof(arr) / sizeof(*arr); int i,j,x; for( i= 1; i<len; i++){ if(arr[i] < arr[i-1]){//若第 i 個元素大于 i-1 元素則直接插入;反之,需要找到適當的插入位置后在插入。 j= i-1; x = arr[i]; while(j>-1 && x < arr[j]){ //采用順序查找方式找到插入的位置,在查找的同時,將數組中的元素進行后移操作,給插入元素騰出空間 arr[j+1] = arr[j]; j--; } arr[j+1] = x; //插入到正確位置 } } for(j=0; j<len; j++){ printf("%d ",arr[j]); } printf("\n"); return 0; }
4.歸并排序
歸并排序(Merge sort)是建立在歸并操作上的一種有效的排序算法。該算法是采用分治法(Divide and Conquer)的一個非常典型的應用。
作為一種典型的分而治之思想的算法應用,歸并排序的實現(xiàn)由兩種方法:
自上而下的遞歸(所有遞歸的方法都可以用迭代重寫,所以就有了第 2 種方法);
自下而上的迭代;
算法步驟 申請空間,使其大小為兩個已經排序序列之和,該空間用來存放合并后的序列;
設定兩個指針,最初位置分別為兩個已經排序序列的起始位置;
比較兩個指針所指向的元素,選擇相對小的元素放入到合并空間,并移動指針到下一位置;
重復步驟 3 直到某一指針達到序列尾;
將另一序列剩下的所有元素直接復制到合并序列尾。
#include <stdio.h>
#define MAXSIZE 10
// 遞歸的方式實現(xiàn)歸并排序
// 實現(xiàn)歸并,并把結果存放到list1
# include <string.h> #include <stdio.h> void merging(int *list1, int list1_size, int *list2, int list2_size) { int i,j,k, m; int temp[MAXSIZE]; i = j = k = 0; while(i < list1_size && j < list2_size) { if(list1[i] < list2[j]) { temp[k] = list1[i]; k++; i++; } else { temp[k++] = list2[j++]; } } while(i < list1_size) { temp[k++] = list1[i++]; } while(j < list2_size) { temp[k++] = list2[j++]; } for(m = 0;m < (list1_size + list2_size);m++) { list1[m] = temp[m]; } } void MergeSort(int k[], int n) { if(n > 1) { /* *list1是左半部分,list2是右半部分 */ int *list1 = k; int list1_size = n/2; int *list2 = k + list1_size; int list2_size = n - list1_size; MergeSort(list1, list1_size); MergeSort(list2, list2_size); // 把兩個合在一起 merging(list1, list1_size, list2, list2_size); } } int main() { int i, arr[] = { 5, 2, 3, -8, 34, 76, 32, 43, 0, -70, 35, 543, 6}; int len = (int) sizeof(arr) / sizeof(*arr); MergeSort(arr, len); printf("排序后的結果是:"); for(i = 0;i < len;i++) { printf("%d", a[i]); } printf("\n\n"); return 0; }
5.快速排序
原理:
快速排序,給基準數據找其正確索引位置的過程.
如下圖所示,假設最開始的基準數據為數組第一個元素23,則首先用一個臨時變量去存儲基準數據,即tmp=23;然后分別從數組的兩端掃描數組,設兩個指示標志:low指向起始位置,high指向末尾.
如果掃描到的值大于基準數據就讓high減1,如果發(fā)現(xiàn)有元素比該基準數據的值小(如上圖中18<=tmp),就將high位置的值賦值給low位置。
如果掃描到的值小于基準數據就讓low加1,如果發(fā)現(xiàn)有元素大于基準數據的值(如上圖46=>tmp),就再將low位置的值賦值給high位置的值.
算法步驟 從數列中挑出一個元素,稱為 “基準”(pivot);
重新排序數列,所有元素比基準值小的擺放在基準前面,所有元素比基準值大的擺在基準的后面(相同的數可以到任一邊)。在這個分區(qū)退出之后,該基準就處于數列的中間位置。這個稱為分區(qū)(partition)操作;
遞歸地(recursive)把小于基準值元素的子數列和大于基準值元素的子數列排序;
#include "stdio.h" typedef struct _Range { int start, end; //開始指向分別指向兩端 } Range; Range new_Range(int s, int e) { Range r; r.start = s; //開始指向需要排序數組的兩端 r.end = e; return r; //返回一個結構體 } void swap(int *x, int *y) { //交換數據函數 int t = *x; *x = *y; *y = t; } void quick_sort(int arr[], const int len) { if (len <= 0) return; //保證數據長度大于0 Range r[len]; int p = 0; r[p++] = new_Range(0, len - 1); while (p) { Range range = r[--p]; if (range.start >= range.end) continue; int mid = arr[(range.start + range.end) / 2]; // 選取中間點作為基準點 int left = range.start, right = range.end; do { while (arr[left] < mid) ++left; // 檢測基準點左側是否符合要求 while (arr[right] > mid) --right; //檢測基準點右側是否符合要求 if (left <= right) { swap(&arr[left], &arr[right]); left++; right--; // 移動指針以繼續(xù) } } while (left <= right); if (range.start < right) r[p++] = new_Range(range.start, right); if (range.end > left) r[p++] = new_Range(left, range.end); } } int main() { int j; int arr[] = { 5, 2, 3, -8, 34, 76, 32, 43, 0, -70, 35, 543, 6}; int len = (int) sizeof(arr) / sizeof(*arr); quick_sort(arr,len); for(j=0; j<len; j++){ printf("%d ",arr[j]); } printf("\n"); }
總結
本篇文章就到這里了,希望能給你帶來幫助,也希望您能夠多多關注腳本之家的更多內容!
相關文章
用C/C++實現(xiàn)linux下檢測網絡接口狀態(tài)
這篇文章主要為大家詳細介紹了用c/c++實現(xiàn)linux下檢測網絡接口狀態(tài),具有一定的參考價值,感興趣的小伙伴們可以參考一下2018-06-06