排序 | 冒泡 插入 希尔 选择 堆 快排 归并 非递归 计数 基数 排序

news2024/12/23 5:55:10

排序 | 冒泡 插入 希尔 选择 堆 快排 归并 非递归 计数 基数 排序

文章目录

  • 排序 | 冒泡 插入 希尔 选择 堆 快排 归并 非递归 计数 基数 排序
  • 前言:
    • 冒泡排序
    • 插入排序
    • 希尔排序
    • 选择排序
    • 堆排序
    • 快速排序--交换排序
      • 三数取中
      • 快速排序hoare版本
      • 快速排序挖坑法
      • 快速排序前后指针法
    • 快速排序--非递归实现
    • 归并排序
    • 归并排序非递归实现
    • 非比较排序【计数排序】
    • 基数排序
    • 排序算法复杂度及稳定性分析

前言:

排序算法是一种将一组数据按照特定顺序排列的算法。数据结构排序算法的选择取决于数据的特征、规模和性能需求。
接下来我们就要实现排序~~


我们需要实现的一些功能:

#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<stdbool.h>
#include<time.h>
// 打印
void Print_a(int* a, int sz);
// 交换
void Swap(int* p1, int* p2);
// 插入排序
void InsertSort(int* a, int n);
// 冒泡排序
void BubbleSort(int* a, int n);
// 希尔排序
void ShellSort(int* a, int n);
// 堆排序
void AdjustDwon(int* a, int n, int root);
void HeapSort(int* a, int n);
// 选择排序
void SelectSort(int* a, int n);
// ------------------------------------------------------
// 快速排序hoare版本 
int PartSort1(int* a, int begin, int end);
// 快速排序挖坑法
int PartSort2(int* a, int begin, int end);
// 快速排序前后指针法 
int PartSort3(int* a, int begin, int end);
// 排序函数
void QuickSort(int* a, int begin, int end);
//------------------------------------------------------
// 快速排序 非递归实现 
void QuickSortNonR(int* a, int begin, int end);
// 归并排序递归实现 
void MergeSort(int* a, int n);
// 归并排序非递归实现 
void MergeSortNonR(int* a, int n);
// 非比较排序
void CountSort(int* a, int n);

冒泡排序

  • 冒泡排序是一种基本的排序算法,其核心思想是通过多次交换相邻元素的位置,使得每一轮循环都将最大(或最小)的元素移动到序列的最后。这个过程就像气泡逐渐上升到表面一样,因而得名"冒泡排序"。

在这里插入图片描述

在这里插入图片描述
代码实现:

void bubbleSort(int* a,int n)
{
	for (int i = 0; i < n-1; i++)
	{
		for (int j = 0; j < n - 1 - i; j++)
		{
			if (a[j] > a[j + 1])
			{
				Swap(&a[j], &a[j + 1]);
			}
		}
	}
}

优点:

  1. 简单易懂: 冒泡排序的思想非常简单,容易理解和实现,适合初学者学习排序算法的基本概念。
  2. 原地排序: 冒泡排序是一种原地排序算法,不需要额外的空间来存储临时数据,只需要一个常数级的辅助空间。
  3. 稳定性: 冒泡排序是一种稳定的排序算法,相等元素的相对位置不会发生变化。

缺点:

  1. 效率低: 冒泡排序的平均时间复杂度为O(n^2),在处理大规模数据时性能较差,比较和交换的操作太过频繁。
  2. 不适合大规模数据: 冒泡排序的性能不如一些更高效的排序算法,如快速排序、归并排序等,特别是在数据规模较大的情况下。
  3. 对基本有序的序列效率低下: 在实际应用中,如果序列已经基本有序,冒泡排序仍然需要进行多次比较和交换,效率不高。

插入排序

  • 插入排序是一种简单直观的排序算法,其核心思想是将一个元素插入到已经排好序的数组(或子数组)中的合适位置,以达到整体有序的效果。插入排序的工作方式类似于整理扑克牌的过程:手里的牌是已经有序的部分,新摸到的牌则需要插入到适当的位置,保持有序性。

在这里插入图片描述

在这里插入图片描述
代码实现:

void InsertSort(int* a, int n)
{
	for (int i = 0; i < n - 1; i++)
	{
		// 记录每次i的位置
		int end = i;
		// 将i+1的位置保存
		int tmp = a[end + 1];
		// 一次排序
		while (end >= 0)
		{
			// 如果后面的数字小于前面的那一个就进行往后覆盖,
			// 然后end--,继续排序
			if (tmp < a[end])
			{
				a[end + 1] = a[end];
				--end;
			}
			else
			{
				// 如果大于或者等于了就跳出
				break;
			}
		}
		// 因为--end了,所以就要在end+1的位置放刚刚保存的值tmp
		a[end + 1] = tmp;
	}
}

