zoukankan      html  css  js  c++  java
  • 二叉堆部分练习

    本练习主要做了几个工作:

          1.给定一个数组来初始化二叉堆,第一种方法是通过不断插入,时间复杂度是O(nlgn),第二种方法是先把数组填入二叉堆,再从下标为H->SIZE/2的节点开始下滤,这是因为只有下标小于为H->SIZE/2才有孩子,从而可以用线性时间完成二叉堆的初始化。

          2.二叉堆的下滤和上滤,对二叉堆关键字的操作诸如删除最小,增加某关键字值,减少某关键字值等会可能改变堆性质的操作,必须用上滤和下滤来保证二叉堆的性质,比如增加了某关键字的值,该关键字可能往下走,因此对该节点采用下滤,与之相反,减少时则要上滤。

          3.删除操作是把要删除关键字的位置先求出来,用二叉树最后一个数字来覆盖这个位置的值,并使堆的大小减1,此时要比较要删除的值与该位置新值,根据大小关系来采用上滤或下滤。

          4.给定一个x,用O(K)的时间求出小于x的关键字,主要用了层序遍历的思想,从上往下一层层地走,遇到比x小的把它儿子入栈,循环至栈空。

          5.用至多3N/4的时间找出任意一个关键字x的位置,主要是先比较倒数第二层的节点,看x和这层节点的关系,若x大于这层所有的节点,这只需要检查所有叶节点,而叶节点的数量不会超过N/2,倒数第二层的节点不会超过N/4,故此时总的比较次数不会超过3N/4。若x小于这层所有的节点,则只需要检查剩下的N/2个节点即可,满足题意。

    // youxianduilie_lianxi.cpp : 定义控制台应用程序的入口点。
    //
    
    #include "stdafx.h"
    #include<iostream>
    #include<stack>
    using namespace std;
    struct HeapStruct;
    typedef struct HeapStruct *PriorityQueue;
    #define MinData -65536
    #define notExist -1
    struct HeapStruct
    {
    	int Capacity;
    	int Size;
    	int *Elements;
    };
    
    PriorityQueue Initialize(int MaxElement)                    //初始化堆
    {
    	PriorityQueue H = (PriorityQueue)malloc(sizeof(HeapStruct));  //1.申请一个堆
    	if (H == NULL) cout << "out of space";
    
    	H->Elements = (int *)malloc(sizeof(int)*(MaxElement + 1));  //2.申请一片地址空间用于存放数组
    	if (H->Elements == NULL) cout << "out of space";
    
    	H->Capacity = MaxElement;                     //3.其余值初始化
    	H->Size = 0;
    	H->Elements[0] = MinData;
    
    	return H;
    }
    
    bool isFull(PriorityQueue H)
    {
    	if (H->Size >= H->Capacity)
    		return 1;
    	else
    		return 0;
    }
    
    void Insert(int x, PriorityQueue H)
    {
    	int i;
    	if (isFull(H))                                //1.先检查堆有没有满
    	{
    		cout << "Priority queue is full";
    		return;
    	}
    	for (i = ++H->Size; H->Elements[i / 2] > x; i /= 2)   //循环,在Size+1的位置创建一个空穴,
    		H->Elements[i] = H->Elements[i / 2];              //通过x不断与其父节点比较,空穴不断上移,直到父节点小于X;
    	H->Elements[i] = x;                                  //此时i的值就是x所插入的位置。
    }
    
    PriorityQueue InitByInsert(int MaxElement,const int Given[],int num)
    {
    	PriorityQueue H = Initialize(MaxElement);
    	for (int i = 0; i < num; i++)
    		Insert(Given[i], H);
    	return H;
    }
    
    void PercDown(PriorityQueue H, int i)             //下滤操作
    {
    	int Tmp;
    	int child;
    
    	for (Tmp = H->Elements[i]; 2 * i <= H->Size; i = child)
    	{
    		child = 2 * i;
    		if (H->Elements[child + 1] && H->Elements[child + 1] < H->Elements[child])    //第一次比较,如果右儿子存在而且比较小
    			child += 1;
    		if (Tmp > H->Elements[child])                   //i位原来的值是Tmp,当儿子比他还小时就下滤,循环直到儿子都比他大
    			H->Elements[i] = H->Elements[child];        //每次这样赋值后,child位会被空出来,结合循环i=child          
    		else
    			break;
    	}
    	H->Elements[i] = Tmp;
    }
    
    void PercUp(PriorityQueue H, int i)             //上滤操作跟下滤操作基本是对偶的
    {
    	int Tmp;
    	int parent;
    
    	for (Tmp = H->Elements[i]; i > 0; i = parent)
    	{
    		parent = i >> 1;
    		if (Tmp < H->Elements[parent])
    			H->Elements[i] = H->Elements[parent];
    		else
    			break;
    	}
    	H->Elements[i] = Tmp;
    }
    
    PriorityQueue InitByGiven(int MaxElement, const int Given[], int num)
    {
    	PriorityQueue H = Initialize(MaxElement);
    	for (int i = 1; i <= num; i++)
    		H->Elements[i] = Given[i - 1];
    	H->Size = num;
        /*下滤的循环
    	for (int i = H->Size / 2; i > 0; i--)  //二叉树节点的数量最多为 H->Size / 2,比如一个满二叉树有一半是叶子
    		PercDown(H, i);
    	*/
    	//上滤的循环
    	for(int i=2;i<=H->Size;i++)
    		PercUp(H, i);
    	return H;
    }
    
    int DeleteMin(PriorityQueue H)           //删除并返回最小值
    {
    	int i, child;
    	int minElement, LastElement;
    	if (H->Size==0) {                            //先检查是否为空
    		cout << "is Empty";
    		return H->Elements[0];
    	}
    
    	minElement = H->Elements[1];               //最小的在堆顶
    	LastElement = H->Elements[H->Size--];      //删除后Size-1;
    	H->Elements[1] = LastElement;            //堆顶的元素取走以后,留下一个空穴,把堆的最后一个元素放在堆顶上,对其进行下滤即可。
    	PercDown(H, 1);
    	return minElement;
    }
    
    void Delete(PriorityQueue H, int key)     //直接删除
    {
    	int i;
    	for (i = 1; i <= H->Size; i++)      //先找到位置
    		if (H->Elements[i] == key) break;
    
    	H->Elements[i]= H->Elements[H->Size--]; //用最后一个元素覆盖要删除位置的值,再下滤
    	PercDown(H, i);
    }
    
    void rewrite(int oldkey, int newkey, PriorityQueue H)
    {
    	int i;
    	for (i = 1; i <= H->Size; i++)      //先找到位置
    		if (H->Elements[i] == oldkey) break;
    	H->Elements[i] = newkey;
    	if(newkey>oldkey)                //比较新旧值,比旧值大的话就要下滤,相当于IncreaseKey操作,比旧值小的话就要上滤,相当于DecreaseKey操作
    		PercDown(H, i);
    	else
    		PercUp(H, i);
    }
    
    void lessThan(int x, PriorityQueue H)       //用O(K)时间打印出堆中小于x的数
    {
    	if (H->Elements[1] >= x) cout << H->Elements[0];
    	else
    	{
    		int i = 1;
    		stack<int> reme;                                 //类似层序遍历,当父节点的值大于x时,就不再去检查子节点
    		reme.push(i);
    		while (!reme.empty())
    		{
    			if (H->Elements[reme.top()] < x)
    			{
    				i = reme.top();
    				cout << H->Elements[reme.top()]<<'	';
    				reme.pop();
    				if (2 * i <= H->Size) reme.push(2 * i);
    				if (2 * i+1 <= H->Size) reme.push(2 * i+1);
    			}
    			else
    				reme.pop();
    		}
    	}
    
    }
    
    int quickFind(PriorityQueue H, int x)
    {
    	int left, right;
    	for (int i = H->Size / 2; i >= H->Size / 4; i--)   //此循环用于检查叶节点的上一层
    	    if (H->Elements[i] < x)
    		{
    			left = 2 * i;
    			right = 2 * i + 1;
    			if (left <= H->Size&&H->Elements[left] == x)
    				return left;
    			if (right <= H->Size&&H->Elements[right] == x)
    				return right;
    		}
    	for (int i = H->Size / 2; i > 0; i--)      //如果上面在叶节点检查出来了就不会执这个循环           
    	{
    		if (x == H->Elements[i])
    			return i;
    	}
    	return notExist;
    }
    
    void print(PriorityQueue H)
    {
    	for (int i = 1; i <= H->Size; i++)
    		cout << H->Elements[i] << '	';
    	cout << endl;
    }
    
    
    int main()
    {
    	int Given[] = { 10,12,1,14,6,5,8,15,3,9,7,4,11,13,2 };
    	//PriorityQueue H=InitByInsert(100, Given, 15);
    	PriorityQueue H = InitByGiven(100, Given, 15);
    	print(H);
    	//lessThan(6, H);
    	cout << quickFind(H, 12);
    	//int min = DeleteMin(H);
    	//cout << min << endl;
    	//print(H);
    	//Delete(H, 3);
    	//print(H);
    	//rewrite(2,77,H);
    	//print(H);
    	while (1);
        return 0;
    }
    

      

  • 相关阅读:
    Log4j.properties配置详解
    在Listener(监听器)定时启动的TimerTask(定时任务)中使用Spring@Service注解的bean
    Java定时任务:利用java Timer类实现定时执行任务的功能
    java关于Timer schedule执行定时任务 !!!!!!!!!
    java关于Timer schedule执行定时任务
    Cannot call sendRedirect() after the response has been committed错误;
    http status 301/302 & java重定向/转发
    jQuery判断checkbox是否选中的3种方法
    怎样从Mysql官网下载mysql.tar.gz版本的安装包
    (转)Nagios 配置及监控
  • 原文地址:https://www.cnblogs.com/linear/p/6657466.html
Copyright © 2011-2022 走看看