/**************************************************************************** ** ** Copyright (C) 2021 The Qt Company Ltd. ** Contact: https://www.qt.io/licensing/ ** ** This file is part of the Qt VS Tools. ** ** $QT_BEGIN_LICENSE:GPL-EXCEPT$ ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and The Qt Company. For licensing terms ** and conditions see https://www.qt.io/terms-conditions. For further ** information use the contact form at https://www.qt.io/contact-us. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 3 as published by the Free Software ** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT ** included in the packaging of this file. Please review the following ** information to ensure the GNU General Public License requirements will ** be met: https://www.gnu.org/licenses/gpl-3.0.html. ** ** $QT_END_LICENSE$ ** ****************************************************************************/ using System; using System.Collections; using System.Collections.Generic; using System.Linq; namespace QtVsTools { public class PriorityQueue : BasePriorityQueue where TPriority : IComparable { public PriorityQueue() : base() { } public PriorityQueue(Func getItemKey) : base(getItemKey) { } public new void Enqueue(T item, TPriority priority) { base.Enqueue(item, priority); } } public abstract class BasePriorityQueue : Concurrent, IEnumerable where TPriority : IComparable { SortedDictionary ItemsByPriority { get; set; } Dictionary ItemPriority { get; set; } T Head { get; set; } public int Count { get; private set; } public bool IsEmpty => (Count == 0); IEnumerable Items => ThreadSafe(() => ItemsByPriority.Values.ToList()); IEnumerator IEnumerable.GetEnumerator() => Items.GetEnumerator(); IEnumerator IEnumerable.GetEnumerator() => Items.GetEnumerator(); Func GetItemKey { get; set; } public BasePriorityQueue() : this(x => x) { } public BasePriorityQueue(Func getItemKey) { ItemsByPriority = new SortedDictionary(); ItemPriority = new Dictionary(); Head = default(T); Count = 0; GetItemKey = getItemKey; } public void Clear() { lock (CriticalSection) { if (Count == 0) return; ItemsByPriority.Clear(); ItemPriority.Clear(); Head = default(T); Count = 0; } } public bool Contains(T item) { lock (CriticalSection) { return ItemPriority.ContainsKey(GetItemKey(item)); } } // Base Enqueue() is protected to allow specialized implementations to // hide the concept of priority (e.g. PunisherQueue). // protected void Enqueue(T item, TPriority priority) { if (item == null) throw new InvalidOperationException("Item cannot be null."); lock (CriticalSection) { T oldItem; if (ItemsByPriority.TryGetValue(priority, out oldItem) && !item.Equals(oldItem)) throw new InvalidOperationException("An item with the same priority exists."); TPriority oldPriority; if (ItemPriority.TryGetValue(GetItemKey(item), out oldPriority)) { ItemsByPriority.Remove(oldPriority); --Count; } ItemPriority[GetItemKey(item)] = priority; ItemsByPriority[priority] = item; Head = ItemsByPriority.First().Value; ++Count; } } public bool TryPeek(out T result) { lock (CriticalSection) { result = Head; return Count > 0; } } public T Peek() { lock (CriticalSection) { T result; if (!TryPeek(out result)) throw new InvalidOperationException("Queue is empty."); return result; } } public void Remove(T item) { lock (CriticalSection) { object key = GetItemKey(item); if (key == null) return; ItemsByPriority.Remove(ItemPriority[key]); ItemPriority.Remove(key); --Count; if (key == GetItemKey(Head)) { if (IsEmpty) Head = default(T); else Head = ItemsByPriority.First().Value; } } } public bool TryDequeue(out T result) { lock (CriticalSection) { result = Head; if (IsEmpty) return false; Remove(Head); return true; } } public T Dequeue() { lock (CriticalSection) { T result; if (!TryDequeue(out result)) throw new InvalidOperationException("Queue is empty."); return result; } } } }