优点:

  1. 简单易理解: 插入排序的实现非常简单,易于理解和实现,适用于小规模数据或部分有序的数据。
  2. 稳定性: 插入排序是一种稳定的排序算法,相等元素的相对位置不会发生变化。
  3. 适应性好: 如果数据已经基本有序,插入排序的性能会比较好,因为大部分元素都已经在正确的位置上,只需少量的比较和移动。
  4. 原地排序: 插入排序是一种原地排序算法,不需要额外的空间来存储临时数据。

缺点:

  1. 效率低: 插入排序的平均时间复杂度为O(n^2),在数据规模较大时,性能不如一些更高效的排序算法,如快速排序、归并排序等。
  2. 对大规模数据排序效率低下: 插入排序在处理大规模数据时性能较差,因为它需要大量的比较和移动操作。
  3. 不适合链表结构: 插入排序需要频繁地移动元素,对于链表结构来说,由于不支持随机访问,插入排序效率较低。

希尔排序

  • 希尔排序(Shell Sort)是一种插入排序的改进版本,也称为缩小增量排序。它通过比较相距一定间隔的元素,逐步减小这个间隔,直到间隔为1时完成最后一轮排序。希尔排序的核心思想是先使数组中任意间隔为h的元素有序,然后逐步减小h,最终使整个数组有序。

代码实现:

void ShellSort(int* a, int n)
{
	int gap = n;
	
	// gap > 1时是预排序,目的让他接近有序
	// gap == 1是直接插入排序,目的是让他有序
	while (gap > 1)
	{
		// gap = gap / 2; // log 2 N
		gap = gap / 3 + 1; // log 3 N

		//每次排gap次
		for (int j = 0; j < n - gap; j++)
		{
			//插入排序
			int end = j;
			int tmp = a[end + gap];
			while (end >= 0)
			{
				if (tmp < a[end])
				{
					a[end + gap] = a[end];
					end -= gap;
				}
				else
				{
					break;
				}
			}
			a[end + gap] = tmp;
		}
	}
}

优点:

  1. 相对于插入排序的改进: 希尔排序是插入排序的改进版本,通过引入间隔(gap)的概念,减少了数据的搬移次数,提高了效率。
  2. 适用于中等规模数据: 希尔排序相对于一些简单的排序算法,对中等规模的数据表现较好,比如对于几千甚至几万个元素的排序。
  3. 不稳定性: 相比较于一些稳定排序算法(如归并排序、插入排序),希尔排序的不稳定性可能在某些情况下是一个优势,特别是在排序过程中需要对元素进行位置交换的场景。

缺点:

  1. 不适用于大规模数据: 希尔排序的性能相对较好,但在处理非常大规模数据时,效率可能不如一些更为高级的排序算法,比如快速排序、归并排序等。
  2. 不稳定性: 尽管不稳定性在某些情况下可以是优势,但在某些应用场景下,需要保持相等元素的相对位置不变,这时候希尔排序的不稳定性可能成为一个缺点。

选择排序

  • 选择排序(Selection Sort)是一种简单直观的排序算法,其基本思想是通过不断选择未排序序列中的最小(或最大)元素,将其与未排序序列的第一个元素交换,从而逐步构建有序序列。

在这里插入图片描述

代码实现:

// 选择排序
void SelectSort(int* a, int n)
{
	int begin = 0, end = n - 1;
	while (begin < end)
	{
		// 两头找
		int mini = begin, maxi = begin;
		for (int i = begin + 1; i <= end; i++)
		{
			// 如果i的位置比mini小就更新一下
			if (a[i] < a[mini])
			{
				mini = i;
			}
			//如果i的位置比maxi大就更新一下
			if (a[i] > a[maxi])
			{
				maxi = i;
			}
		}
		// 走到这里就说明小的要和左边交换一下
		Swap(&a[mini], &a[begin]);
		// 注意:这里Eugene左边的和maxi相等了要更新一下maxi
		if (begin == maxi)
		{
			maxi = mini;
		}
		// 然后交换maxi和end
		Swap(&a[end], &a[maxi]);
		++begin;
		--end;
	}
}

优点:

  1. 简单易理解: 选择排序的实现非常简单,易于理解和实现,适合初学者学习排序算法的基本概念。
  2. 原地排序: 选择排序是一种原地排序算法,不需要额外的空间来存储临时数据,只需要一个常数级的辅助空间。
  3. 不稳定性: 选择排序是一种不稳定的排序算法,相等元素的相对位置可能发生变化,但在某些情况下,不稳定性可能是一个优势。

