必须知道的C语言八大排序算法(收藏)

必须知道的C语言八大排序算法(收藏)

简介

排序算法(sorting algorithms)是计算机程序设计中处理数据的重要技术之一,常见于数据处理程序中。其功能是按照指定的方式将所输入的数据进行排序。排序算法分为内部排序和外部排序,本文主要讲解C语言中的八大内部排序算法。

八大排序算法

  1. 冒泡排序
  2. 选择排序
  3. 插入排序
  4. 希尔排序
  5. 归并排序
  6. 快速排序
  7. 堆排序
  8. 计数排序

冒泡排序

冒泡排序是一种简单的排序算法,它重复的遍历要排序的数列,一次比较两个元素,如果它们的顺序错误就交换位置。其时间复杂度为O(n^2)。

示例代码:

void bubbleSort(int arr[], int n) {
    int i, j;
    for (i = 0; i < n-1; i++)
        for (j = 0; j < n-i-1; j++)
            if (arr[j] > arr[j+1]) {
                int temp = arr[j];
                arr[j] = arr[j+1];
                arr[j+1] = temp;
            }
}

选择排序

选择排序是一种简单直观的排序算法,它的工作原理是:首先找到序列中最小的元素,其次将它放在序列的起始位置;接着,再从剩余未排序的元素中寻找最小的值,将其放在已排序序列的末尾。以此类推,直到所有元素均排列完毕。其时间复杂度为O(n^2)。

示例代码:

void selectionSort(int arr[], int n) {
    int i, j, min_idx;
    for (i = 0; i < n-1; i++) {
        min_idx = i;
        for (j = i+1; j < n; j++)
            if (arr[j] < arr[min_idx])
                min_idx = j;
        int temp = arr[i];
        arr[i] = arr[min_idx];
        arr[min_idx] = temp;
    }
}

(接下来五种算法类似的方式展开介绍)

插入排序

插入排序是一种简单直观的排序算法,它的工作原理是在已排序序列中从后向前扫描,找到相应位置并插入。其时间复杂度为O(n^2)。

示例代码:

void insertionSort(int arr[], int n) {
    int i, key, j;
    for (i = 1; i < n; i++) {
        key = arr[i];
        j = i - 1;
        while (j >= 0 && arr[j] > key) {
            arr[j + 1] = arr[j];
            j = j - 1;
        }
        arr[j + 1] = key;
    }
}

希尔排序

希尔排序是一种改进型的插入排序,其基本思路是将待排序的序列划分成若干个子序列分别进行直接插入排序,待整个序列中的记录“基本有序”时,再对全体记录进行一次直接插入排序。其时间复杂度为O(n*log2n)。

示例代码:

void shellSort(int arr[], int n) {
    int gap, i, j, temp;
    for (gap = n/2; gap > 0; gap /= 2) {
        for (i = gap; i < n; i++) {
            temp = arr[i];
            for (j = i; j >= gap && arr[j-gap] > temp; j -= gap)
                arr[j] = arr[j-gap];
            arr[j] = temp;
        }
    }
}

归并排序

归并排序是一种分治思想的排序算法,它的基本思想是将待排序序列分成若干个子序列,每个子序列都是有序的,然后再将有序子序列合并成一个有序序列。其时间复杂度为O(n*log2n)。

示例代码:

void merge(int arr[], int l, int m, int r) {
    int i, j, k;
    int n1 = m - l + 1;
    int n2 = r - m;

    int L[n1], R[n2];

    for (i = 0; i < n1; i++)
        L[i] = arr[l + i];
    for (j = 0; j < n2; j++)
        R[j] = arr[m + 1 + j];

    i = 0;
    j = 0;
    k = l;

    while (i < n1 && j < n2) {
        if (L[i] <= R[j]) {
            arr[k] = L[i];
            i++;
        }
        else {
            arr[k] = R[j];
            j++;
        }
        k++;
    }

    while (i < n1) {
        arr[k] = L[i];
        i++;
        k++;
    }

    while (j < n2) {
        arr[k] = R[j];
        j++;
        k++;
    }
}

