Coursera Algorithms Programming Assignment 2: Deque and Randomized Queue (100分)
作业原文:http://coursera.cs.princeton.edu/algs4/assignments/queues.html
这次作业与第一周作业相比,稍微简单一些。有三个编程练习:双端队列(Deque)设计、随机队列(Randomized Queue)设计,还有一个排列组合类Permutation。
一、双端队列Deque
设计要求:A double-ended queue or deque (pronounced "deck") is a generalization of a stack and a queue that supports adding and removing items from either the front or the back of the data structure.
异常处理要求:Throw a java.lang.IllegalArgumentException if the client attempts to add a null item; throw a java.util.NoSuchElementException if the client attempts to remove an item from an empty deque; throw a java.lang.UnsupportedOperationException if the client calls the remove() method in the iterator; throw a java.util.NoSuchElementException if the client calls the next() method in the iterator and there are no more items to return.
性能要求:Your deque implementation must support each deque operation (including construction) in constant worst-case time. A deque containing n items must use at most 48n + 192 bytes of memory and use space proportional to the number of items currently in the deque. Additionally, your iterator implementation must support each operation (including construction) in constant worst-case time.
分析:
要求可以同时从头尾移除元素,那么该队列内部采用链表更合适,代码如下
import java.util.Iterator;
import java.util.NoSuchElementException; public class Deque<Item> implements Iterable<Item> {
private Node first; // 8 bytes
private Node last; // 8 bytes
private int size; // 4 bytes private class Node { // 16字节对象开销+8字节内部类额外开销+8+8+8=48 bytes, n个节点就是48n bytes
private Node preNode; // 前一个节点的引用
private Item item;
private Node nextNode; // 后一个节点的引用
} private class ListIterator implements Iterator<Item> {
// 16字节对象开销+8字节内部类额外开销+8=32 bytes
private Node curr = first; @Override
public boolean hasNext() {
// TODO Auto-generated method stub
return curr != null;
} @Override
public Item next() {
// TODO Auto-generated method stub
if (curr == null)
throw new NoSuchElementException("there are no more items!");
Item item = curr.item;
curr = curr.nextNode;
return item;
}
//remove不用设计,父类Iterator的remove方法就抛出UnsupportedOperationException("remove");
} public Deque() {
// construct an empty deque
size = 0;
first = null;
last = null;
} public boolean isEmpty() {
// is the deque empty?
return (size == 0);
} public int size() {
// return the number of items on the deque
return size;
} public void addFirst(Item item) {
// add the item to the front
valivate(item);
Node newNode = new Node();
newNode.item = item;
if (size == 0) { // 空队列的情况
newNode.preNode = null;
newNode.nextNode = null;
first = newNode;
last = newNode;
} else {
newNode.preNode = null;
newNode.nextNode = first;
first.preNode = newNode;
first = newNode;
}
size++;
} public void addLast(Item item) {
// add the item to the end
valivate(item);
Node newNode = new Node();
newNode.item = item;
if (size == 0) { // 空队列的情况
newNode.preNode = null;
newNode.nextNode = null;
first = newNode;
last = newNode;
} else {
last.nextNode = newNode;
newNode.preNode = last;
newNode.nextNode = null;
last = newNode;
}
size++;
} public Item removeFirst() {
// remove and return the item from the front
if (size == 0)
throw new NoSuchElementException("the deque is empty!");
Item returnItem = null;
if (size == 1) {
returnItem = first.item;
first = null;
last = null;
} else {
Node oldfirst = first;
returnItem = oldfirst.item;
first = oldfirst.nextNode;
first.preNode = null;
oldfirst.nextNode = null;
oldfirst.item = null;
}
size--;
return returnItem;
} public Item removeLast() {
// remove and return the item from the end
if (size == 0)
throw new NoSuchElementException("the deque is empty!");
Item returnItem = null;
if (size == 1) {
returnItem = first.item;
first = null;
last = null;
} else {
Node oldlast = last;
returnItem = oldlast.item;
last = oldlast.preNode;
last.nextNode = null;
oldlast.preNode = null;
oldlast.item = null;
}
size--;
return returnItem;
} public Iterator<Item> iterator() {
// return an iterator over items in order from front to end
return new ListIterator();
} private void valivate(Item item) {
if (item == null)
throw new IllegalArgumentException("the item is null!");
} public static void main(String[] args) {
// unit testing (optional)
Deque<String> queue = new Deque<String>();
System.out.println(queue.size);
queue.addFirst("a");
queue.addFirst("b");
queue.addLast("c");
queue.addFirst("d");
queue.addLast("e");
System.out.println(queue.size);
Iterator<String> iter = queue.iterator();
while (iter.hasNext()) {
System.out.println(iter.next());
}
}
}
二、随机队列Randomized Queue
设计要求:A randomized queue is similar to a stack or queue, except that the item removed is chosen uniformly at random from items in the data structure.
异常处理:The order of two or more iterators to the same randomized queue must be mutually independent; each iterator must maintain its own random order. Throw a java.lang.IllegalArgumentException if the client attempts to add a null item; throw a java.util.NoSuchElementException if the client attempts to sample or dequeue an item from an empty randomized queue; throw a java.lang.UnsupportedOperationException if the client calls the remove() method in the iterator; throw a java.util.NoSuchElementException if the client calls the next() method in the iterator and there are no more items to return.
性能要求: Your randomized queue implementation must support each randomized queue operation (besides creating an iterator) in constant amortized time. That is, any sequence of m randomized queue operations (starting from an empty queue) should take at most cm steps in the worst case, for some constant c. A randomized queue containing n items must use at most 48n + 192 bytes of memory. Additionally, your iterator implementation must support operations next() and hasNext() in constant worst-case time; and construction in linear time; you may (and will need to) use a linear amount of extra memory per iterator.
分析:
该队列每次移除的元素是随机的,性能要求提到迭代器的next方法必须是常数时间,很容易发现链表不容易满足该需求,需要用数组,代码如下:
import java.util.Iterator;
import java.util.NoSuchElementException;
import edu.princeton.cs.algs4.StdRandom; public class RandomizedQueue<Item> implements Iterable<Item> {
private Item[] rqArrays;
private int size; private class RandomIterator implements Iterator<Item> {
private int rank; // rank 记录便利的次数
private Item[] iterArrays; //两个迭代器必须相互独立,并且拥有自己的随机顺序 public RandomIterator(){
rank = size;
iterArrays = (Item[]) new Object[rank];
for(int i = 0; i<size; i++){
iterArrays[i] = rqArrays[i];
}
}
@Override
public boolean hasNext() {
// TODO Auto-generated method stub
return (rank > 0);
}
@Override
public Item next() {
// TODO Auto-generated method stub
if (rank == 0)
throw new NoSuchElementException("there are no more items!");
int r = StdRandom.uniform(0, rank); // 随机选取一个位置的元素返回
rank--;
Item item = iterArrays[r];
iterArrays[r] = iterArrays[rank];
iterArrays[rank] = item; // 将已经遍历过的元素放置队列末尾,这样下次迭代就不会被选到
return item;
}
} public RandomizedQueue() {
// construct an empty randomized queue
rqArrays = (Item[]) new Object[1];
size = 0;
} private void valivate(Item item) {
if (item == null)
throw new IllegalArgumentException("the item is null!");
} public boolean isEmpty() {
// is the queue empty?
return (size == 0);
} public int size() {
// return the number of items on the queue
return size;
} private void resize(int cap) {
Item[] temp = (Item[]) new Object[cap];
for (int i = 0; i < size; i++)
temp[i] = rqArrays[i];
rqArrays = temp;
} public void enqueue(Item item) {
// add the item
valivate(item);
rqArrays[size++] = item;
if (size == rqArrays.length)
resize(2 * rqArrays.length);
} public Item dequeue() {
// remove and return a random item
// 随机选取一个位置,将这个位置的元素与队列末尾的元素交换位置
// dequeue末尾元素时就达到随机remove元素的目的
if (size == 0)
throw new NoSuchElementException("the RandomizeQueue is empty!");
int r = StdRandom.uniform(0, size);
size--;
Item delItem = rqArrays[r];
rqArrays[r] = rqArrays[size];
rqArrays[size] = null;
if (size > 0 && size == rqArrays.length / 4)
resize(rqArrays.length / 2);
return delItem;
} public Item sample() {
// return (but do not remove) a random item
if (size == 0)
throw new NoSuchElementException("the RandomizeQueue is empty!");
return rqArrays[StdRandom.uniform(0, size)];
} public Iterator<Item> iterator() {
// return an independent iterator over items in random order
return new RandomIterator();
} public static void main(String[] args) {
// unit testing (optional)
RandomizedQueue<String> rq = new RandomizedQueue<String>();
rq.enqueue("a");
rq.enqueue("b");
rq.enqueue("c");
rq.enqueue("d");
rq.enqueue("e");
rq.enqueue("f");
rq.enqueue("g");
rq.dequeue();
Iterator<String> iter1 = rq.iterator();
Iterator<String> iter2 = rq.iterator();
while (iter1.hasNext()) {
System.out.print(iter1.next() + ",");
}
System.out.println();
while (iter2.hasNext()) {
System.out.print(iter2.next() + ",");
}
System.out.println(); }
}
三、 排列组合类Permutation
设计要求:Write a client program Permutation.java that takes a command-line integer k; reads in a sequence of strings from standard input using StdIn.readString(); and prints exactly k of them, uniformly at random. Print each item from the sequence at most once.
性能要求:The running time of Permutation must be linear in the size of the input. You may use only a constant amount of memory plus either one Deque or RandomizedQueue object of maximum size at most n. (For an extra challenge, use only one Deque or RandomizedQueue object of maximum size at most k.)
输出示例:

