Thuật Toán Sắp Xếp Chọn (Selection Sort) - DNMTechs

Giải thuật sắp xếp chọn (Selection Sort) là một giải thuật đơn giản. Giải thuật sắp xếp này là một giải thuật dựa trên việc so sánh in-place (hay còn gọi là So sánh tại chỗ), trong đó danh sách được chia thành hai phần, phần được sắp xếp (sorted list) ở bên trái và phần chưa được sắp xếp (unsorted list) ở bên phải. Ban đầu, phần được sắp xếp là trống và phần chưa được sắp xếp là toàn bộ danh sách ban đầu.

Chọn phần tử nhỏ nhất trong n phần tử ban đầu, đưa phần tử này về vị trí đúng là đầu tiên của dãy hiện hành. Sau đó không quan tâm đến nó nữa, xem dãy hiện hành chỉ còn n-1 phần tử của dãy ban đầu, bắt đầu từ vị trí thứ 2. Lặp lại quá trình trên cho dãy hiện hành đến khi dãy hiện hành chỉ còn một phần tử. Dãy ban đầu có n phần tử, vậy tóm tắt ý tưởng thuật toán là thực hiện n-1 lượt việc đưa phần tử nhỏ nhất trong dãy hiện hành về vị trí đúng ở đầu dãy.

Giải thuật này không phù hợp với tập dữ liệu lớn khi mà độ phức tạp trường hợp xấu nhất và trường hợp trung bình là O(n2) với n là số phần tử.

Phân loại: Thuật toán sắp xếp Cấu trúc dữ liệu: Ngẫu nhiên Phức tạp thời gian: Trung bình O(n2) Phức tạp dữ liệu: Không tốn thêm vùng nhớ Tối ưu: Thỉnh thoảng

Các bước thực hiện

  • Bước 1: i=1
  • Bước 2: Tìm phần tử a[min] nhỏ nhất trong dãy hiện hành từ a[i] đến a[n]
  • Bước 3: Hoán vị a[min] và a[i]
  • Bước 4: Nếu i<=n-1 thì i=i+1; Lặp lại bước 2
  • Ngược lại: Dừng. n-1 phần tử đã nằm đúng vị trí.

Ví dụ: Sắp xếp một dãy theo thứ tự tăng dần từ 1 đến 6 sử dụng thuật toán sắp xếp chọn  (Selection Sort) .

Đánh giá

  • Thuật toán ít phải đổi chỗ các phần tử nhất trong số các thuật toán sắp xếp(n lần hoán vị) nhưng có độ phức tạp so sánh là O(n2) (n2/2 phép so sánh).
  • Thuật toán tốn thời gian gần như bằng nhau đối với mảng đã được sắp xếp cũng như mảng chưa được sắp xếp.

Ví dụ minh họa bằng ngôn ngữ C++

Viết chương trình C++ nhập vào một mảng và sắp xếp mảng đã nhập sử dụng thuật toán sắp xếp chọn.

#include<iostream> using namespace std; void input_func(int A[], int &n) { cout << "nhap so phan tu cua mang n:"; cin >> n; for (int i = 0; i<n; i++) { cout << "phan tu A[" << i << "]" << "="; cin >> A[i]; } } void swap(int *xp, int *yp) { int temp = *xp; *xp = *yp; *yp = temp; } void SelectionSort(int A[], int n) { int i, j, min_idx; for (i = 0; i < n - 1; i++) { // Tìm phần tử nhỏ nhất trong mảng min_idx = i; for (j = i + 1; j < n; j++) if (A[j] < A[min_idx]) min_idx = j; // Đổi chỗ phần tử nhỏ nhất trong mảng swap(&A[min_idx], &A[i]); } } void output_func(int A[], int &n) { for (int i = 0; i < n; i++) { cout << A[i] << " "; } } int main() { int A[100], n; input_func(A, n); cout << "mang ban dau la:"; output_func(A, n); SelectionSort(A, n); cout << "mang khi sap xep la:"; output_func(A, n); cin.get(); getchar(); return 0; }

Ta cũng có thể cải tiến thuật toán trên để giảm bớt số vòng lặp for như sau:

#include<iostream> using namespace std; void input_func(int A[], int &n) { cout << "nhap so phan tu cua mang n:"; cin >> n; for (int i = 0; i<n; i++) { cout << "phan tu A[" << i << "]" << "="; cin >> A[i]; } } void SelectionSort_Improve(int A[], int n) { for (int i = 0; i < n - 1; i++){ int max = i; for (int j = i + 1; j < n; j++) { if (A[max] > A[j]){ max = j; } } if (max != i){// Sử dụng phép OR loại trừ bit A[max] ^= A[i]; A[i] ^= A[max]; A[max] ^= A[i]; } } } void output_func(int A[], int &n) { for (int i = 0; i < n; i++) { cout << A[i] << " "; } } int main() { int A[100], n; input_func(A, n); cout << "mang ban dau la:"; output_func(A, n); SelectionSort_Improve(A, n); cout << "mang khi sap xep la:"; output_func(A, n); cin.get(); getchar(); return 0; }

Chạy thử và xem kết quả, nhấp đôi vào code để copy.

Từ khóa » độ Phức Tạp Của Selection Sort