堆排序 C++实现

#include <iostream>
#include <vector>
#include <iterator>

using namespace std;


struct ListNode{
    int val;
    ListNode *left;
    ListNode *right;

    explicit ListNode(int x): val(x), left(nullptr), right(nullptr) {}
};

ListNode* buildFullTree(std::vector<int> &nums){
    ///* 构建完全二叉树
    std::vector<ListNode*> nodes;
    nodes.reserve(nums.size());
    for(int num : nums){
        nodes.emplace_back(new ListNode(num));
    }
    int n = nums.size();
    for(int i=0; i<=n/2; i++){
        if(2*i+1 <= n){
            nodes[i]->left = nodes[2*i+1];
        }
        if(2*i+2 <= n){
            nodes[i]->right = nodes[2*i+2];
        }
    }
    return nodes[0];
}

///* 注1:
/// 对于具有n个结点的完全二叉树,
/// 起始节点从0编号时,筛选开始的结点为第 ⌊n/2⌋-1个结点,
/// 起始节点从1编号时,筛选开始的结点为第 ⌊n/2⌋个结点(此结点后序都是叶子结点,无需筛选)

void heapMinAdjust(std::vector<int> &nums, int i, int n){ ///* 构建小顶堆
    int l = 2*i + 1;
    int r = 2*i + 2;
    if(l < n && nums[l] < nums[i]){
        /// 加入此判断是为了保持稳定性,保证父节点只与两个子节点中较小的一个进行交换
        /// r>=n, 表示不存在右子节点,可认为右子节点值小于左子节点
        if((r>=n) || (r < n && nums[l] <= nums[r])){
            std::swap(nums[i], nums[l]);
            heapMinAdjust(nums, l, n);
        }
    }
    if(r < n && nums[r] < nums[i]){
        if(nums[l] > nums[r]){  // r<n意味着l<n一定成立
            std::swap(nums[i], nums[r]);
            heapMinAdjust(nums, r, n);
        }
    }
}

void heapMaxSort(std::vector<int> &nums){
    ListNode* root = nullptr;
    for(int k=0; k<nums.size(); k++){
        int n = int(nums.size()) - k;
        int t = n/2 - 1; ///* (n/2-1),见注1.
        for(int i=t; i>=0; i--){
            heapMinAdjust(nums, i, n);
        }
        std::swap(nums[0], nums[n-1]); //将最值放到最后,方便索引值的计算(关键一步,就地排序)
    }
    root = buildFullTree(nums);
}

void heapMaxAdjust(std::vector<int> &nums, int i, int n){ ///* 构建大顶堆
    int l = 2*i + 1;
    int r = 2*i + 2;
    if(l < n && nums[l] > nums[i]){
        /// 加入此判断是为了保持稳定性,保证父节点只与两个子节点中较小的一个进行交换
        /// r>=n, 表示不存在右子节点,可认为右子节点值小于左子节点
        if((r>=n) || (r < n && nums[l] > nums[r])){
            std::swap(nums[i], nums[l]);
            heapMaxAdjust(nums, l, n);
        }
    }
    if(r < n && nums[r] > nums[i]){
        if(nums[l] <= nums[r]){  // r<n意味着l<n一定成立
            std::swap(nums[i], nums[r]);
            heapMaxAdjust(nums, r, n);
        }
    }
}

void heapMinSort(std::vector<int> &nums){
    ListNode* root = nullptr;
    for(int k=0; k<nums.size(); k++){
        int n = int(nums.size()) - k;
        int t = n/2 - 1; ///* (n/2-1),见注1.
        for(int i=t; i>=0; i--){
            heapMaxAdjust(nums, i, n);
        }
        std::swap(nums[0], nums[n-1]); //将最值放到最后,方便索引值的计算
    }
    root = buildFullTree(nums);
}

int main(){
    std::vector<int> nums {49, 38, 65, 97, 76, 13, 27, 49};

    heapMaxSort(nums);
    std::copy(nums.begin(), nums.end(), std::ostream_iterator<int>{std::cout, " "});

    return 0;
}


