Quick sort là gì? Thuật toán sắp xếp và phân loại nhanh trong C++

4.0/5 (1 Reviews)

Quick sort là một loại thuật toán dùng để sắp xếp và phân loại nhanh tại chỗ, được ứng dụng phổ biến trong ngôn ngữ lập trình C++.

Một trong những thuật toán phổ biến nhất được sử dụng trong lập trình không thể không nhắc đến Quick short. Đây là một thuật toán hỗ trợ, có chức năng chính là sắp xếp và phân loại dữ liệu. Tuy nhiên, để hiểu rõ thuật toán này và biết cách sử dụng nó đòi hỏi người dùng phải dành thời gian phân tích kỹ lưỡng. Ở bài viết hôm nay, LPTech sẽ hướng dẫn đến bạn các khái niệm cơ bản về Quick sort. Cùng tìm hiểu nhé!

Quick sort là gì?

Quick sort là một thuật toán dùng trong ngôn ngữ lập trình C++, dùng để sắp xếp, phân loại nhanh. Quick sort hoạt động theo cơ chế: Chọn một phần tử bất kỳ làm điểm đánh dấu và chia mảng thành hai mảng con bằng cách so sánh các phần tử trong cùng mảng với điểm đã đánh dấu. Lúc này, mảng 1 sẽ chia ra làm các phần tử nhỏ hơn hoặc bằng điểm đánh dấu, mảng 2 sẽ chia ra làm các phần tử lớn hơn điểm đánh dấu.

Tùy theo mỗi người là thuật ngữ "điểm đánh dấu" quan trọng trong Quick sort sẽ có những cách gọi khác nhau như pivot hoặc phần tử chốt. Trong bài viết này, pivot hay phần tử chốt cũng đều được hiểu là điểm đánh dấu.

Hiện nay Quick sort được ứng dụng để sắp xếp dữ liệu vì nó có hiệu suất cao và có thể xử lý được tệp dữ liệu lớn. Thuật toán này cũng được sử dụng trong việc tìm kiếm danh sách sắp xếp. Bằng cách sắp xếp danh sách trước khi thực hiện thao tác tìm kiếm, qua đó, thuật toán sẽ được thực hiện nhanh chóng hơn.

> Xem thêm: Regex là gì? Ứng dụng và cách viết Regex chi tiết nhất

Quick sort là gì?

Cách chọn pivot (phần tử chốt)

Quick sort áp dụng cách thức Divide and Conquer để sắp xếp dữ liệu. Vì vậy nên tốc độ sắp xếp dữ liệu phụ thuộc vào khá nhiều vào kĩ thuật chọn pivot. Tùy theo từng trường hợp cụ thể mà ta sẽ có những cách chọn như sau:

  • Chọn phần tử đầu tiên trong mảng làm pivot.
  • Chọn phần tử cuối cùng trong mảng làm pivot.
  • Chọn phần tử có giá trị nằm giữa mảng làm pivot.
  • Chọn random một phần tử bất kỳ trong mảng làm pivot. Tuy nhiên nên hạn chế cách này vì sẽ dễ làm chúng ta rơi vào các trường hợp đặc biệt nơi các vòng lặp vô hạn diễn ra.

Cách 1: Cách chọn phần tử đầu làm pivot

Để chọn phần tử đầu làm pivot, bạn thực hiện lệnh sau:

quickSort = (unSortedArr) => {

        // nếu mảng không quá 1 phần tử thì mảng đó đã được sản xuất

        if (unSortedArr.length < 2) return unSortedArr;

        const pivot = unSortedArr[0]; //lấy phần tử đầu của mảng làm phần tử chốt

        const leftArr = []; // mảng chứa phần tử nhỏ hơn pivot

        const rightArr = []; // mảng chứa phần tử lớn hơn pivot

        let currentItem; // phần tử đang được xét

        // loop các phần tử còn lại trong mảng trừ phần tử chốt.

        // Do pivot là ptu đầu tiên nên i sẽ bắt đầu từ 1

        for (let i = 1; i < unSortedArr.length; i++) {     

            currentItem = unSortedArr[i];

            if (currentItem < pivot) {

                leftArr.push(currentItem);

            } else {

                rightArr.push(currentItem);

            }

        }

        return […this.quickSort(leftArr), pivot, …this.quickSort(rightArr)];

    }