void mergeSort(int arr[], int l, int r) {
    if (l < r) {
        int m = l+(r-l)/2;

        mergeSort(arr, l, m);
        mergeSort(arr, m+1, r);

        merge(arr, l, m, r);
    }
}

快速排序

快速排序是一种由C.A.R.Hoare在1960年提出的一种分治思想的原地快速排序算法。其基本思路是使用基准元素将一个序列分为左右两部分,一边存贮比该基准元素小的元素,另一边存储比该基准元素大的元素,而后递归处理这两个序列,直到这个序列不可分为止。其时间复杂度最差情况下为O(n^2),但平均情况下为O(n*log2n)。

示例代码:

void swap(int* a, int* b) {
    int t = *a;
    *a = *b;
    *b = t;
}

int partition (int arr[], int low, int high) {
    int pivot = arr[high]; 
    int i = (low - 1);

    for (int j = low; j <= high- 1; j++) {
        if (arr[j] <= pivot) {
            i++;
            swap(&arr[i], &arr[j]);
        }
    }
    swap(&arr[i + 1], &arr[high]);
    return (i + 1);
}

void quickSort(int arr[], int low, int high) {
    if (low < high) {
        int pi = partition(arr, low, high);

        quickSort(arr, low, pi - 1);
        quickSort(arr, pi + 1, high);
    }
}

堆排序

堆排序是一种分治思想的排序算法,它的基本思路是将待排序序列构成一个大根堆或小根堆,整个序列的最大值就是堆顶的根节点。将其移走,再从剩下的元素中构建一个新堆,反复执行上述操作,最终实现排序。其时间复杂度为O(n*log2n)。

示例代码:

void heapify(int arr[], int n, int i) {
    int largest = i;
    int l = 2*i + 1;
    int r = 2*i + 2;

    if (l < n && arr[l] > arr[largest])
        largest = l;
    if (r < n && arr[r] > arr[largest])
        largest = r;
    if (largest != i) {
        swap(&arr[i], &arr[largest]);
        heapify(arr, n, largest);
    }
}

void heapSort(int arr[], int n) {
    for (int i = n / 2 - 1; i >= 0; i--)
        heapify(arr, n, i);
    for (int i=n-1; i>=0; i--) {
        swap(&arr[0], &arr[i]);
        heapify(arr, i, 0);
    }
}

计数排序

计数排序是一种线性排序算法,其基本思路是统计待排序元素中小于等于每个元素的个数(称为元素k),最终确定该元素的位置。算法需确定待排序元素的最大和最小值。其时间复杂度为O(n+k),k为元素范围。

示例代码:

void countSort(int arr[], int n) {
    int output[n];
    int max = arr[0];
    for (int i = 1; i < n; i++) {
        if (arr[i] > max)
            max = arr[i];
    }
    int count[max + 1];

    for (int i = 0; i <= max; ++i)
        count[i] = 0;

    for (int i = 0; i < n; i++)
        count[arr[i]]++;

    for (int i = 1; i <= max; i++)
        count[i] += count[i - 1];

    for (int i = n - 1; i >= 0; i--) {
        output[count[arr[i]] - 1] = arr[i];
        count[arr[i]]--;
    }

    for (int i = 0; i < n; i++)
        arr[i] = output[i];
}

总结

在实际应用中,我们需要考虑算法的时间复杂度、空间复杂度和稳定性这三个因素。每一种排序都有其独特的应用场景和相应的优缺点。选择合适的排序算法对于提升程序运行效率是相当必要的。

本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:必须知道的C语言八大排序算法(收藏) - Python技术站

(0)
上一篇 2023年5月19日
下一篇 2023年5月19日

