3.7问题:
有家动物收容所只收容狗与猫,且严格遵守“先进先出”的原则。在收养该收容所的动物时,收养人只能收养所有动物中“最老”(根据进入收容所的时间长短)的动物,或者,可以挑选猫或狗(同时必须收养此类动物中“最老”的)。换言之,收养人不能自由挑选想收养的对象。请创建适用于这个系统的数据结构,实现各种操作方法,比如enqueue、dequeueany、dequeuedog和dequeuecat等。允许使用Java内置的linkedlist数据结构。
思想:
这个问题有多种不同的解法。比如,我们可以只维护一个队列。这么做的话,dequeueany(收养任意一种动物)实现起来就很简单,但dequeuedog(受养狗)和dequeuecat(收养猫)就要迭代访问整个队列,才能找到第一只该被收养的的狗或猫。这会增加整个解法的复杂度,降低执行效率。
另外一种解法简单明了而又高效,只需为狗和猫各自创建一个队列,然后将两者放进名为animalqueue的包裹类,并且存储某种形式的时间戳,以标记每只动物进入队列(即收容所)时的时间。当调用dequeueany时,查看狗队列和猫队列的首部,并返回“最老”的那一只。
import java.util.*;
class catdog{
public static void main(String args[]){
Dog puppy = new Dog("puppy");
Cat catty = new Cat("catty");
AnimalQueue animalqueue = new AnimalQueue();
animalqueue.enqueue(puppy);
animalqueue.enqueue(catty);//进收容所
System.out.println(animalqueue.dequeueAny().name);//出收容所
System.out.println(animalqueue.dequeueAny().name);
}
}
abstract class Animal{
private int order;
protected String name;
public Animal(String n){
name=n;
}
public void setOrder(int ord){//设置时间戳
order =ord;
}
public int getOrder(){//获取时间戳
return order;
}
public boolean isOlderThan(Animal a){//比较时间戳
return this.order<a.getOrder();
}
}
class AnimalQueue{
LinkedList<Dog> dogs=new LinkedList<Dog>();
LinkedList<Cat> cats=new LinkedList<Cat>();
private int order = 0;//用于时间戳
public void enqueue(Animal a){//动物入队
a.setOrder(order);
order++;
if(a instanceof Dog) dogs.addLast((Dog) a);
else if(a instanceof Cat) cats.addLast((Cat) a);
}
public Animal dequeueAny(){
if(dogs.size()==0){
return dequeueCats();
}else if(cats.size()==0){
return dequeueDogs();
}
Dog dog = dogs.peek();
Cat cat = cats.peek();
if(dog.isOlderThan(cat)){
return dequeueDogs();
}else{
return dequeueCats();
}
}
public Dog dequeueDogs(){
return dogs.poll();
}
public Cat dequeueCats(){
return cats.poll();
}
}
class Dog extends Animal{
public Dog(String n){
super(n);
}
}
class Cat extends Animal{
public Cat(String n){
super(n);
}
}