缺点:

  1. 效率低: 选择排序的平均时间复杂度为O(n^2),在处理大规模数据时性能较差,由于每次只能确定一个元素的位置,比较和交换的操作过于频繁。
  2. 对基本有序的序列效率低下: 在实际应用中,如果序列已经基本有序,选择排序仍然需要进行大量的比较和交换,效率不高。
  3. 不适合大规模数据: 选择排序的性能不如一些更高效的排序算法,如快速排序、归并排序等,特别是在数据规模较大的情况下。

堆排序

  • 堆排序是一种基于二叉堆数据结构的排序算法。它利用了堆的性质来进行排序,其中堆分为最大堆和最小堆两种类型。在最大堆中,父节点的值大于或等于其子节点的值;在最小堆中,父节点的值小于或等于其子节点的值。

  • 这里在堆排序章节已经讲过了,这里就不细讲了~~

代码实现:

void AdjustDwon(int* a, int n, int root)
{
	int parent = root;
	int child = parent * 2 + 1;
	while (child < n)
	{
		if (child + 1 < n && a[child + 1] > a[child])
			++child;

		if (a[child] > a[parent])
		{
			Swap(&a[child], &a[parent]);

			parent = child;
			child = parent * 2 + 1;
		}
		else
		{
			break;
		}
	}
}

void HeapSort(int* a, int n)
{
	for (int i = (n - 1 - 1) / 2; i >= 0; i--)
	{
		AdjustDwon(a, n, i);
	}

	int end = n - 1;
	while (end > 0)
	{
		Swap(&a[0], &a[end]);
		AdjustDwon(a, end, 0);
		--end;
	}
}

优点:

  1. 稳定性: 堆排序是一种不稳定的排序算法,因为在构建初始堆和进行堆调整的过程中可能破坏相同元素的相对顺序。然而,如果对于相同的元素,可以使用索引来保持其相对顺序,就可以避免这个问题。
  2. 原地排序: 堆排序是一种原地排序算法,不需要额外的存储空间来存储待排序的数据,只需要常数级别的辅助空间。
  3. 时间复杂度: 堆排序的平均、最好和最坏情况下的时间复杂度都是 O(n log n),其中 n 是待排序元素的数量。这使得堆排序在大数据集上表现良好。

缺点:

  1. 非自适应性: 堆排序的时间复杂度在各种情况下都是 O(n log n),无论输入数据的初始顺序如何。因此,它对于部分有序的数据或者小规模数据集的排序效率可能不如一些自适应性较强的算法。
  2. 不稳定: 在排序过程中,堆排序可能破坏相同元素的相对顺序,因此是一种不稳定的排序算法。如果对稳定性有要求,可能需要考虑其他算法。
  3. 不适用于链式存储结构: 堆排序通常需要对数组进行直接访问,而不适用于链式存储结构。因此,如果数据结构采用链表形式存储,需要将其转换为数组再进行排序,这可能引入额外的开销。

快速排序–交换排序

三数取中

int GetMidi(int* a, int left, int right)
{
	//int midi = (begin + end) / 2;
	int mid = (left + right) >> 1;
	if (a[left] < a[mid])
	{
		if (a[mid] < a[right])
			return mid;
		else if (a[left] > a[right])
			return left;
		else
			return right;
	}
	else // a[left] > a[mid]
	{
		if (a[mid] > a[right])
			return mid;
		else if (a[left] < a[right])
			return left;
		else
			return right;
	}
}

快速排序hoare版本

  • 快速排序是一种常用的排序算法,Hoare版本是其中一种实现方式,由Tony Hoare提出。

在这里插入图片描述

代码实现:

int PartSort1(int* a, int begin, int end)
{
	// 三数取中
	int midi = GetMidi(a, begin, end);
	Swap(&a[midi], &a[begin]);

	// 要在最左边开始
	int left = begin, right = end;
	int keyi = begin;

	while (left < right)
	{
		// 右边找小
		while (left < right && a[right] >= a[keyi])
		{
			--right;
		}
		// 左边找大
		while (left < right && a[left] <= a[keyi])
		{
			++left;
		}

		// 找到了,就交换
		Swap(&a[left], &a[right]);
	}

	// 交换的左边的和keyi,然后更新一下keyi
	Swap(&a[left], &a[keyi]);
	return left;
}

