程序员面试经典--动物收容所

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);
	}
}




  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值