Cách 2: Cách chọn phần tử cuối làm pivot

Đối với cách chọn phần tử chốt làm điểm đánh dấu, bạn áp dụng câu lệnh dưới đây nhé:

quickSort = (unSortedArr) => {

        if (unSortedArr.length < 2) return unSortedArr;

       

        const pivot = unSortedArr[unSortedArr.length – 1]; //phần tử cuối mảng làm chốt

        const leftArr = []; 

        const rightArr = []; 

        let currentItem;

        

        // Do pivot là ptu cuối nên length sẽ trừ đi 1 

        for (let i = 0; i < unSortedArr.length – 1; i++) {

            currentItem = unSortedArr[i];

            if (currentItem < pivot) {

                leftArr.push(currentItem);

            } else {

                rightArr.push(currentItem);

            }

        }

    

        return […this.quickSort(leftArr), pivot, …this.quickSort(rightArr)];

    }

Cách 3: Cách chọn phần tử giữa làm pivot

Với cách áp dụng chọn phần tử giữa làm pivot, cách thức thực hiện như sau:

quickSort = (unSortedArr) => {

        if (unSortedArr.length < 2) return unSortedArr;

        

        // lấy phần tử giữa làm chốt

        const pivotIndex = Math.floor(unSortedArr.length / 2);

        const pivot = unSortedArr[pivotIndex]; 

        const leftArr = []; 

        const rightArr = []; 

        let currentItem;

        

        unSortedArr.splice(pivotIndex, 1); // loại bỏ ptu pivot trong mảng

        

        for (let i = 0; i < unSortedArr.length; i++) {

            currentItem = unSortedArr[i];

            if (currentItem < pivot) {

                leftArr.push(currentItem);

            } else {

                rightArr.push(currentItem);

            }

        }

    

        return […this.quickSort(leftArr), pivot, …this.quickSort(rightArr)];

    }

Cách triển khai thuật toán Quick sort

Quick sort có độ phức tạp như thế nào?

Độ phức tạp của thuật toán Quick Sort phụ thuộc rất nhiều vào cách chọn phần tử chốt (pivot) và cấu trúc dữ liệu ban đầu. Quick sort có công thức tính thời gian như sau:

T(n) = T(k) + T(n-k-1) + θ(n)

Trong đó:

  • T(k) và T(n-k-1): Thời gian cho 2 cuộc gọi đệ quy.
  • θ(n): Tiến trình phân vùng.
  • k: Số phần tử nhỏ hơn phần tử chốt

Lưu ý: Thời gian của thuật toán Quick sort phụ thuộc vào mảng đầu và chiến lược chia trong mảng.

Từ công thức tính thời gian trong quick sort trên, chúng ta có thể suy ra được mức độ phức tạp của các trường hợp khi sử dụng quick sort. Cụ thể:

  • Trường hợp tốt nhất: Khi mỗi lần phân hoạch chia mảng thành hai phần có kích thước gần bằng nhau, độ phức tạp của Quick Sort là O(n log n). Đây là trường hợp lý tưởng và thường xảy ra khi dữ liệu đầu vào phân bố ngẫu nhiên.
  • Trường hợp trung bình: Trong hầu hết các trường hợp, độ phức tạp của Quick Sort cũng là O(n log n).
  • Trường hợp xấu nhất: Khi mảng đã được sắp xếp hoặc gần như sắp xếp, và việc chọn pivot luôn là phần tử nhỏ nhất hoặc lớn nhất, độ phức tạp sẽ lên đến O(n^2). Điều này xảy ra vì mỗi lần phân hoạch chỉ tạo ra một phần tử con có kích thước n-1.