void QuickSort(int* a, int begin, int end)
{
	if (begin >= end)
		return;

	// 小区间
	if (end - begin + 1 < 10)
	{
		InsertSort(a + begin, end - begin + 1);
	}
	else
	{
		int keyi = PartSort1(a, begin, end);

		QuickSort(a, begin, keyi - 1);
		QuickSort(a, keyi + 1, end);
	}
}

优点:

  1. 原地排序: 快速排序是一种原地排序算法,不需要额外的空间来存储临时数据,只需要一个常数级的辅助空间。
  2. 平均情况下具有较好的性能: 在平均情况下,快速排序的时间复杂度为O(n log n),这使得它在实践中具有较好的性能。
  3. 适用于大规模数据: 快速排序在处理大规模数据时通常表现良好,尤其是相对于一些平均时间复杂度较高的排序算法而言。
  4. 对基本有序的数据排序效果好: 在一些情况下,快速排序对基本有序的数据排序效果较好。

缺点:

  1. 不稳定性: 快速排序是一种不稳定的排序算法,相等元素的相对位置可能发生变化,如果需要稳定性,可能需要额外的处理。
  2. 对于极端情况的性能: 在最坏情况下,即已经有序的序列,快速排序的时间复杂度为O(n^2),这时性能可能较差。为了避免这种情况,通常会使用一些优化策略,比如随机化选择枢轴。
  3. 对于小规模数据性能较差: 在小规模数据的排序中,快速排序的递归调用会增加额外的开销,性能可能不如一些简单的排序算法,如插入排序。

快速排序挖坑法

  • 快速排序的挖坑法(也称为Lomuto分区方案)是快速排序的一种实现方式。在挖坑法中,选择一个基准元素,通过不断交换元素将数组分成两个部分,左边的部分都小于基准元素,右边的部分都大于基准元素。接着,对左右两个部分分别递归进行同样的操作。

在这里插入图片描述

代码实现:

int PartSort2(int* a, int begin, int end)
{
	int midi = GetMidi(a, begin, end);
	Swap(&a[midi], &a[begin]);

	int key = a[begin];
	int holei = begin;

	while (begin < end)
	{
		// 右边找小
		while (begin < end && a[end] >= key)
		{
			--end;
		}

		a[holei] = a[end];
		holei = end;

		// 左边找大
		while (begin < end && a[begin] <= key)
		{
			++begin;
		}
		a[holei] = a[begin];
		holei = begin;
	}

	a[holei] = key;
	return holei;
}

void QuickSort(int* a, int begin, int end)
{
	if (begin >= end)
		return;

	// 小区间
	if (end - begin + 1 < 10)
	{
		InsertSort(a + begin, end - begin + 1);
	}
	else
	{
		int keyi = PartSort2(a, begin, end);

		QuickSort(a, begin, keyi - 1);
		QuickSort(a, keyi + 1, end);
	}
}

优点:

  1. 原地排序: 挖坑法是一种原地排序算法,不需要额外的空间来存储临时数据,只需要一个常数级的辅助空间。
  2. 简单直观: 挖坑法实现相对较简单,容易理解和实现,适用于教学和学习排序算法。

缺点:

  1. 不稳定性: 挖坑法是一种不稳定的排序算法,相等元素的相对位置可能发生变化,如果需要稳定性,可能需要额外的处理。
  2. 最坏情况下的性能: 在最坏情况下,即已经有序的序列,挖坑法的性能可能较差。这时的时间复杂度为O(n^2),因为每次分区只能使序列中的一个元素有序。
  3. 对于小规模数据性能较差: 在小规模数据的排序中,挖坑法的递归调用会增加额外的开销,性能可能不如一些简单的排序算法,如插入排序。

快速排序前后指针法

  • 快速排序的前后指针法(也称为Hoare分区方案)是另一种实现方式。在这个方法中,通过两个指针从数组的两端分别向中间移动,交换不符合排序条件的元素,最终将数组分为两个部分,左边部分小于基准元素,右边部分大于基准元素。

在这里插入图片描述

代码实现:

int PartSort3(int* a, int begin, int end)
{
	int midi = GetMidi(a, begin, end);
	Swap(&a[begin], &a[midi]);

	int prev = begin;
	int cur = prev + 1;
	int keyi = begin;
	while (cur <= end)
	{
		if (a[cur] < a[keyi] && ++prev != cur)
			Swap(&a[prev], &a[cur]);

		++cur;
	}

	Swap(&a[keyi], &a[prev]);
	keyi = prev;
	return prev;
}

void QuickSort(int* a, int begin, int end)
{
	if (begin >= end)
		return;

	// 小区间
	if (end - begin + 1 < 10)
	{
		InsertSort(a + begin, end - begin + 1);
	}
	else
	{
		int keyi = PartSort3(a, begin, end);

		QuickSort(a, begin, keyi - 1);
		QuickSort(a, keyi + 1, end);
	}
}

