- mongodb - 在 MongoDB mapreduce 中,如何展平值对象?
- javascript - 对象传播与 Object.assign
- html - 输入类型 ="submit"Vs 按钮标签它们可以互换吗?
- sql - 使用 MongoDB 而不是 MS SQL Server 的优缺点
我目前正在处理 Queues assignment来自普林斯顿的算法第一部分。其中一项任务是实现一个随机队列。这是一个关于使用不同数据结构的实现和权衡的问题。
问题:
随机队列类似于堆栈或队列,只是删除的项目是从数据结构中的项目中均匀随机选择的。创建实现以下 API 的通用数据类型 RandomizedQueue:
public class RandomizedQueue<Item> implements Iterable<Item> {
public RandomizedQueue() // construct an empty randomized queue
public boolean isEmpty() // is the queue empty?
public int size() // return the number of items on the queue
public void enqueue(Item item) // add the item
public Item dequeue() // remove and return a random item
public Item sample() // return (but do not remove) a random item
public Iterator<Item> iterator() // return an independent iterator over items in random order
public static void main(String[] args) // unit testing
}
这里的关键是实现出队操作和迭代器,因为出队移除并返回一个随机元素,并且迭代器以随机顺序遍历队列。
1.数组实现:
我考虑的主要实现是数组实现。除了随机性之外,这将与数组队列的实现相同。
查询 1.1: 对于出队操作,我只是简单地从数组的大小中随机选择一个数字并返回该项目,然后将数组中的最后一项移动到返回的位置项目。
但是,这种方法会改变队列的顺序。在这种情况下,这并不重要,因为我以随机顺序出队。但是,我想知道是否有一种时间/内存有效的方法可以从数组中取出一个随机元素,同时保留队列的顺序,而不必创建一个新数组并将所有数据传输给它。
// Current code for dequeue - changes the order of the array after dequeue
private int[] queue; // array queue
private int N; // number of items in the queue
public Item dequeue() {
if (isEmpty()) throw NoSuchElementException("Queue is empty");
int randomIndex = StdRandom.uniform(N);
Item temp = queue[randomIndex]
if (randomIndex == N - 1) {
queue[randomIndex] = null; // to avoid loitering
} else {
queue[randomIndex] = queue[N - 1];
queue[randomIndex] = null;
}
// code to resize array
N--;
return temp;
}
查询 1.2: 为了让迭代器满足随机返回元素的要求,我用队列的所有索引创建了一个新数组,然后用 Knuth shuffle 操作对数组进行洗牌并返回队列中那些特定索引处的元素。但是,这涉及创建一个与队列长度相等的新数组。同样,我确信我错过了一种更有效的方法。
<强>2。内部类实现
第二种实现涉及一个内部节点类。
public class RandomizedQueue<Item> {
private static class Node<Item> {
Item item;
Node<Item> next;
Node<Item> previous;
}
}
查询 2.1。 在这种情况下,我了解如何有效地执行出队操作:返回一个随机节点并更改相邻节点的引用。
但是,我对如何返回一个以随机顺序返回节点的迭代器感到困惑,而不必创建一个以随机顺序附加节点的全新队列。
此外,除了可读性和易于实现之外,在数组上使用这种数据结构还有什么好处?
这篇文章有点长。感谢你们花时间阅读我的问题并帮助我。谢谢!
最佳答案
在您的数组实现中,您的 Query 1.1 似乎是做事的最佳方式。删除随机元素的唯一其他方法是将所有内容向上移动以填充其位置。因此,如果您有 [1,2,3,4,5]
并且您删除了 2
,您的代码会将项目 3、4 和 5 向上移动并且您会减少计数。每次移除平均需要 n/2 个项目移动。所以去除是O(n)。不好。
如果您不会在迭代时添加和删除项目,那么只需在现有数组上使用 Fisher-Yates 洗牌,然后开始从前到后返回项目。没有理由复制。这实际上取决于您的使用模式。如果您设想在迭代时从队列中添加和删除项目,那么如果您不制作副本,事情就会变得不稳定。
使用链表的方法,随机出队操作很难有效地实现,因为为了得到一个随机项,你必须从前面遍历列表。因此,如果您在队列中有 100 个项目并且您想要删除第 85 个项目,则您必须从前面开始并遵循 85 个链接,然后才能到达要删除的那个。由于您使用的是双向链表,因此当要删除的项目超过中点时,您可以通过从末尾倒数来将时间减半,但是当队列中的项目数量时,它仍然非常低效很大。想象一下从一百万个项目的队列中删除第 500,000 个项目。
对于随机迭代器,您可以在开始迭代之前就地打乱链表。这需要 O(n log n) 时间,但只需要 O(1) 额外空间。同样,您在添加或删除的同时存在迭代问题。如果你想要那个能力,那么你需要制作一个副本。
关于java - 数据结构 - 随机队列,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31143654/
我正在编写一个具有以下签名的 Java 方法。 void Logger(Method method, Object[] args); 如果一个方法(例如 ABC() )调用此方法 Logger,它应该
我是 Java 新手。 我的问题是我的 Java 程序找不到我试图用作的图像文件一个 JButton。 (目前这段代码什么也没做,因为我只是得到了想要的外观第一的)。这是我的主课 代码: packag
好的,今天我在接受采访,我已经编写 Java 代码多年了。采访中说“Java 垃圾收集是一个棘手的问题,我有几个 friend 一直在努力弄清楚。你在这方面做得怎么样?”。她是想骗我吗?还是我的一生都
我的 friend 给了我一个谜语让我解开。它是这样的: There are 100 people. Each one of them, in his turn, does the following
如果我将使用 Java 5 代码的应用程序编译成字节码,生成的 .class 文件是否能够在 Java 1.4 下运行? 如果后者可以工作并且我正在尝试在我的 Java 1.4 应用程序中使用 Jav
有关于why Java doesn't support unsigned types的问题以及一些关于处理无符号类型的问题。我做了一些搜索,似乎 Scala 也不支持无符号数据类型。限制是Java和S
我只是想知道在一个 java 版本中生成的字节码是否可以在其他 java 版本上运行 最佳答案 通常,字节码无需修改即可在 较新 版本的 Java 上运行。它不会在旧版本上运行,除非您使用特殊参数 (
我有一个关于在命令提示符下执行 java 程序的基本问题。 在某些机器上我们需要指定 -cp 。 (类路径)同时执行java程序 (test为java文件名与.class文件存在于同一目录下) jav
我已经阅读 StackOverflow 有一段时间了,现在我才鼓起勇气提出问题。我今年 20 岁,目前在我的家乡(罗马尼亚克卢日-纳波卡)就读 IT 大学。足以介绍:D。 基本上,我有一家提供簿记应用
我有 public JSONObject parseXML(String xml) { JSONObject jsonObject = XML.toJSONObject(xml); r
我已经在 Java 中实现了带有动态类型的简单解释语言。不幸的是我遇到了以下问题。测试时如下代码: def main() { def ks = Map[[1, 2]].keySet()
一直提示输入 1 到 10 的数字 - 结果应将 st、rd、th 和 nd 添加到数字中。编写一个程序,提示用户输入 1 到 10 之间的任意整数,然后以序数形式显示该整数并附加后缀。 public
我有这个 DownloadFile.java 并按预期下载该文件: import java.io.*; import java.net.URL; public class DownloadFile {
我想在 GUI 上添加延迟。我放置了 2 个 for 循环,然后重新绘制了一个标签,但这 2 个 for 循环一个接一个地执行,并且标签被重新绘制到最后一个。 我能做什么? for(int i=0;
我正在对对象 Student 的列表项进行一些测试,但是我更喜欢在 java 类对象中创建硬编码列表,然后从那里提取数据,而不是连接到数据库并在结果集中选择记录。然而,自从我这样做以来已经很长时间了,
我知道对象创建分为三个部分: 声明 实例化 初始化 classA{} classB extends classA{} classA obj = new classB(1,1); 实例化 它必须使用
我有兴趣使用 GPRS 构建车辆跟踪系统。但是,我有一些问题要问以前做过此操作的人: GPRS 是最好的技术吗?人们意识到任何问题吗? 我计划使用 Java/Java EE - 有更好的技术吗? 如果
我可以通过递归方法反转数组,例如:数组={1,2,3,4,5} 数组结果={5,4,3,2,1}但我的结果是相同的数组,我不知道为什么,请帮助我。 public class Recursion { p
有这样的标准方式吗? 包括 Java源代码-测试代码- Ant 或 Maven联合单元持续集成(可能是巡航控制)ClearCase 版本控制工具部署到应用服务器 最后我希望有一个自动构建和集成环境。
我什至不知道这是否可能,我非常怀疑它是否可能,但如果可以,您能告诉我怎么做吗?我只是想知道如何从打印机打印一些文本。 有什么想法吗? 最佳答案 这里有更简单的事情。 import javax.swin
我是一名优秀的程序员,十分优秀!