Quick sort có độ phức tạp như thế nào?

Ưu, nhược điểm của quick sort

Là một thuật toán được sử dụng phổ biến trong ngôn ngữ lập trình, Quick sort có những ưu và nhược điểm cụ thể.

Ưu điểm của Quick sort

Quick sort là thuật toán được sử dụng rất phổ biến bởi nó mang theo những ưu điểm nổi bật như sau:

  • Quick sort là một trong những thuật toán sắp xếp và phân bố nhanh nhất, đặc biệt là khi áp dụng với những tệp dữ liệu lớn. Quick sort có độ phức tạp trung bình O(n log n) với tốc độ chạy nhanh dù cho đó là bộ dữ liệu lớn.
  • Thuật toán quick sort có thiết kế đơn giản, dễ triển khai và cũng rất dễ hiểu. Việc này hỗ trợ quá trình cài đặt và sử dụng Quick sort nhanh hơn khi so với những thuật toán còn lại.
  • Quick sort có khả năng tự sắp xếp trong chính nó, nghĩa là nó không cần dùng thêm mảng phụ để lưu trữ giá trị trung gian. Từ đó, Quick sort giúp hạn chế việc sử dụng bộ nhớ và gia tăng hiệu suất.
  • Thuật toán này có thể tùy biến theo từng tình huống khác nhau cho phù hợp với dữ liệu thực tế bằng cách chọn pivot hoặc xử lý phần tử bị trùng lặp.

Nhược điểm của Quick sort

Dù có nhiều ưu điểm, Quick sort vẫn là thuật toán tồn tại hạn chế nhất định. Nhược điểm lớn nhất của Quick sort đó là trong trường hợp xấu nhất, khi mảng đã bị đảo ngược, thuật toán sẽ hoạt động chậm và hao tốn nhiều bộ nhớ hơn.

Vì thế, khi dùng Quick sort, bạn cần cân nhắc kết hợp thêm những thuật toán khác để đảm bảo được hiệu suất tối ưu nhất.

Ưu, nhược điểm của quick sort

Minh họa thuật toán quick sort trong C++

Để hiểu rõ về thuật toán Quick sort và cách hoạt động của nó, hãy cùng LPTech phân tích qua ví dụ minh họa trong C++ sau đây. Các bước thực hiện bao gồm: Chọn phần tử chốt (Pivot) > Phân hoạch mảng (Partition) > Di chuyển chỉ số  > Đặt pivot vào đúng vị trí > Gọi đệ quy

Cho một mã triển khai của Quick Sort như sau:

#include <bits/stdc++.h>

using namespace std;

int partition(vector<int>& arr, int low, int high) {

  

    //Chọn pivot 

    int pivot = arr[high];

  

    // Chỉ số của phần tử nhỏ hơn và chỉ ra vị trí đúng của pivot đã tìm thấy cho đến nay

    int i = low - 1;

    // Duyệt qua mảng arr từ low đến high và di chuyển tất cả các phần tử nhỏ hơn về phía bên trái. Các phần tử từ low đến i sẽ nhỏ hơn sau mỗi lần lặp

    for (int j = low; j <= high - 1; j++) {

        if (arr[j] < pivot) {

            i++;

            swap(arr[i], arr[j]);

        }

    }

    

    // Di chuyển pivot đến sau các phần tử nhỏ hơn và trả về vị trí của nó

    swap(arr[i + 1], arr[high]);  

    return i + 1;

}

// Thực hiện hàm sắp xếp nhanh QuickSort

void quickSort(vector<int>& arr, int low, int high) {

  

    if (low < high) {

      

        // pi là chỉ số trả về của pivot sau khi phân hoạch

        int pi = partition(arr, low, high);

        // Gọi đệ quy cho các phần tử nhỏ hơn và các phần tử lớn hơn hoặc bằng

        quickSort(arr, low, pi - 1);

        quickSort(arr, pi + 1, high);

    }

}

