.net core 中实现一个堆结构
堆结构的内部是以数组实现,表现形式为一个完全二叉树,对应关系上,上级节点的下标始终等于直接下级节点的下标(任意一个)除2的除数,下级节点的坐标左孩子为上级坐标的位置2+1,右孩子为上级坐标的位置2+2,这个条件始终满足
如下代码就是一个简易的堆结构实现
using System;
namespace test1
{
public enum HeapType
{
Max,
Min
}
public class Heap<T> where T:IComparable<T>
{
private T[] _source;
private int _heapSize;
private HeapType _heapType;
public Heap(HeapType heapType)
{
this._heapType = heapType;
_source = new T[0];
_heapSize = 0;//堆大小为0;
}
public Heap(HeapType heapType,T[] lst)
{
this._heapType = heapType;
_source = lst;
_heapSize = lst.Length;//堆大小为0;
Heapfiy();
}
/// <summary>
/// 获取当前最前边的值
/// </summary>
public T GetTopValue
{
get
{
if (_heapSize > 0)
{
var result = _source[0];
swap(0, _heapSize-- - 1);//交换最后一个项,并调整堆大小
Heapfiy(0);
return result;
}
else
{
return default(T);
}
}
}
/// <summary>
/// 插入数据
/// </summary>
/// <param name="item"></param>
public void HeapInsert(T item)
{
if (++_heapSize > _source.Length)
{
var newlst = new T[_heapSize];
for (int i = 0; i < _source.Length; i++)
{
newlst[i] = _source[i];
}
_source = newlst;
}
_source[_heapSize-1] = item;
//与上级比较 父index= i-1/2
var currentindex = _heapSize-1;
var parrentindex = (currentindex - 1) >> 1;
while (currentindex>0)
{
switch (_heapType)
{
case HeapType.Max:
if (_source[currentindex].CompareTo(_source[parrentindex])>0)
{
swap(currentindex, parrentindex);
}
break;
case HeapType.Min:
if (_source[currentindex].CompareTo(_source[parrentindex]) < 0)
{
swap(currentindex, parrentindex);
}
break;
}
currentindex = parrentindex;
parrentindex = (currentindex - 1) >> 1;
}
}
/// <summary>
/// 某一个位置的值下移到合适的位置
/// </summary>
/// <param name="index"></param>
private void Heapfiy(int index)
{
var i = index;
switch (_heapType)
{
case HeapType.Max:
while (i * 2 + 1 < _heapSize)//有子项就一路执行,并且小于子项
{
//找到目标孩子的i
int targetchlidi = i * 2 + 1;
if (i * 2 + 2 < _heapSize)//有右孩子
{
targetchlidi = _source[i * 2 + 1].CompareTo(_source[i * 2 + 2]) > 0 ? i * 2 + 1 : i * 2 + 2;
}
if (_source[i].CompareTo(_source[targetchlidi]) < 0)
{
swap(i, targetchlidi);
i = targetchlidi;
}
else
{
break;
}
}
break;
case HeapType.Min:
while (i * 2 + 1 < _heapSize)//有子项就一路执行
{
//找到目标孩子的i
int targetchlidi = i * 2 + 1;
if (i * 2 + 2 < _heapSize)//有右孩子
{
targetchlidi = _source[i * 2 + 1].CompareTo(_source[i * 2 + 2]) < 0 ? i * 2 + 1 : i * 2 + 2;
}
if (_source[i].CompareTo(_source[targetchlidi]) > 0)
{
swap(i, targetchlidi);
i = targetchlidi;
}
else
{
break;
}
}
break;
default:
break;
}
}
/// <summary>
/// 对所有位置执行下移操作
/// </summary>
private void Heapfiy()
{
for (int i = _heapSize >> 1; i >=0; i--)
{
Heapfiy(i);
}
}
private void swap(int a, int b)
{
T r = _source[a];
_source[a] = _source[b];
_source[b] = r;
}
}
}