欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

c++深入淺出講解堆排序和堆

 更新時間:2022年03月29日 15:15:38   作者:YR_T  
在c++里有很多排序方法,比如相對簡單的冒泡排序、選擇排序、插入排序,還有 STL里的sort函數(shù)  手寫快排  歸并排序等,還有就是堆排序,這次主要說堆排序和堆

堆是什么

堆是一種特殊的完全二叉樹

如果你是初學(xué)者,你的表情一定是這樣的??

別想復(fù)雜

首先,你一定見過這種圖

watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETkBZUl9U,size_12,color_FFFFFF,t_70,g_se,x_16

咱們暫時不管數(shù)字

這就是一個堆

堆又分為最大堆和最小堆

最大堆

看這張圖

watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETkBZUl9U,size_20,color_FFFFFF,t_70,g_se,x_16

上面的節(jié)點的數(shù)都比下面的節(jié)點的數(shù)大,最上面的數(shù)是最大的,這就叫最大堆??

最小堆

還是一樣的數(shù),看這張圖

watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETkBZUl9U,size_20,color_FFFFFF,t_70,g_se,x_16

這是一個最小堆,同最大堆,最上面的節(jié)點的數(shù)最小,上面的節(jié)點的數(shù)比下面的節(jié)點的數(shù)大

怎么樣,是不是很簡單?

堆排序

堆排序的基本思想是利用堆,使在排序中比較的次數(shù)明顯減少使速度更快

堆排序的時間復(fù)雜度為O(n*log(n)), 非穩(wěn)定排序,原地排序(空間復(fù)雜度O(1))。

堆排序的關(guān)鍵在于建堆和調(diào)整堆,下面簡單介紹一下建堆的過程:

可以用STL下的

make_heap()

具體步驟:

第1趟將索引0至n-1處的全部數(shù)據(jù)建大頂(或小頂)堆,就可以選出這組數(shù)據(jù)的最大值(或最小值)。將該堆的根節(jié)點與這組數(shù)據(jù)的最后一個節(jié)點交換,就使的這組數(shù)據(jù)中最大(最小)值排在了最后。

第2趟將索引0至n-2處的全部數(shù)據(jù)建大頂(或小頂)堆,就可以選出這組數(shù)據(jù)的最大值(或最小值)。將該堆的根節(jié)點與這組數(shù)據(jù)的倒數(shù)第二個節(jié)點交換,就使的這組數(shù)據(jù)中最大(最小)值排在了倒數(shù)第2位。

第k趟將索引0至n-k處的全部數(shù)據(jù)建最大(或最小)堆,就可以選出這組數(shù)據(jù)的最大值(或最小值)。將該堆的根節(jié)點與這組數(shù)據(jù)的倒數(shù)第k個節(jié)點交換,就使的這組數(shù)據(jù)中最大(最小)值排在了倒數(shù)第k位。

其實整個堆排序過程中, 我們只需重復(fù)做兩件事:

建堆(初始化+調(diào)整堆, 時間復(fù)雜度為O(n));

拿堆的根節(jié)點和最后一個節(jié)點交換(siftdown, 時間復(fù)雜度為O(n*log n) ).

因而堆排序整體的時間復(fù)雜度為O(n*log n)

沒看懂可以看看這個圖

最終代碼

#include <iostream>
#include <stdlib.h>
using namespace std;
 
/*******************************************/
/*  堆排序
/******************************************/
 
void swap(int &a, int &b)  //位置互換函數(shù)
{
	int temp = a;
	a = b;
	b = temp;
}
 
 
void Heap(int array[], int length, int index)  //堆排序算法(大頂堆)
{
	int left = 2 * index + 1;  //左節(jié)點數(shù)組下標(biāo)
	int right = 2 * index + 2;  //右節(jié)點數(shù)組下標(biāo)
	int max = index;  //index是父節(jié)點
 
	if (left < length && array[left] > array[max])  //左節(jié)點與父節(jié)點比較
	{
		max = left;
	}
	
	if (right < length && array[right] > array[max])  //右節(jié)點與父節(jié)點比較
	{
		max = right;
	}
 
	if (array[index] != array[max])
	{
		swap(array[index], array[max]);
		Heap(array, length, max);  //遞歸調(diào)用
	}
}
 
 
void HeapSort(int array[], int size)  //堆排序函數(shù)
{
	for (int i = size / 2 - 1; i >= 0; i--)  // 創(chuàng)建一個堆
	{
		Heap(array, size, i);
	}
 
	for (int i = size - 1; i >= 1; i--)
	{
		swap(array[0], array[i]);  //將array[0]的最大值放到array[i]的位置上,最大值往后靠
		Heap(array, i, 0);  //調(diào)用堆排序算法進行比較
	}
}
 
 
int main(void)  //主程序
{
	const int n = 6;  //數(shù)組元素的數(shù)量
	int array[n];
	cout << "請輸入6個整數(shù):" << endl;
	for (int i = 0; i < n; i++)
	{
		cin >> array[i];
	}
 
	cout << endl;  //換行
 
	HeapSort(array, n);  // 調(diào)用HeapSort函數(shù)  進行比較
 
	cout << "由小到大的順序排列后:" << endl;
	for (int i = 0; i < n; i++)
	{
		cout << "Array" << "[" << i << "]" << " = " << array[i] << endl;
	}
 
	cout << endl << endl;  //換行
 
	system("pause");  //調(diào)試時,黑窗口不會閃退,一直保持
	return 0;
}