优点:

  1. 原地排序: 前后指针法是一种原地排序算法,不需要额外的空间来存储临时数据,只需要一个常数级的辅助空间。
  2. 相对较好的性能: 在平均情况下,快速排序的前后指针法具有较好的性能,时间复杂度为O(n log n)。
  3. 不需要额外的空间: 与挖坑法相比,前后指针法在实际的操作中,不需要额外的元素用于填坑,从而减少了一些操作。

缺点:

  1. 不稳定性: 前后指针法是一种不稳定的排序算法,相等元素的相对位置可能发生变化,如果需要稳定性,可能需要额外的处理。
  2. 最坏情况下的性能: 在最坏情况下,即已经有序的序列,前后指针法的性能可能较差。这时的时间复杂度为O(n^2),因为每次分区只能使序列中的一个元素有序。
  3. 对于小规模数据性能较差: 在小规模数据的排序中,前后指针法的递归调用会增加额外的开销,性能可能不如一些简单的排序算法,如插入排序。

快速排序–非递归实现

  • 我们这里使用栈来解决这个问题~~
  • 先入栈,然后再进行分割
  • 注意: 如果是先入右后入左,那么出的时候就要先出左后出右
  • 栈不为空就继续,然后分割排左边和右边

代码实现:

#include"Stack.h"
// 快速排序 非递归实现 
void QuickSortNonR(int* a, int begin, int end)
{
	ST s;
	StackInit(&s);
	// 先入右后入左
	StackPush(&s, end);
	StackPush(&s, begin);

	while (!StackEmpty(&s))
	{
		// 先出左后出右
		int left = StackTop(&s);
		StackPop(&s);
		int right = StackTop(&s);
		StackPop(&s);

		// 排序
		int keyi = PartSort3(a, left, right);
		// [left keyi-1] keyi [keyi+1 right]
		if (left < keyi - 1)
		{
			StackPush(&s, keyi - 1);
			StackPush(&s, left);
		}
		if (keyi + 1 < right)
		{
			StackPush(&s, right);
			StackPush(&s, keyi + 1);
		}
	}

	StackDestroy(&s);
}

归并排序

  • 归并排序(Merge Sort)是一种分治算法,它的基本思想是将待排序的数组分成两个相等大小的子数组,然后分别对这两个子数组进行排序,最后将排序好的子数组合并成一个有序的数组。

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

代码实现:

void _MergeSort(int* a, int begin, int end, int* tmp)
{
	if (begin >= end)
		return;

	// 分割  // 这里右移一位相当于 /2 
	int mid = (begin + end) >> 1; 
	
	// 递归
	_MergeSort(a, begin, mid, tmp);
	_MergeSort(a, mid + 1, end, tmp);

	// [begin mid][mid+1 end]  归并
	int begin1 = begin, end1 = mid;
	int begin2 = mid + 1, end2 = end;
	int i = begin;

	while (begin1 <= end1 && begin2 <= end2)
	{
		if (a[begin1] < a[begin2])
		{
			tmp[i++] = a[begin1++];
		}
		else
		{
			tmp[i++] = a[begin2++];
		}
	}

	while (begin1 <= end1)
	{
		tmp[i++] = a[begin1++];
	}
	while (begin2 <= end2)
	{
		tmp[i++] = a[begin2++];
	}

	// 拷贝回原数组

	for (int i = begin; i <= end; ++i)
	{
		a[i] = tmp[i];
	}

	//memcpy(a + begin, tmp + begin, sizeof(int) * (end - begin) + 1);
}

// 归并排序递归实现 
void MergeSort(int* a, int n)
{
	int* tmp = (int*)malloc(sizeof(int) * n);
	if (tmp == NULL)
	{
		perror("malloc fail!\n");
		return;
	}

	_MergeSort(a, 0, n - 1, tmp);

	free(tmp);
}

优点:

  1. 稳定性: 归并排序是一种稳定的排序算法,即对于相等的元素,它们在排序后的相对位置保持不变。
  2. 适用于链表: 归并排序对于链表等非随机访问结构的数据也非常有效,因为它不涉及随机访问,只涉及指针操作。
  3. 适用于外部排序: 归并排序在外部排序(需要对外部存储进行排序的情况)中表现良好,因为它可以很容易地通过合并有序的外部文件来实现。
  4. 稳定的时间复杂度: 归并排序的时间复杂度是稳定的,不受输入数据的影响,总是O(n log n)。