相关文章

  • 详解高性能缓存Caffeine原理及实战

    详解高性能缓存Caffeine原理及实战 简介 Caffeine是一个基于Java的高性能缓存库,其目标是提供比ConcurrentHashMap更高效、更灵活的缓存方案。Caffeine支持多种缓存策略、过期机制以及可自定义的缓存加载策略等功能。本文将详细介绍Caffeine的原理、使用方法及实现实例。 Caffeine的原理 Caffeine的核心是一个…

    算法与数据结构 2023年5月19日
    00
  • Javascript排序算法之合并排序(归并排序)的2个例子

    下面我将详细讲解“Javascript排序算法之合并排序(归并排序)的2个例子”的完整攻略。该攻略包含以下内容: 合并排序算法的原理介绍 归并排序实现流程 两个例子的具体实现及演示 合并排序算法的原理介绍 合并排序是一种基于分治思想的排序算法。它的基本思路是将待排序序列分成若干个子序列,对每个子序列递归地进行排序,最后合并所有子序列,得到最终的排序结果。 具…

    算法与数据结构 2023年5月19日
    00
  • c++归并排序详解

    C++归并排序详解 归并排序是一种基于分治思想的高效排序算法,它的时间复杂度为O(nlogn),并且它的稳定性使得它在实际应用中得到了广泛的应用。在本文中,我们将为大家详细讲解C++归并排序的具体实现过程和算法思想。 算法原理 归并排序基于分治算法,首先将待排序序列不断二分,直到每个子序列只剩一个元素,然后将相邻的子序列进行归并,合并后的子序列再次进行归并,…

    算法与数据结构 2023年5月19日
    00
  • Java实现单向链表的基本功能详解

    Java实现单向链表的基本功能详解 单向链表是一种常见的数据结构,它由一系列节点组成,每个节点包含存储数据的元素和一个指向下一个节点的指针。Java语言可以很方便地实现单向链表,本文将详细介绍Java实现单向链表的基本功能。 一、定义链表节点类 链表的基本单元是节点,我们需要定义一个节点类来描述它。节点类需要包含两个部分:存储数据的元素和指向下一个节点的指针…

    算法与数据结构 2023年5月19日
    00
  • Javascript实现快速排序(Quicksort)的算法详解

    Javascript实现快速排序的算法详解 在这个攻略中,我们将通过Javascript实现快速排序算法,并讲解算法的详细过程。 快速排序的基本思想 快速排序是一种基于交换的排序算法,其基本思想是通过选择一个基准元素,在一趟排序过程中,将之前需要排序的序列中的元素分割成两个部分,其中,左边部分元素的值都小于基准元素的值,右边部分元素的值都大于基准元素的值,然…

    算法与数据结构 2023年5月19日
    00
  • C++实现双向冒泡排序算法

    C++实现双向冒泡排序算法 算法介绍 双向冒泡排序,也称为鸡尾酒排序或定向冒泡排序,是冒泡排序的改进版本。其基本思路与冒泡排序相同,不同之处在于每次排序时同时从数组两侧开始,分别向中间移动。这种方法能够更快地将大数和小数分别冒泡到数组的两端,从而减少了排序次数,提高了排序效率。 下面是双向冒泡排序的具体步骤:1. 从左往右进行一轮冒泡排序,将最小的数排到数组…

    算法与数据结构 2023年5月19日
    00
  • 归并排序时间复杂度过程推导详解

    归并排序时间复杂度过程推导详解 什么是归并排序 归并排序是一种基于分治思想的排序算法,将一个无序的数组划分成若干子数组,对每个子数组进行排序,然后再将排好序的子数组进行合并,最终得到一个完整有序的数组。 归并排序的时间复杂度 归并排序的时间复杂度是O(nlogn),其中n表示数组的长度。接下来我们将详细讲解归并排序的时间复杂度推导过程。 假设有一个长度为n的…

    算法与数据结构 2023年5月19日
    00
  • c语言实现冒泡排序、希尔排序等多种算法示例

    当涉及到算法时,实现该算法的语言是一个非常重要的话题。为了帮助初学者理解和重视这一问题,我们提供了“c语言实现冒泡排序、希尔排序等多种算法示例”的完整攻略。 什么是排序算法? 首先,让我们讨论一下排序算法的基本概念。在计算机科学中,排序是一种重要的算法,其目的是将一组数据按照特定的顺序排列。常见的排序算法有冒泡排序、希尔排序、快速排序等。 冒泡排序和希尔排序…

    算法与数据结构 2023年5月19日
    00
合作推广
合作推广
分享本页
返回顶部