int main() {

    vector<int> arr = {10, 7, 8, 9, 1, 5};

    int n = arr.size();

    quickSort(arr, 0, n - 1);

    cout << "Sorted Array\n";

    for (int i = 0; i < n; i++) {

        cout << arr[i] << " ";

    }

    return 0;

}

Đầu ra

Mảng được sắp xếp

1 5 7 8 9 10

Minh họa thuật toán quick sort

Ứng dụng của quick sort

Quick Sort được sử dụng trong nhiều lĩnh vực của khoa học máy tính, từ các hệ thống cơ sở dữ liệu đến các thuật toán tìm kiếm và đồ họa.

Trong lĩnh vực thiết kế app và thiết kế website chuyên nghiệp, quick sort hỗ trợ lập trình viên rất nhiều trong các công việc như sắp xếp danh sách, tìm kiếm nhị phân, phân tích dữ liệu và các thuật toán khác. Cụ thể:

  • Ứng dụng thương mại điện tử: Quick sort lúc này sẽ sắp xếp danh sách sản phẩm theo giá từ thấp đến cao hoặc ngược lại để người dùng dễ dàng tìm thấy sản phẩm phù hợp.
  • Ứng dụng quản lý danh bạ: Quick sort sắp xếp danh bạ theo tên, số điện thoại hoặc địa chỉ để thuận tiện cho việc tìm kiếm.
  • Ứng dụng bản đồ: Các điểm đánh dấu trên bản đồ được quick sort sắp xếp theo khoảng cách đến vị trí hiện tại của người dùng.

Bài viết trên đây đã mang đến những thông tin hữu ích về thuật toán sắp xếp Quick sort. Có thể nhận thấy được rằng, đây là thuật toán hữu ích và phổ biến được sử dụng trong ngôn ngữ lập trình C++. Hy vọng bài viết đã mang đến cho bạn những kiến thức cơ bản cần thiết về thuật toán Quick sort và phục vụ cho quá trình học tập, làm việc thật hiệu quả nhé!

Thông Tin Liên Hệ

ĐỀ XUẤT CHO BẠN

Bài Viết Cùng Chuyên Mục

article-img

Repository là gì? Các đặc điểm và tính năng của Repo Github

Repository là kho lưu trữ mã nguồn quan trọng trong lập trình, giúp quản lý và chia sẻ mã nguồn hiệu quả. Cùng tìm hiểu chi tiết về repository là gì!

author-avatar

Nguyễn Thị Ái Vy

05/01/2025

XEM CHI TIẾT
article-img

LLM là gì? Tổng quan chi tiết về mô hình ngôn ngữ lớn

LLM là gì? Mô hình ngôn ngữ lớn (LLM) là một bước đột phá trong trí tuệ nhân tạo, giúp máy hiểu và xử lý ngôn ngữ tự nhiên vượt trội. Tìm hiểu ngay!

author-avatar

Nguyễn Thị Ái Vy

05/01/2025

XEM CHI TIẾT
article-img

Redis là gì? Các đặc điểm và phân loại dữ liệu trong Redis

Redis là gì? Hệ thống cơ sở dữ liệu NoSQL phổ biến với tốc độ xử lý vượt trội, hỗ trợ lưu trữ linh hoạt và nhiều ứng dụng trong công nghệ hiện đại.

author-avatar

Nguyễn Thị Ái Vy

05/01/2025

XEM CHI TIẾT
article-img

NGINX là gì? Hướng dẫn cài đặt và cấu hình NGINX

NGINX là gì? NGINX là một máy chủ web phổ biến được sử dụng rộng rãi nhờ khả năng xử lý lượng lớn kết nối và tối ưu hóa hiệu suất.

author-avatar

Nguyễn Thị Ái Vy

04/01/2025

XEM CHI TIẾT
article-img

Buffer là gì? Công dụng của Buffer trong truyền dữ liệu