缺点:

  1. 额外空间需求: 归并排序需要额外的内存空间来存储临时数组,这使得它的空间复杂度相对较高,是O(n)。
  2. 非原地排序: 归并排序是一种非原地排序算法,即它需要额外的空间来存储临时数组,而不是在原始数组上进行排序。这对于内存受限的情况可能不太理想。
  3. 常数因子较大: 归并排序的常数因子较大,因此在实际应用中可能被一些其他排序算法(如快速排序)超越。

归并排序非递归实现

  • 思想和上面的归并排序差不多~~

代码实现:

void MergeSortNonR(int* a, int n)
{
	int* tmp = (int*)malloc(sizeof(int) * n);

	int gap = 1; // 每组数据个数
	while (gap < n)
	{
		for (int i = 0; i < n; i += 2 * gap)
		{
			// [i, i+gap-1] [i+gap,i+2*gap-1]
			int begin1 = i, end1 = i + gap - 1;
			int begin2 = i + gap, end2 = i + 2 * gap - 1;

			// 归并过程中右半区间可能就不存在
			if (begin2 >= n)
				break;

			// 归并过程中右半区间算多了, 修正一下
			if (end2 >= n)
			{
				end2 = n - 1;
			}

			int index = i;
			while (begin1 <= end1 && begin2 <= end2)
			{
				if (a[begin1] < a[begin2])
				{
					tmp[index++] = a[begin1++];
				}
				else
				{
					tmp[index++] = a[begin2++];
				}
			}

			while (begin1 <= end1)
			{
				tmp[index++] = a[begin1++];
			}

			while (begin2 <= end2)
			{
				tmp[index++] = a[begin2++];
			}

			// 拷贝回去
			for (int j = i; j <= end2; ++j)
			{
				a[j] = tmp[j];
			}
		}
		gap *= 2;
	}

	free(tmp);
}

非比较排序【计数排序】

  • 计数排序(Counting Sort)是一种非比较性的整数排序算法,它通过确定每个元素在输出序列中的位置来实现排序。

在这里插入图片描述

代码实现:

void CountSort(int* a, int n)
{
	int max = a[0], min = a[0];
	for (int i = 0; i < n; i++)
	{
		if (a[i] > max)
			max = a[i];
		if (a[i] < min)
			min = a[i];
	}
	int range = max - min + 1;

	int* count = (int*)malloc(sizeof(int) * range);
	if (count == NULL)
	{
		perror("malloc fail!\n");
		return;
	}

	memset(count, 0, sizeof(int) * range);
	//统计次数
	for (int i = 0; i < n; i++)
	{
		count[a[i] - min]++;
	}

	int i = 0;
	for (int j = 0; j < range; j++)
	{
		while (count[j]--)
		{
			a[i++] = j + min;
		}
	}

	free(count);
}

优点:

  1. 线性时间复杂度: 计数排序是一种具有线性时间复杂度的排序算法,其时间复杂度为O(n + k),其中n是输入元素的数量,k是输入范围的大小。在输入范围较小的情况下,计数排序通常比其他O(n log n)的排序算法更快。
  2. 稳定性: 计数排序是一种稳定的排序算法,即相等元素的相对顺序在排序后仍然保持不变。
  3. 适用于整数排序: 计数排序适用于整数排序,尤其是在知道输入范围不太大的情况下。它不依赖于比较操作,因此在某些情况下可能比基于比较的排序算法更高效。

缺点:

  1. 空间复杂度: 计数排序的主要缺点是它需要额外的空间来存储计数数组。如果输入范围很大,可能需要较大的额外空间,这可能导致空间复杂度较高。
  2. 仅适用于整数: 计数排序仅适用于整数排序,因为它依赖于将元素映射到计数数组的索引。对于浮点数或其他数据类型,需要进行额外的转换。
  3. 对输入范围的限制: 计数排序要求输入的元素必须在已知范围内,否则需要进行范围的确定和调整,增加了实现的复杂性。

基数排序

  • 基数排序是一种非比较性的排序算法,它根据关键字的每一位来排序数据。

在这里插入图片描述

代码实现:

这里就先不实现,之后会开一个新文章来进行阐述,并且使用C++来写~~

优点:

  1. 稳定性: 基数排序是一种稳定的排序算法,即相等元素的相对顺序在排序后保持不变。
  2. 适用范围广: 基数排序对于数据的分布没有特殊的要求,适用于各种数据类型,包括整数、字符串等。
  3. 适用于大量数据: 在某些情况下,基数排序的性能可能比一些常见的比较性排序算法(如快速排序、归并排序)更好,尤其是当数据量非常大时。
  4. 不受输入数据范围限制: 基数排序不受输入数据范围的限制,可以处理负数和小数。

