数据结构——二叉树——堆以及堆排序(顺序实现和递归实现)

5 篇文章 0 订阅
1 篇文章 0 订阅
 

#include <stdio.h>

#include <stdlib.h>

#include <stdbool.h>

#define TYPE int

#define SWAP(a,b) {typeof(a) t=(a);(a)=(b);(b)=t;}

// 大顶堆

typedef struct Heap

{

TYPE* arr;

size_t cal;

size_t cnt;

}Heap;

Heap* create_heap(size_t cal)

{

Heap* heap = malloc(sizeof(Heap));

heap->arr = malloc(sizeof(TYPE)*cal);

heap->cal = cal;

heap->cnt = 0;

return heap;

}

// 满堆

bool full_heap(Heap* heap)

{

return heap->cnt >= heap->cal;

}

// 空堆

bool empty_heap(Heap* heap)

{

return 0 == heap->cnt;

}

// 添加

bool add_heap(Heap* heap,TYPE data)

{

if(full_heap(heap)) return false;

heap->arr[heap->cnt++] = data;

// 添加位置进行调整重新形成堆

int i = heap->cnt; //编号

while(i > 1)

{

if(data > heap->arr[i/2-1])

{

SWAP(heap->arr[i-1],heap->arr[i/2-1]);

i = i/2;

}

else

break;

}

return true;

}

// 删除 删除堆顶

bool del_heap(Heap* heap)

{

if(empty_heap(heap)) return false;

// 交换堆顶与末尾 并删除末尾

SWAP(heap->arr[0],heap->arr[heap->cnt-1]);

heap->cnt--;

// 从上往下调整

int i = 1; //编号

while(i <= heap->cnt)

{

// 有右子树

if(i*2+1 <= heap->cnt)

{

//右子树最大 交换右根

if(heap->arr[i*2] > heap->arr[i*2-1] &&

heap->arr[i*2] > heap->arr[i-1])

{

SWAP(heap->arr[i*2],heap->arr[i-1]);

i = i*2+1;

}

//左子树最大,交换左根

else if(heap->arr[i*2-1] > heap->arr[i-1])

{

SWAP(heap->arr[i*2-1],heap->arr[i-1]);

i = i*2;

}

//根最大

else

break;

}

// 没有右,有左子树

else if(i*2 <= heap->cnt)

{

// 左比根大,交换左根

if(heap->arr[i*2-1] > heap->arr[i-1])

{

SWAP(heap->arr[i*2-1],heap->arr[i-1]);

i = i*2;

}

// 左比根小,结束

else

break;

}

//没有左右

else

break;

}

return true;

}

// 遍历

void show_heap(Heap* heap)

{

for(int i=0; i<heap->cnt; i++)

{

printf("%d ",heap->arr[i]);

}

printf("\n");

}

// 堆顶

TYPE top_heap(Heap* heap)

{

return heap->arr[0];

}

// 堆排序 顺序实现

void sort_heap(int* arr,int len)

{

// 把数组调成堆结构

for(int i=1; i<=len; i++)

{

int j=i;

while(j > 1)

{

if(arr[j-1] > arr[j/2-1])

{

SWAP(arr[j-1],arr[j/2-1]);

j = j/2;

}

else

break;

}

}

// 删除堆顶,直到堆为空

while(len > 1)

{

// 交换堆顶 末尾

SWAP(arr[0],arr[len-1]);

len--;

//从上往下调整

int i = 1;

while(i <= len)

{

if(i*2+1 <= len)

{

if(arr[i*2] > arr[i*2-1]

&& arr[i*2] > arr[i-1])

{

SWAP(arr[i*2],arr[i-1]);

i = i*2+1;

}

else if(arr[i*2-1] > arr[i-1])

{

SWAP(arr[i*2-1],arr[i-1]);

i = i*2;

}

else

break;

}

else if(i*2 <= len)

{

if(arr[i*2-1] > arr[i-1])

{

SWAP(arr[i*2-1],arr[i-1]);

i = i*2;

}

else

break;

}

else

break;

}

}

}

// 从top下标 到end下标 从上往下调整成堆结构

void _sort_heap_recursion(int* arr,int top,int end)

{

if(top >= end) return;

int max = top+1; // max是左右根中最大值的编号

int l = max*2;

int r = max*2+1;

if(l-1 <= end && arr[l-1] > arr[max-1])

{

// 有左子树,且左子树大于max的值,更新max

max = l;

}

if(r-1 <= end && arr[r-1] > arr[max-1])

{

// 有右子树,且右子树大于max的值,更新max

max = r;

}

if(max-1 != top)

{

// max是左右根中最大的,交换根与max

SWAP(arr[top],arr[max-1]);

_sort_heap_recursion(arr,max-1,end);

}

}