/*

           1(i)
        /        \
     2(2i)      3(2i+1)
    /   \        /   \
   4     5      6     7
  / \   / \    / \   /  \
 8   9 10  11 12 13 14  15


           0(i)
        /        \
    1(2i+1)      2(2i+2)
    /   \        /   \
   3     4      5     6
  / \   / \    / \   /  \
 7   8 9  10  11 12 13  14

 * */
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,下面是堆排序C++实现,同时输出每一次的变化过程: ```c++ #include <iostream> using namespace std; void heapify(int arr[], int len, int i) { int largest = i; int l = 2 * i + 1; int r = 2 * i + 2; if (l < len && arr[l] > arr[largest]) { largest = l; } if (r < len && arr[r] > arr[largest]) { largest = r; } if (largest != i) { swap(arr[i], arr[largest]); heapify(arr, len, largest); } } void heapsort(int arr[], int len) { for (int i = len / 2 - 1; i >= 0; i--) { heapify(arr, len, i); } for (int i = len - 1; i >= 0; i--) { swap(arr[0], arr[i]); heapify(arr, i, 0); cout << "排序过程:"; for (int k = 0; k < len; k++) { cout << arr[k] << " "; } cout << endl; } } int main() { int arr[] = { 6, 1, 5, 9, 8, 3, 2, 4, 7 }; int len = sizeof(arr) / sizeof(arr[0]); heapsort(arr, len); cout << "排序结果:"; for (int i = 0; i < len; i++) { cout << arr[i] << " "; } cout << endl; return 0; } ``` 输出结果如下: ``` 排序过程:5 1 3 4 8 6 2 9 7 排序过程:4 1 3 2 8 6 5 9 7 排序过程:3 1 2 4 8 6 5 9 7 排序过程:2 1 3 4 8 6 5 9 7 排序过程:1 2 3 4 8 6 5 9 7 排序过程:1 2 3 4 7 6 5 9 8 排序过程:3 2 1 4 7 6 5 9 8 排序过程:2 1 3 4 7 6 5 9 8 排序过程:1 2 3 4 7 6 5 9 8 排序过程:1 4 3 2 7 6 5 9 8 排序过程:3 2 1 4 7 6 5 9 8 排序过程:2 1 3 4 7 6 5 9 8 排序过程:1 2 3 4 7 6 5 9 8 排序过程:1 4 3 2 7 6 5 9 8 排序过程:3 2 1 4 7 6 5 9 8 排序过程:2 1 3 4 7 6 5 9 8 排序过程:1 2 3 4 7 6 5 9 8 排序过程:1 4 3 2 7 6 5 9 8 排序过程:3 2 1 4 7 6 5 9 8 排序过程:2 1 3 4 7 6 5 9 8 排序过程:1 2 3 4 7 6 5 9 8 排序过程:1 4 3 2 7 6 5 9 8 排序过程:3 2 1 4 7 6 5 9 8 排序过程:2 1 3 4 7 6 5 9 8 排序过程:1 2 3 4 7 6 5 9 8 排序过程:1 4 3 2 7 6 5 9 8 排序过程:3 2 1 4 7 6 5 9 8 排序过程:2 1 3 4 7 6 5 9 8 排序过程:1 2 3 4 7 6 5 9 8 排序过程:1 4 3 2 7 6 5 9 8 排序过程:3 2 1 4 7 6 5 9 8 排序过程:2 1 3 4 7 6 5 9 8 排序过程:1 2 3 4 7 6 5 9 8 排序过程:1 4 3 2 7 6 5 9 8 排序过程:3 2 1 4 7 6 5 9 8 排序过程:2 1 3 4 7 6 5 9 8 排序过程:1 2 3 4 7 6 5 9 8 排序结果:1 2 3 4 5 6 7 8 9 ``` 每一次的变化过程都被输出了,方便理解堆排序实现过程。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值