Sắp xếp trộn
Trong khoa học máy tính, sắp xếp trộn (merge sort) là một thuật toán sắp xếp để sắp xếp các danh sách (hoặc bất kỳ cấu trúc dữ liệu nào có thể truy cập tuần tự, v.d. luồng tập tin) theo một trật tự nào đó. Nó được xếp vào thể loại sắp xếp so sánh. Thuật toán này là một ví dụ tương đối điển hình của lối thuật toán chia để trị do John von Neumann đưa ra lần đầu năm 1945[1]. Một thuật toán chi tiết được Goldstine và Neumann đưa ra năm 1948.[2] TrộnGiả sử có hai danh sách đã được sắp xếp và . Ta có thể trộn chúng lại thành một danh sách mới được sắp xếp theo cách sau:
Ví dụ: Cho hai danh sách , quá trình hòa nhập diễn ra như sau:
Trộn tại chỗGiả sử trong danh sách có 2 danh sách con kề nhau và đã được sắp. Ta áp dụng cách trộn tương tự như trên để trộn hai danh sách con vào một danh sách tạm rồi trả lại các giá trị của danh sách tạm T về danh sách A. Làm như vậy gọi là trộn tại chỗ. Trộn từ dưới lênNếu danh sách con chỉ gồm hai phần tử, mỗi nửa của nó gồm một phần tử đương nhiên đã được sắp. Do đó việc trộn tại chỗ hai nửa danh sách này cho danh sách con 2 phân tử được sắp. Xuất phát từ đầu danh sách ta trộn với , với ,... Khi đó mọi danh sách con gồm hai phần tử của đã được sắp. Tiếp tục trộn các danh sách con kế tiếp nhau gồm 2 phần tử thành các danh sách con 4 phần tử... Mỗi lần trộn số các danh sách con cần trộn giảm đi một nửa. Quá trình dừng lại khi số danh sách con chỉ còn một. Ví dụ: Cho danh sách
Sắp xếp trộn đệ quyMột cách gọi đệ quy của sắp xếp trộn cũng thường được hướng dẫn trong các giáo trình giải thuật. Để sắp xếp trộn đoạn của danh sách ta chia đoạn đó thành 2 phần và ,trong đó tiến hành sắp xếp với mỗi phần rồi trộn chúng lại. Lời gọi thủ tục sắp xếp trộn với sẽ cho kết quả là sắp toàn bộ danh sách Ví dụ: Cho danh sách
Trộn các đường tự nhiênNhư trong phần đánh giá giải thuật, một trong những nhược điểm lớn của thuật toán Trộn trực tiếp là không tận dụng những thông tin về đặc tính của dãy cần sắp xếp. Ví dụ trường hợp dãy đã có thứ tự sẵn. Chính vì vậy, trong thực tế người ta ít dùng thuật toán trộn trực tiếp mà người ta dùng phiên bản cải tiến của nó. Phiên bản này thường được biết với tên gọi thuật toán trộn tự nhiên (Natural Merge sort). Khái niệm đường chạyĐể khảo sát thuật toán trộn tự nhiên, trước tiên ta cần định nghĩa khái niệm đường chạy (run):
Giải thuậtCác bước thực hiện thuật toán trộn tự nhiên như sau:
Ưu và nhược điểmThuật toán trộn tự nhiên khác thuật toán trộn trực tiếp ở chỗ thay vì luôn cứng nhắc phân hoạch theo dãy con có chiều dài k, việc phân hoạch sẽ theo đơn vị là đường chạy. ta chỉ cần biết số đường chạy của a sau lần phân hoạch cuối cùng là có thể biết thời điểm dừng của thuật toán vì dãy đã có thứ tự là dãy chi có một đường chạy. Một nhược điểm lớn nữa của thuật toán trộn là khi cài đặt thuật toán đòi hỏi thêm không gian bộ nhớ để lưu các dãy phụ b, c. Hạn chế này khó chấp nhận trong thực tế vì các dãy cần sắp xếp thường có kích thước lớn. Vì vậy thuật toán trộn thường được dùng để sắp xếp các cấu trúc dữ liệu khác phù hợp hơn như danh sách liên kết hoặc file. Mã giảMã giả ngắn gọnprocedure mergesort(l,r: integer);
var i, j, k, m: integer;
begin
if r>l then
begin
m:=(r+l)shr 1;
mergesort(l,m); mergesort(m+1,r);
for i:= m downto l do b[i]:=a[i];
for j:=m+1 to r do b[r+m+1-j]:=a[j];
for k:=l to r do
if b[i] < b[j] then
begin a[k]:=b[i]; i:=i+1 end
else
begin a[k]:=b[j]; j:=j-1 end;
end;
end;
Trộn tại chỗ
Trộn từ dưới lên
Trộn đệ quy
void merge(int array[], int first, int middle, int last) { int temp[last + 1]; int first1, last1, first2, last2; int index = first; first1 = first; last1 = middle; first2 = middle+1; last2 = last; while((first1 <= last1) && (first2 <= last2)) { if(array[first1] < array[first2]) { temp[index] = array[first1]; index ++; first1 ++; } else { temp[index] = array[first2]; index ++; first2 ++; } } if(first2 > last2) { while(first1 <= last1) { temp[index] = array[first1]; index ++; first1 ++; } } if(first1 > last1) { while(first2 <= last2) { temp[index] = array[first2]; index ++; first2 ++; } } for(int i = first; i <= last; i ++) { array[i] = temp[i - first]; } return; } void mergeSort(int array[], int first, int last) { if(first < last) { int middle = int((first + last) / 2); mergeSort(array, first, middle); mergeSort(array, middle + 1, last); merge(array, first, middle, last); } } Xem thêmTham khảo
Liên kết ngoài
|
Portal di Ensiklopedia Dunia