Buffer là gì? Đây là một vùng bộ nhớ tạm thời giúp xử lý và lưu trữ dữ liệu trong lập trình và công nghệ. Tìm hiểu về khái niệm và công dụng của Buffer.

author-avatar

Nguyễn Thị Ái Vy

02/01/2025

XEM CHI TIẾT
article-img

Env là gì? Hướng dẫn lưu trữ biến môi trường hiệu quả

Các lập trình viên thường sử dụng file .env để lưu trữ các biến môi trường một cách an toàn và tiện lợi. Vậy file .env là gì và làm sao để sử dụng hiệu quả?

author-avatar

Nguyễn Thị Ái Vy

31/12/2024

XEM CHI TIẾT

Chất Lượng Sản Phẩm Tạo Nên Uy Tín Doanh Nghiệp.

Tầm nhìn của LPTech mong muốn trở thành công ty Công nghệ không chỉ phát triển tại thị trường Việt Nam mà còn mở rộng ra cả khu vực Asia. Vậy nên, mỗi một công việc mà LPTech làm đều sẽ ảnh hưởng đến thương hiệu của công ty ở hiện tại lẫn tương lai. Chính vì thế, quý khách hàng có thể yên tâm về chất lượng website được thiết kế tại LPTech.

LIÊN HỆ NGAY
bgQuality

THÔNG BÁO

Tin nổi bật

notification-img
Thông báo lịch nghỉ lễ Mùng 10 tháng 3, 30/4 và 1/5 năm 2026

Công ty TNHH Thương mại Điện tử Công nghệ LP xin thông báo đến Quý Khách hàng, Đối tác và toàn thể Nhân viên lịch nghỉ lễ 10/3, 30/4 và 1/5 năm 2026

notification-img
Thông báo nghỉ Tết Nguyên Đán 2026

LPTech kính chúc Quý Khách hàng, Quý Đối tác và toàn thể nhân sự một năm mới an khang, nhiều niềm vui, đủ đầy yêu thương và vững bước thành công.

notification-img
Thông báo lịch nghỉ Tết Dương lịch 2026

Chào đón năm mới 2026, LPTech xin gửi đến Quý Khách hàng, Đối tác và toàn thể nhân viên lời chúc sức khỏe, bình an và thành công, đồng thời thông tin về lịch nghỉ Tết Dương lịch 2026 của Công ty.

notification-img
Vũng Tàu: Du lịch công ty 2 ngày 1 đêm cùng LPTech

Giữa những ngày cuối năm bận rộn, cả team rủ nhau đi trốn một chuyến về Vũng Tàu để đổi gió và tận hưởng biển xanh. Một chuyến đi đầy ắp kỉ niệm.

notification-img
Thông báo lịch nghỉ du lịch thường niên 2025

Một chuyến du lịch ngắn ngày nhưng đầy năng lượng này sẽ giúp đội ngũ LPTech tạm rời nhịp làm việc, nghỉ ngơi và sẵn sàng cho những mục tiêu mới.

notification-img
Tết Trung Thu 2025 – Mùa trăng đoàn viên, mùa yêu thương lan tỏa

Giữa sắc đèn lung linh và hương bánh nồng nàn, Tết Trung Thu trở về như bản nhạc dịu êm của đoàn viên, hạnh phúc và sự gắn kết.

notification-img
LPTech chào mừng Quốc khánh 2/9 – 80 năm tự hào dân tộc

Kỷ niệm Quốc khánh 2/9 – LPTech tự hào đồng hành cùng tinh thần dân tộc, tổ chức nhiều hoạt động nội bộ ý nghĩa để gắn kết tập thể và lan tỏa giá trị yêu nước.

notification-img
Thông báo lịch nghỉ lễ Quốc khánh 2025

LPTech kính chúc Quý khách hàng, Đối tác và toàn thể nhân viên có một kỳ nghỉ lễ vui bên gia đình và người thân!