缺点:

  1. 空间复杂度较高: 基数排序的空间复杂度取决于数据的位数,如果数据位数很大,可能需要较大的辅助空间来存储中间结果。
  2. 不适用于小范围数据: 当数据范围比较小而位数较大时,基数排序可能不是最优选择,因为它需要较大的辅助空间。
  3. 只能处理正整数或字符串: 基数排序主要用于整数或字符串的排序,对于其他数据类型可能需要转换为整数或字符串形式,增加了额外的开销。
  4. 效率受制于位数: 基数排序的效率受制于位数,如果位数很大,可能需要进行多轮排序,导致时间复杂度较高。

排序算法复杂度及稳定性分析

在这里插入图片描述

在这里插入图片描述

本期内容就到这里了,感谢大家的收看,欢迎三连~~

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1320409.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Git总结 | Git面试都问些啥?

什么是Git为什么要用Git等等这些相信看到该标题点进来的同学也不希望浪费时间再看一遍&#xff0c;那么直接进入主题&#xff0c;对于日常工作中常用的Git相关操作进行整理&#xff0c;一起看看吧 面试官&#xff1a;你常用的Git操作是什么? 候选人&#xff1a;git clone 面试…

Java序列化、反序列化-为什么要使用序列化?Serializable接口的作用?

什么是序列化和反序列化&#xff1f; 把对象转换成字节序列把字节序列恢复成对象 结合OSI七层协议模型&#xff0c;序列化和反序列化是在那一层做的&#xff1f; 在OSI七层模型中&#xff0c;序列化工作的层级是表示层。这一层的主要功能包括把应用层的对象转换成一段连续的二进…

5.5 DataFrame.rolling()创建滚动窗口对象