關(guān)于堆

C++中堆的應(yīng)用:make_heap, pop_heap, push_heap, sort_heap

函數(shù)說明: make_heap將[start, end)范圍進行堆排序,默認(rèn)使用less, 即最大元素放在第一個。

pop_heap將front(即第一個最大元素)移動到end的前部,同時將剩下的元素重新構(gòu)造成(堆排序)一個新的heap。

push_heap對剛插入的(尾部)元素做堆排序。

sort_heap將一個堆做排序,最終成為一個有序的系列,可以看到sort_heap時,必須先是一個堆(兩個特性:1、最大元素在第一個 2、添加或者刪除元素以對數(shù)時間),因此必須先做一次make_heap.

到此這篇關(guān)于c++深入淺出講解堆排序和堆的文章就介紹到這了,更多相關(guān)c++ 堆排序內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Qt實現(xiàn)矩形大小任意縮放的示例代碼

    Qt實現(xiàn)矩形大小任意縮放的示例代碼

    這篇文章主要介紹了Qt如何實現(xiàn)在窗口上繪制任意大小的矩形,并且通過邊角的拖曳按鈕可改變矩形大小,感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下
    2022-06-06
  • C語言中數(shù)據(jù)如何存儲進內(nèi)存揭秘

    C語言中數(shù)據(jù)如何存儲進內(nèi)存揭秘

    使用編程語言進行編程時,需要用到各種變量來存儲各種信息。變量保留的是它所存儲的值的內(nèi)存位置。這意味著,當(dāng)您創(chuàng)建一個變量時,就會在內(nèi)存中保留一些空間。您可能需要存儲各種數(shù)據(jù)類型的信息,操作系統(tǒng)會根據(jù)變量的數(shù)據(jù)類型,來分配內(nèi)存和決定在保留內(nèi)存中存儲什么
    2022-08-08
  • 基于C語言實現(xiàn)簡易的掃雷游戲

    基于C語言實現(xiàn)簡易的掃雷游戲

    這篇文章主要為大家詳細(xì)介紹了基于C語言實現(xiàn)簡易的掃雷游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-06-06
  • C++中關(guān)于互斥量的全面認(rèn)知

    C++中關(guān)于互斥量的全面認(rèn)知

    線程的主要優(yōu)勢在于,能夠通過全局變量來共享信息。不過,這種便捷的共享是有代價的:必須確保多個線程不會同時修改同一變量,或者某一線程不會讀取正由其他線程修改的變量。為了防止出現(xiàn)線程某甲試圖訪?問一共享變量時,線程某乙正在對其進行修改。引入了互斥量
    2022-05-05
  • C++數(shù)據(jù)結(jié)構(gòu)與算法之反轉(zhuǎn)鏈表的方法詳解

    C++數(shù)據(jù)結(jié)構(gòu)與算法之反轉(zhuǎn)鏈表的方法詳解

    這篇文章主要介紹了C++數(shù)據(jù)結(jié)構(gòu)與算法之反轉(zhuǎn)鏈表的方法,結(jié)合實例形式分析了C++反轉(zhuǎn)鏈表的原理、實現(xiàn)方法及相關(guān)注意事項,需要的朋友可以參考下
    2017-08-08
  • C++ 類和對象基礎(chǔ)篇

    C++ 類和對象基礎(chǔ)篇

    類是創(chuàng)建對象的模板,一個類可以創(chuàng)建多個對象,每個對象都是類類型的一個變量;創(chuàng)建對象的過程也叫類的實例化。每個對象都是類的一個具體實例(Instance),擁有類的成員變量和成員函數(shù)
    2020-01-01
  • C語言 動態(tài)內(nèi)存開辟常見問題解決與分析流程

    C語言 動態(tài)內(nèi)存開辟常見問題解決與分析流程

    動態(tài)內(nèi)存是相對靜態(tài)內(nèi)存而言的。所謂動態(tài)和靜態(tài)就是指內(nèi)存的分配方式。動態(tài)內(nèi)存是指在堆上分配的內(nèi)存,而靜態(tài)內(nèi)存是指在棧上分配的內(nèi)存
    2022-03-03
  • C語言圖書管理系統(tǒng)課程設(shè)計

    C語言圖書管理系統(tǒng)課程設(shè)計

    這篇文章主要為大家詳細(xì)介紹了C語言圖書管理系統(tǒng)課程設(shè)計,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2018-01-01
  • c++11?實現(xiàn)枚舉值到枚舉名的轉(zhuǎn)換問題

    c++11?實現(xiàn)枚舉值到枚舉名的轉(zhuǎn)換問題

    這篇文章主要介紹了c++11?實現(xiàn)枚舉值到枚舉名的轉(zhuǎn)換,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2022-03-03
  • C++宏函數(shù)和內(nèi)聯(lián)函數(shù)的使用

    C++宏函數(shù)和內(nèi)聯(lián)函數(shù)的使用

    本文主要介紹了C++宏函數(shù)和內(nèi)聯(lián)函數(shù)的使用,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2022-07-07

最新評論