// 堆排序 递归实现

void sort_heap_recursion(int* arr,int len)

{

// 把数组调成堆结构

for(int i=2; i<=len; i++)

{

int j=i;

while(j > 1)

{

if(arr[j-1] > arr[j/2-1])

{

SWAP(arr[j-1],arr[j/2-1]);

j = j/2;

}

else

break;

}

}

for(int i=len-1; i>0; i--)

{

SWAP(arr[0],arr[i]);

_sort_heap_recursion(arr,0,i-1);

}

}

void show_arr(int* arr,int len)

{

for(int i=0; i<len; i++)

{

printf("%d ",arr[i]);

}

printf("\n");

}

int main(int argc,const char* argv[])

{

int arr[10] = {};

for(int i=0; i<10; i++)

{

arr[i] = rand()%100;

}

sort_heap_recursion(arr,10);

show_arr(arr,10);

/*

Heap* heap = create_heap(20);

for(int i=0; i<10; i++)

{

add_heap(heap,rand()%100);

}

show_heap(heap);

add_heap(heap,99);

show_heap(heap);

while(!empty_heap(heap))

{

printf("%d \n",top_heap(heap));

del_heap(heap);

}

*/

}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
好的,我可以为您解答这个问题。 首先,堆排序、快速排序、归并排序和计数排序都是常见的排序算法,它们的实现方式各不相同。下面分别介绍它们的C++实现。 1. 堆排序(Heap Sort) 堆排序是一种选择排序,它利用的性质进行排序。是一个完全二叉树,其中每个节点的值都大于或等于其子节点的值(最大),或者小于或等于其子节点的值(最小)。在堆排序中,我们将待排序数组构建成一个最大,然后将顶元素(即最大值)与数组最后一个元素交换,然后重新调整,重复执行此操作,直到整个数组有序。 下面是堆排序C++实现: ```c++ 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); } } ``` 2. 快速排序(Quick Sort) 快速排序是一种分治排序,它利用分治的思想将大问题分解成小问题,然后递归地解决小问题。在快速排序中,我们选择一个枢轴元素(一般选择第一个元素),将数组分为两个部分,左边部分的元素均小于枢轴元素,右边部分的元素均大于枢轴元素。然后分别对左右两部分递归执行此操作,直到整个数组有序。 下面是快速排序的C++实现: ```c++ int partition(int arr[], int low, int high) { int pivot = arr[low]; // 枢轴元素 int i = low, j = high; while (i < j) { // 从右往左找第一个小于枢轴元素的位置 while (i < j && arr[j] >= pivot) j--; // 从左往右找第一个大于枢轴元素的位置 while (i < j && arr[i] <= pivot) i++; // 交换左右两个元素 if (i < j) swap(arr[i], arr[j]); } // 将枢轴元素放到正确的位置上 arr[low] = arr[i]; arr[i] = pivot; return i; } void quickSort(int arr[], int low, int high) { if (low < high) { int pivot = partition(arr, low, high); quickSort(arr, low, pivot - 1); quickSort(arr, pivot + 1, high); } } ``` 3. 归并排序(Merge Sort) 归并排序也是一种分治排序,它将待排序数组分成两部分,分别排序,然后合并成一个有序数组。在归并排序中,我们先将数组分为两半,分别对左半部分和右半部分递归执行归并排序,然后将两个有序子数组合并成一个有序数组。 下面是归并排序的C++实现: ```c++ void merge(int arr[], int l, int m, int r) { int n1 = m - l + 1; int n2 = r - m; // 创建临时数组 int L[n1], R[n2]; // 将左半部分复制到临时数组 L for (int i = 0; i < n1; i++) L[i] = arr[l + i]; // 将右半部分复制到临时数组 R for (int i = 0; i < n2; i++) R[i] = arr[m + 1 + i]; // 合并左右两个有序子数组 int 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); } } ``` 4. 计数排序(Counting Sort) 计数排序是一种非比较排序,它利用元素出现的次数来确定每个元素在有序数组中的位置。在计数排序中,我们先统计每个元素出现的次数,然后计算每个元素在有序数组中的位置,最后将元素放到正确的位置上。 下面是计数排序的C++实现: ```c++ void countingSort(int arr[], int n) { int max_val = 0; for (int i = 0; i < n; i++) { if (arr[i] > max_val) max_val = arr[i]; } // 创建计数数组 int count[max_val + 1] = {0}; // 统计每个元素出现的次数 for (int i = 0; i < n; i++) count[arr[i]]++; // 计算每个元素在有序数组中的位置 for (int i = 1; i <= max_val; i++) count[i] += count[i - 1]; // 将元素放到正确的位置上 int output[n]; 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++实现,希望能对您有所帮助!

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

xiaoyu1381

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值