分析:
显然Permutation要做的事情就是读取一个k,并且从加载的输入文件内容中选取k个String进行展示,故采用RandomizedQueue比较合适。将文件中所有String都放入RandomizedQueue的话其size就是n,如果只放入k个String的话,RandomizedQueue的size就会是k,目前我只实现了size==n的方法,作业提交得分100。size==k的方法需要再斟酌下,若有进展,及时更新。
import edu.princeton.cs.algs4.StdIn;
public class Permutation {
public static void main(String[] args) {
RandomizedQueue<String> rq = new RandomizedQueue<String>();
int k = Integer.parseInt(args[0]);
while (!StdIn.isEmpty()) {
rq.enqueue(StdIn.readString());
// System.out.println(StdIn.readString());
}
while (k > 0) {
System.out.println(rq.dequeue());
k--;
}
}
}
Coursera Algorithms Programming Assignment 2: Deque and Randomized Queue (100分)的更多相关文章
- Coursera Algorithms Programming Assignment 3: Pattern Recognition (100分)
题目原文详见http://coursera.cs.princeton.edu/algs4/assignments/collinear.html 程序的主要目的是寻找n个points中的line seg ...
- Coursera Algorithms Programming Assignment 5: Kd-Trees (98分)
题目地址:http://coursera.cs.princeton.edu/algs4/assignments/kdtree.html 分析: Brute-force implementation. ...
- Coursera Algorithms Programming Assignment 4: 8 Puzzle (100分)
题目原文:http://coursera.cs.princeton.edu/algs4/assignments/8puzzle.html 题目要求:设计一个程序解决8 puzzle问题以及该问题的推广 ...
- Coursera Algorithms Programming Assignment 1: Percolation(100分)
题目来源http://coursera.cs.princeton.edu/algs4/assignments/percolation.html 作业分为两部分:建立模型和仿真实验. 最关键的部分就是建 ...
- Algorithms : Programming Assignment 3: Pattern Recognition
Programming Assignment 3: Pattern Recognition 1.题目重述 原题目:Programming Assignment 3: Pattern Recogniti ...
- Programming Assignment 2: Deques and Randomized Queues
编程作业二 作业链接:Deques and Randomized Queues & Checklist 我的代码:Deque.java & RandomizedQueue.java & ...
- Coursera Algorithms week2 栈和队列 练习测验: Queue with two stacks
题目原文: Implement a queue with two stacks so that each queue operations takes a constant amortized num ...
- Programming Assignment 2: Randomized Queues and Deques
实现一个泛型的双端队列和随机化队列,用数组和链表的方式实现基本数据结构,主要介绍了泛型和迭代器. Dequeue. 实现一个双端队列,它是栈和队列的升级版,支持首尾两端的插入和删除.Deque的API ...
- Algorithms: Design and Analysis, Part 1 - Programming Assignment #1
自我总结: 1.编程的思维不够,虽然分析有哪些需要的函数,但是不能比较好的汇总整合 2.写代码能力,容易挫败感,经常有bug,很烦心,耐心不够好 题目: In this programming ass ...
随机推荐
- Java 中访问数据库的步骤?Statement 和PreparedStatement 之间的区别?
Java 中访问数据库的步骤?Statement 和PreparedStatement 之间的区别? Java 中访问数据库的步骤 1)注册驱动: 2)建立连接: 3)创建Statement: 4)执 ...
- [Windows Server 2012] 安装Apache+PHP+MySQL
★ 欢迎来到[护卫神·V课堂],网站地址:http://v.huweishen.com★ 护卫神·V课堂 是护卫神旗下专业提供服务器教学视频的网站,每周更新视频.★ 本节我们将带领大家:Win2012 ...
- 查找java文件
想要在eclipse里找一个类文件,可以用快捷键CTRL + SHIFT + R,但是有些文件是jar包里的类文件,可以用下图方法,创建一个变量,然后按住CTRL键,点击类名称 这样就找到jar文件了
- div的浮动、清除浮动和布局
总结: 1.无序列表去除前面的小点点:list-style-type: none; 2.设置左浮动的间距. 外边距:margin :如果设定4个值就是,上右下左的顺序设置 如果设置3个值,那么左和右边 ...
- Notepad++运行JAVA代码
第一种方法: 工具栏->运行 点击后选择运行 1.在运行窗口中输入: cmd /k javac "$(FULL_CURRENT_PATH)" & echo 编译成功 ...
- 【原创】使用HTML5+canvas+JavaScript开发的原生中国象棋游戏及源码分享
目前已经实现的功能: V1.0 : 实现棋子的布局,画布及游戏场景的初始化V2.0 : 实现棋子的颜色改变V3.0 :实现所有象棋的走棋规则V4.0 : 实现所有棋子的吃子功能 GItHub源码下载地 ...
- [C++] muParser 的简单使用方法
关于 muParser 库 许多应用程序需要解析数学表达式.该库的主要目的是提供一种快速简便的方法. muParser是一个用C ++编写的可扩展的高性能数学表达式解析器库. 它的工作原理是将数学表达 ...
- 00.不规则json序列化使用eval、demjson
有下面一段字符串 import json str0 = '[{"name":"白云大道营业厅","siteaddr":"x...& ...
- System----堡垒机
你知道嘛是堡垒机吗? 你知道堡垒机是奏嘛的吗? 1,改server 端 socket server 接受到的请求 执行指令前,记录收到的指令,来源ip 用户名 缺点:每台机器都要更改源码,加入指令记录 ...
- tensorflow的数据输入
tensorflow有两种数据输入方法,比较简单的一种是使用feed_dict,这种方法在画graph的时候使用placeholder来站位,在真正run的时候通过feed字典把真实的输入传进去.比较 ...