DataFrame.rolling创建滚动窗口对象 一、介绍二、代码一、介绍 DataFrame.rolling() 是 pandas 中用于创建滚动窗口对象的函数,它可以对时间序列或其他类型的数据进行滚动计算。下面是该函数的一些参数说明: DataFrame.rolling(window, min_periods=None, center=False, win_…

Flink系列之:自定义函数

Flink系列之&#xff1a;自定义函数 一、自定义函数二、概述三、开发指南四、函数类五、求值方法六、类型推导七、自动类型推导八、定制类型推导九、确定性十、内置函数的确定性十一、运行时集成十二、标量函数十三、表值函数十四、聚合函数十五、表值聚合函数 一、自定义函数 …

Windows使用VNC Viewer远程桌面Ubuntu【内网穿透】

文章目录 前言1. ubuntu安装VNC2. 设置vnc开机启动3. windows 安装VNC viewer连接工具4. 内网穿透4.1 安装cpolar【支持使用一键脚本命令安装】4.2 创建隧道映射4.3 测试公网远程访问 5. 配置固定TCP地址5.1 保留一个固定的公网TCP端口地址5.2 配置固定公网TCP端口地址5.3 测试…

微信小程序背景图片设置

问题 :微信小程序通过css:background-image引入背景图片失败 [渲染层网络层错误] pages/wode/wode.wxss 中的本地资源图片无法通过 WXSS 获取&#xff0c;可以使用网络图片&#xff0c;或者 base64&#xff0c;或者使用<image/>标签 解决方法微信小程序在使用backgroun…

每日一题:LeetCode-LCR 016. 无重复字符的最长子串

每日一题系列&#xff08;day 15&#xff09; 前言&#xff1a; &#x1f308; &#x1f308; &#x1f308; &#x1f308; &#x1f308; &#x1f308; &#x1f308; &#x1f308; &#x1f308; &#x1f308; &#x1f308; &#x1f308; &#x1f308; &#x1f50e…

[Kubernetes]3. k8s集群Service详解

在上一节讲解了k8s 的pod,deployment,以及借助pod,deployment来部署项目,但会存在问题: 每次只能访问一个 pod,没有负载均衡自动转发到不同 pod访问还需要端口转发Pod重创后IP变了,名字也变了针对上面的问题,可以借助Service来解决,下面就来看看Service怎么使用 一.Service详…

SpringBoot中使用@Async实现异步调用

SpringBoot中使用Async实现异步调用 什么是异步调用?异步调用对应的是同步调用&#xff0c;同步调用指程序按照定义顺序依次执行&#xff0c;每一行程序都必须等待上 一行程序执行完成之后才能执行&#xff1b;异步调用指程序在顺序执行时&#xff0c;不等待异步调用的语句返…

Python 实现:OCR在图片中提取文字(基于Gradio实现)

Paddle OCR PaddleOCR 基于深度学习技术实现的&#xff0c;使用十分简单。 先看效果 可以看出来识别效果还是不错的&#xff0c;里面的“湿”字识别成了繁体字。如果不是连体字&#xff0c;就不会出现这个问题。 1.测试环境 操作系统&#xff1a;Win10 Python&#xff1a;3…

ROS机器人入门

http://www.autolabor.com.cn/book/ROSTutorials/ 1、ROS简介 ROS 是一个适用于机器人的开源的元操作系统。其实它并不是一个真正的操作系统&#xff0c;其 底层的任务调度、编译、寻址等任务还是由 Linux 操作系统完成&#xff0c;也就是说 ROS 实际上是运 行在 Linux 上的次级…

xv6 文件系统(下)

〇、前言 计算机崩溃后如何恢复&#xff0c;是一个很重要的话题。对于内存中的数据无关痛痒&#xff0c;开机后重新载入就能解决问题&#xff1b;但是对于持久化存储设备&#xff0c;当你尝试修改一个文件&#xff0c;突然断电当你重新打开文件后&#xff0c;这个文件的状态是…

Java基础回顾——面向对象编程

文章目录 面向对象基础方法构造方法默认构造方法多构造方法 方法重载继承多态抽象类接口静态字段和静态方法包作用域内部类 写在最后 https://www.liaoxuefeng.com/wiki/1252599548343744/1255943520012800 面向对象编程Object-Oriented Programming&#xff0c;简称OOP&#…

06. Python模块

目录 1、前言 2、什么是模块 3、Python标准库模块 3.1、os模块 3.2、datetime 模块 3.3、random模块 4、自定义模块 4.1、创建和使用 4.2、模块命名空间 4.3、作用域 5、安装第三方依赖 5.1、使用 pip 安装单个依赖 5.2、从 requirements.txt 安装依赖 5.3、安装指…

Python:(Sentinel-1)如何解析SNAP输出的HDF5文件并输出为GeoTIFF?

博客已同步微信公众号&#xff1a;GIS茄子&#xff1b;若博客出现纰漏或有更多问题交流欢迎关注GIS茄子&#xff0c;或者邮箱联系(推荐-见主页). Python&#xff1a;&#xff08;Sentinel-1&#xff09;如何解析SNAP输出的HDF5文件并输出为GeoTIFF&#xff1f; 01 前言 最近…

【NI-RIO入门】使用LabVIEW进行数据采集测量

于ni kb摘录 选择合适的编程模式 CompactRIO系统具有至少两个用户可选模式。某些CompactRIO型号具有附加的用户可选模式&#xff0c;可以在实时NI-DAQmx中进行编程。请参考本文以判断您的CompactRIO是否能够使用实时NI-DAQmx。将目标添加到项目后&#xff0c;将提示您选择要使…

TestSSLServer4.exe工具使用方法简单介绍(查SSL的加密版本SSL3或是TLS1.2)

一、工具使用方法介绍 工具使用方法参照&#xff1a;http://www.bolet.org/TestSSLServer/ 全篇英文看不懂&#xff0c;翻译了下&#xff0c;能用到的简单介绍如下&#xff1a; 将下载的TestSSLServer4.exe工具放到桌面上&#xff0c;CMD命令行进入到桌面目录&#xff0c;执…

Gitee基础知识

目录 1-gitee 1.1gitee介绍 1.2git与gitee的关系 1.3在国内为什么选择Gitee 2-注册与创建远程仓库 2.1注册 2.2创建远程仓库 2.3配置ssh公钥 2.3.1公钥的生成方法&#xff1a; 2.3.2 在gitee中配置公钥 2.3.4验证公钥 3-添加与推送远程仓库master 3.1基本命令…

78-C语言-完数的判断,以及输出其因子

简介&#xff1a;一个数如果恰好等于它的因子之和&#xff0c;这个数就称为完数&#xff0c;C语言编程找出1000之内的所有完数&#xff0c;并输出其因子。因子可以整除该数字的数&#xff0c; 如6的因子&#xff1a;1 2 3&#xff0c;6%10 6%20 6%30 解释全在注…

20 5G中高速列车通信:设计相关元素以减轻高移动性带来的影响

文章目录 一 、物理层设计1 DMRS2 CSI 和SRS3 PTRS4 多天线配置 二 初始接入三 目前面临困难 解决问题&#xff1a;列车高速移动&#xff0c;会使信道相干时间较短、多普勒频移和多普勒扩展较大等问题。为了在列车高速移动中解决这些问题&#xff0c;这篇文章概括了5G关键技术&…