149 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			C#
		
	
	
	
			
		
		
	
	
			149 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			C#
		
	
	
	
| // this code is borrowed from RxOfficial(rx.codeplex.com) and modified
 | |
| 
 | |
| using System;
 | |
| using System.Collections.Generic;
 | |
| using System.Text;
 | |
| using System.Threading;
 | |
| 
 | |
| namespace UniRx.InternalUtil
 | |
| {
 | |
|     internal class PriorityQueue<T> where T : IComparable<T>
 | |
|     {
 | |
|         private static long _count = long.MinValue;
 | |
| 
 | |
|         private IndexedItem[] _items;
 | |
|         private int _size;
 | |
| 
 | |
|         public PriorityQueue()
 | |
|             : this(16)
 | |
|         {
 | |
|         }
 | |
| 
 | |
|         public PriorityQueue(int capacity)
 | |
|         {
 | |
|             _items = new IndexedItem[capacity];
 | |
|             _size = 0;
 | |
|         }
 | |
| 
 | |
|         private bool IsHigherPriority(int left, int right)
 | |
|         {
 | |
|             return _items[left].CompareTo(_items[right]) < 0;
 | |
|         }
 | |
| 
 | |
|         private void Percolate(int index)
 | |
|         {
 | |
|             if (index >= _size || index < 0)
 | |
|                 return;
 | |
|             var parent = (index - 1) / 2;
 | |
|             if (parent < 0 || parent == index)
 | |
|                 return;
 | |
| 
 | |
|             if (IsHigherPriority(index, parent))
 | |
|             {
 | |
|                 var temp = _items[index];
 | |
|                 _items[index] = _items[parent];
 | |
|                 _items[parent] = temp;
 | |
|                 Percolate(parent);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         private void Heapify()
 | |
|         {
 | |
|             Heapify(0);
 | |
|         }
 | |
| 
 | |
|         private void Heapify(int index)
 | |
|         {
 | |
|             if (index >= _size || index < 0)
 | |
|                 return;
 | |
| 
 | |
|             var left = 2 * index + 1;
 | |
|             var right = 2 * index + 2;
 | |
|             var first = index;
 | |
| 
 | |
|             if (left < _size && IsHigherPriority(left, first))
 | |
|                 first = left;
 | |
|             if (right < _size && IsHigherPriority(right, first))
 | |
|                 first = right;
 | |
|             if (first != index)
 | |
|             {
 | |
|                 var temp = _items[index];
 | |
|                 _items[index] = _items[first];
 | |
|                 _items[first] = temp;
 | |
|                 Heapify(first);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public int Count { get { return _size; } }
 | |
| 
 | |
|         public T Peek()
 | |
|         {
 | |
|             if (_size == 0)
 | |
|                 throw new InvalidOperationException("HEAP is Empty");
 | |
| 
 | |
|             return _items[0].Value;
 | |
|         }
 | |
| 
 | |
|         private void RemoveAt(int index)
 | |
|         {
 | |
|             _items[index] = _items[--_size];
 | |
|             _items[_size] = default(IndexedItem);
 | |
|             Heapify();
 | |
|             if (_size < _items.Length / 4)
 | |
|             {
 | |
|                 var temp = _items;
 | |
|                 _items = new IndexedItem[_items.Length / 2];
 | |
|                 Array.Copy(temp, 0, _items, 0, _size);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public T Dequeue()
 | |
|         {
 | |
|             var result = Peek();
 | |
|             RemoveAt(0);
 | |
|             return result;
 | |
|         }
 | |
| 
 | |
|         public void Enqueue(T item)
 | |
|         {
 | |
|             if (_size >= _items.Length)
 | |
|             {
 | |
|                 var temp = _items;
 | |
|                 _items = new IndexedItem[_items.Length * 2];
 | |
|                 Array.Copy(temp, _items, temp.Length);
 | |
|             }
 | |
| 
 | |
|             var index = _size++;
 | |
|             _items[index] = new IndexedItem { Value = item, Id = Interlocked.Increment(ref _count) };
 | |
|             Percolate(index);
 | |
|         }
 | |
| 
 | |
|         public bool Remove(T item)
 | |
|         {
 | |
|             for (var i = 0; i < _size; ++i)
 | |
|             {
 | |
|                 if (EqualityComparer<T>.Default.Equals(_items[i].Value, item))
 | |
|                 {
 | |
|                     RemoveAt(i);
 | |
|                     return true;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         struct IndexedItem : IComparable<IndexedItem>
 | |
|         {
 | |
|             public T Value;
 | |
|             public long Id;
 | |
| 
 | |
|             public int CompareTo(IndexedItem other)
 | |
|             {
 | |
|                 var c = Value.CompareTo(other.Value);
 | |
|                 if (c == 0)
 | |
|                     c = Id.CompareTo(other.Id);
 | |
|                 return c;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| } |