用 C# 实现带键值的优先队列

在上一篇随笔 Timus 1037. Memory management 的“进一步的讨论”小节中,我提到:

这个程序中使用 KeyedPriorityQueue 来存储已分配的“内存块”,使用 PriorityQueue 来存储尚未分配的“自由块”。这两个优先队列的算法是一样的,可以想办法合并。这将在下一篇随笔中讨论。

现在,就开始行动吧。

首先,需要一个接口,用来获取键以及获取和设置值,如下所示:

namespace Skyiv.Util{ interface IKeyValue { K GetKey(T x); V GetValue(T x); void SetValue(T x, V v); }}

接着,就是我们的带键值的优先队列 KeyedPriorityQueue 登场了:

using System;using System.
Collections.Generic;namespace Skyiv.Util{ class KeyedPriorityQueue { IComparer comparer; IKeyValue kver; Dictionaryint> keys; bool hasKey; T[] heap; public int Count { get; private set; } public KeyedPriorityQueue(IKeyValue kv) : this(null, kv) { } public KeyedPriorityQueue(int capacity, IKeyValue kv) : this(capacity, null, kv) { } public KeyedPriorityQueue(IComparer comparer, IKeyValue kv) : this(16, comparer, kv) { } public KeyedPriorityQueue(int capacity, IComparer comparer, IKeyValue kver) { this.keys = new Dictionaryint>(); this.comparer = (comparer == null) ? Comparer.Default : comparer; this.kver = kver; this.hasKey = (kver != null); this.heap = new T[capacity]; } public bool ContainsKey(K key) { return keys.ContainsKey(key); } public void Update(T v) { if (!hasKey) throw new NotSupportedException(); if (typeof(T).IsValueType) throw new InvalidOperationException("T 不能是值类型"); if (!ContainsKey(kver.GetKey(v))) throw new ArgumentOutOfRangeException("v", v, "更新优先队列时无此健值"); var id = keys[kver.GetKey(v)]; var cmp = comparer.Compare(v, heap[id]); kver.SetValue(heap[id], kver.GetValue(v)); // 注意: 这一句要求 T 是引用类型,不能是值类型 if (cmp < 0) SiftDown(id); else if (cmp > 0) SiftUp(id); } public void Push(T v) { if (Count >= heap.Length) Array.Resize(ref heap, Count * 2); if (hasKey) keys[kver.GetKey(v)] = Count; heap[Count] = v; SiftUp(Count++); } public T Pop() { var v = Top(); if (hasKey) keys.Remove(kver.GetKey(v)); heap[0] = heap[--Count]; if (Count > 0) SiftDown(hasKey ? (keys[kver.GetKey(heap[0])] = 0) : 0); return v; } public T Top() { if (Count > 0) return heap[0]; throw new InvalidOperationException("优先队列为空"); } void SiftUp(int n) { var v = heap[n]; for (var n2 = n / 2; n > 0 && comparer.Compare(v, heap[n2]) > 0; n = n2, n2 /= 2) heap[hasKey ? (keys[kver.GetKey(heap[n2])] = n) : n] = heap[n2]; heap[hasKey ? (keys[kver.GetKey(v)] = n) : n] = v; } void SiftDown(int n) { var v = heap[n]; for (var n2 = n * 2; n2 < Count; n = n2, n2 *= 2) { if (n2 + 1 < Count && comparer.Compare(heap[n2 + 1], heap[n2]) > 0) n2++; if (comparer.Compare(v, heap[n2]) >= 0) break; heap[hasKey ? (keys[kver.GetKey(heap[n2])] = n) : n] = heap[n2]; } heap[hasKey ? (keys[kver.GetKey(v)] = n) : n] = v; } }}

上述 KeyedPriorityQueue 类中,T 是要存储在这个带键值的优先队列中的数据的类型,K 是键的数据类型,V 是值的数据类型。

Dictionaryint> keys 字段用于存储键(K)在堆(heap)中的索引。

bool ContainsKey(K key) 方法用于查找指定的键是否在该优先队列中。

Update(T v) 方法用于更新指定项目的值。注意,如果要使用这个方法的话,T 不能是值类型。之所以在这个方法的第二行:

if (typeof(T).IsValueType) throw new InvalidOperationException("T 不能是值类型");

进行这个判断,而不是在将该类声明为:

class KeyedPriorityQueue where T : class { ... }

这是因为,如果不需要调用 Update(T v) 方法的话,T 还是允许是值类型的。

该类的其他方面,除了加入对 keys 字段的处理以外,就和标准的优先队列差不多了。

有了这个 KeyedPriorityQueue 类,就可以从中派生出 PriorityQueue 类来:

class PriorityQueue : KeyedPriorityQueueobject, object>{ public PriorityQueue() : base(null) { } public PriorityQueue(int capacity) : base(capacity, null) { } public PriorityQueue(IComparer comparer) : base(comparer, null) { } public PriorityQueue(int capacity, IComparer comparer) : base(capacity, comparer, null) { }}

对于 PriorityQueue 类的实例,如果调用 ContainsKey 方法,总是返回 false。如果调用 Update 方法,总是引发 NotSupportedException 异常。

现在让我们回到上一篇随笔 Timus 1037. Memory management 中,需要稍微修改一下我们的内存管理器程序。

首先,Block 必须从结构改为类,如下所示:

sealed class Block{ public int Id { get; private set; } public int Time { get; set; } public Block(int id, int time) { Id = id; Time = time; }}

然后,需要一个实现 IKeyValue 接口的类:

sealed class IdTime : IKeyValue<Block, int, int>{ public int GetKey(Block x) { return x.Id; } public int GetValue(Block x) { return x.Time; } public void SetValue(Block x, int v) { x.Time = v; }}

最后,就剩下最简单的工作了,将 Main 方法的第二行:

var used = new KeyedPriorityQueue(new TimeComparer());

修改为:

var used = new KeyedPriorityQueue<Block, int, int>(new TimeComparer(), new IdTime());

就可以了。

当然,这也是有代价的,就是运行时间和内存使用都增加了:

ID Date Author Problem
Language Judgement

result Execution

time Memory

used 2568007 21:22:09 18 Apr 2009 skyivben 1037 C# Accepted 0.406 6 129 KB 2566773 09:24:17 18 Apr 2009 skyivben 1037 C# Accepted 0.265 4 521 KB

上表中第二行就是上一篇随笔中的程序提交的结果,第一行就是按照本篇随笔修改后的程序提交的结果。

实际上,虽然 KeyedPriorityQueue 类中的代码与 PriorityQueue 类中代码有大量的重复,可以用本篇随笔的方法将 KeyedPriorityQueue 类改为 KeyedPriorityQueue 泛型类,再从中派生出 PriorityQueue 类来,以消除重复的代码。但是,这样必然造成 PriorityQueue 类的运行效率降低。所以,一般情况下,PriorityQueue 类还是使用原来的代码为好。

当然,如果能够从 PriorityQueue 类派生出 KeyedPriorityQueue 类,那就比较完美了。不知各位大侠是否还有更好的方法?

时间: 2025-01-24 02:06:30

用 C# 实现带键值的优先队列的相关文章

用C#实现带键值的优先队列

首先,需要一个接口,用来获取键以及获取和设置值,如下所示: namespace Skyiv.Util { interface IKeyValue<T, K, V> { K GetKey(T x); V GetValue(T x); void SetValue(T x, V v); } } 接着,就是我们的带键值的优先队列 KeyedPriorityQueue<T, K, V> 登场了: using System; using System.Collections.Generic;

艾伟:用 C# 实现带键值的优先队列

在上一篇随笔 Timus 1037. Memory management 的"进一步的讨论"小节中,我提到: 这个程序中使用 KeyedPriorityQueue 来存储已分配的"内存块",使用 PriorityQueue 来存储尚未分配的"自由块".这两个优先队列的算法是一样的,可以想办法合并.这将在下一篇随笔中讨论. 现在,就开始行动吧. 首先,需要一个接口,用来获取键以及获取和设置值,如下所示: namespace Skyiv.Util {

JavaScript将地址栏参数拆分成键值对的对象

window.location可获取地址栏的一系列信息,并且每个浏览器都支持该属性,非常方便.而获取到的问号后面的参数可以进行加工转变成我们所想要的键值对. Location的属性 属性名 例子 说明 hash #contents 返回URL的hash(#后跟零或多个字符),如果URL中不包含散列,则返回空字符串 host www.wrox.com:80 返回服务器名称和端口号(如果有) hostname www.wrox.com 返回不带端口号的服务器名称 href http://www.wr

mysql insert语句后如何获取insert数据的主键值自动编号

关于mysql教程 insert语句后如何获取insert数据的主键值自动编号呢, 方法很简单的,mysql数据自带的了mysql_insert_id ( );函数 使用方法: insert into(a')values('b') $nid = mysql_insert_id ( ); 方法二: LAST_INSERT_ID(),不过关于这个函数,与mysql_insert_id()比较有很多的区别,mysql_insert_id ()是直接获取当前session的insert_id,而LAST

通过分区键值发现性能问题

在很多应用中如果数据量少有规模,都会有大量的分区表存在,使用比较多的是range partition. 一般的range partition都一时间为键值,或者根据业务绑定的关键id值. 虽然已经做了一些大数据量的数据迁移,但是不管是按照分区抽取,还是根据数据条数抽取,发现有一个表比较奇怪,一个100G左右的分区表,80%以上的数据都分布在一个分区里面,而这个大分区表却有180多个分区表. 如下所示,对于表charge,如果分区的大小在200M以内,就标记为1,如果大于200M,则按照200M为

iOS - KVO 键值观察

1.KVO KVO 是 Key-Value Observing 的简写,是键值观察的意思,属于 runtime 方法.Key Value Observing 顾名思义就是一种 observer 模式用于监听属性变量值的变化,也是运行时的方法,当实例变量改变时,系统会自动采取一些动作.KVO 跟 NSNotification 有很多相似的地方,用 addObserver:forKeyPath:options:context: 去 start observer, 用 removeObserver:f

在Python中用get()方法获取字典键值的教程

  这篇文章主要介绍了在Python中用get()方法获取字典键值的教程,是Python入门中的基础知识,需要的朋友可以参考下 get()方法返回给定键的值.如果键不可用,则返回默认值None. 语法 以下是get()方法的语法: ? 1 dict.get(key, default=None) 参数 key -- 这是要搜索在字典中的键. default -- 这是要返回键不存在的的情况下默认值. 返回值 该方法返回一个给定键的值.如果键不可用,则返回默认值为None. 例子 下面的例子显示了g

拒绝远程修改Windows8注册表的键值

很多用户为了操作方便,可能会开启Windows 8系统的远程修改键值功能,允许用户从网络中的任何位置,远程修改Windows 8系统的相关注册表键值,以达到高效管理目的. 不过,当Windows 8系统开启了这项功能后,很容易被恶意用户通过专业工具侦测扫描到,这样该功能可能会被偷偷地非法利用,例如,恶意用户可以远程修改注册表键值,将攻击程序偷偷添加到注册表启动项中,以达到攻击Windows 8系统的目的.为了阻止病毒借助番茄花园这种功能攻击本地系统的安全,我们不妨进行下面的操作,拒绝远程修改键值

json-android把一个字符串变成键值对的形式

问题描述 android把一个字符串变成键值对的形式 android客户端把一个字符串变成键值对(JSON之类的)的形式,比如:{name=张三,age=20,sex=男}..实在不明白..唉..求解答.. 解决方案 String[] arrays = new String[]{"name=张三", "age=20", "sex=男"}; JSONObject element = new JSONObject(); for(int i= 0; i