如何实现具有最大值、最小值和中间值的栈和队列

转自:http://blog.csdn.net/yangzhongblog/article/details/12391959#comments

在研究“如何实现具有最大值、最小值和中间值的栈和队列”前,我们先考虑以下问题,然后由此过度到题目问题。

1)如何用两个栈实现队列

2)如何用两个队列实现栈

3)如何实现包含获取最小值函数getMin()的栈

4)如何实现包含获取中间值函数getMedian()的栈

5)如何实现包含获取最小值函数getMin()的队列

1 如何用两个栈实现队列

在研究问题前,我们可以用2个栈模拟一下具体操作过程,可以总结出以下规律:

入队:元素插入stack1;

出队:如果stack2中为空,先将stack1中元素入栈stack2,然后再将stack2的栈顶元素出栈。否则直接将stack2中元素出栈。

队列为空:stack1和stack2同时为空

队列大小:为stack1和stack2大小之和

具体过程见下图(图来自《剑指offer》)


Java实现代码如下

[java]  view plain copy
  1. public class QueueByStack<E extends Comparable<E>>  {  
  2.       
  3.     private LinkedList stack1=null;  
  4.     private LinkedList stack2=null;  
  5.       
  6.     //constructor  
  7.     public QueueByStack(){  
  8.         stack1=new LinkedList();  
  9.         stack2=new LinkedList();  
  10.     }//end QueueByStack  
  11.           
  12.     public void insert(E e){  
  13.           stack1.addLast(e);  
  14.     }//end insert()  
  15.       
  16.     public E remove(){  
  17.         if(!isEmpty()){  
  18.             if(stack2.isEmpty()){  
  19.                 stack1Tostack2();  
  20.             }//end if  
  21.   
  22.             return (E)stack2.removeLast();  
  23.         }//end if  
  24.         else{  
  25.             System.out.println("queue is empty");  
  26.             return null;  
  27.         }//end else  
  28.     }//end remove  
  29.       
  30.     //将stack1中元素入栈stack2  
  31.     private void stack1Tostack2(){  
  32.         while(!stack1.isEmpty()){  
  33.             stack2.addLast(stack1.removeLast());  
  34.         }//end while      
  35.     }//end stack1ToStack2()  
  36.       
  37.     public boolean isEmpty(){  
  38.         return stack1.isEmpty() && stack2.isEmpty();  
  39.     }//end isEmpty()  
  40.       
  41.     public int size(){  
  42.         return (stack1.size()+stack2.size());  
  43.     }//end size()  
  44.   
  45.     /** 
  46.      * @param args 
  47.      */  
  48.     public static void main(String[] args) {  
  49.         QueueByStack q=new QueueByStack();  
  50.         q.insert(1);  
  51.         q.insert(2);  
  52.         q.insert(3);  
  53.         System.out.println(q.remove());  
  54.         System.out.println(q.remove());  
  55.         q.insert(4);  
  56.         System.out.println(q.remove());  
  57.         System.out.println(q.remove());  
  58.     }  
  59. }  

2 如何用两个队列实现栈

还是先用2个队列模拟栈的出栈和入栈过程,可以得出以下规律:

入栈:压入非空的那个队列

出栈:将非空队列中的n-1个元素压入空的队列中,然后将第n个元素出栈。

具体过程见下图(图来自《剑指offer》)


3 如何实现包含获取最小值函数的栈

问题:定义栈的数据结构,请在该类型中实现一个能够得到栈的最小元素的getMin函数。在该栈中,调用getMin、push及pop的时间复杂度都是O(1).

思路:用一个辅助栈stack2记住每次入栈stack1的当前最小值:在stack1入栈时,往stack2中加入当前最小值;stack1元素出栈时,stack2也出栈一个元素。最小值从stack2中获取及栈顶元素。

4 如何实现包含获取中间值函数的栈

如果能对栈中元素进行排序,那么排序好的中间值即为所求。问题3和问题4的具体代码如下,代码中同时实现了获取栈的最大值、最小值和中间值

[java]  view plain copy
  1. public class Stack<E extends Comparable<E>> {  
  2.       
  3.     private LinkedList<E> heartStack=new LinkedList<E>();   
  4.     private LinkedList<E> curMinStack=new LinkedList<E>(); //辅助栈,用于记录当前最小值  
  5.     private LinkedList<E> curMaxStack=new LinkedList<E>(); //辅助栈,用于记录当前最大值  
  6.       
  7.     public void push(E e){  
  8.            heartStack.push(e);  
  9.            //当前最小值入栈curMinStack  
  10.            E currentMin=curMinStack.peek();  
  11.            if(currentMin.compareTo(e)>0){  
  12.                curMinStack.push(e);  
  13.            }//end if  
  14.            else  
  15.                curMinStack.push(currentMin);  
  16.              
  17.            //当前最大值入栈curMinStack  
  18.            E currentMax=curMinStack.peek();  
  19.            if(currentMax.compareTo(e)<0){  
  20.                curMaxStack.push(e);  
  21.            }//end if  
  22.            else  
  23.                curMaxStack.push(currentMax);  
  24.     }//end push()  
  25.       
  26.     public E pull(){  
  27.         if(isEmpty())  
  28.             return null;  
  29.         else{  
  30.             E e=heartStack.poll();  
  31.             curMinStack.poll();  
  32.             curMaxStack.poll();  
  33.             return e;  
  34.         }//end else  
  35.     }//end pull()  
  36.       
  37.     public E getMax(){  
  38.         return curMaxStack.peek();  
  39.     }//end getMax()  
  40.       
  41.     public E getMin(){  
  42.         return curMinStack.peek();  
  43.     }//end getMin()  
  44.       
  45.     public int size(){  
  46.         return heartStack.size();  
  47.     }//end size()  
  48.       
  49.     public boolean isEmpty(){  
  50.         return heartStack.isEmpty();  
  51.     }//end isEmpty()  
  52.       
  53.     public E getMedian(){  
  54.         E[] e=(E[]) heartStack.toArray();  
  55.         Arrays.sort(e);  
  56.         return e[e.length/2];  
  57.     }//end getMedian()  
  58.       
  59.     public E[] toArray(){  
  60.         return (E[])heartStack.toArray();  
  61.     }//end toArray()  
  62. }  

5 如何实现包含获取最小值函数的队列

问题:定义队列的数据结构,请在该类型中实现一个能够得到队列的最小元素的getMin函数。在该队列中,调用getMin、insert及remove的时间复杂度都是O(1).

思路1:用最小堆实现优先队列,获取最小值时间复杂度为O(nlogn),但优先队列只能获取最小值,remove获取的不是先入队的元素。

思路2:

如果能用栈有效地实现队列,而栈的获取最小值的操作又很容易实现,那么队列的获取最小值的操作也很容易完成。

因为上面可用2个栈实现栈的min函数,而用2个栈可以实现队列。所以可以用已实现了获取最小值的栈stack1和stack2实现队列,而整个队列的最小值从min(stack1.getMin(),stack2.getMin())中获取。具体实现代码见问题6中代码。

6 如何实现具有最大值、最小值和中间值的栈和队列

问题5解决了用O(1)时间获取栈的最小值,那么解决最大值的问题也迎刃而解。对于获取队列的中间值,可以将队列中所有元素排序,然后获取排序后的中间值。

具体实现如下(代码中的stack类为上述问题4中已实现的可以获取最大值和最小值的栈):

[java]  view plain copy
  1. public class ExmaPeekableQueue<E extends Comparable<E>>  implements IExamPeekableQueue{  
  2.     Stack stack1=new Stack();  
  3.     Stack stack2=new Stack();  
  4.       
  5.     @Override  
  6.     public void enqueue(Comparable e) {  
  7.         stack1.push(e);       
  8.     }//end enqueue()  
  9.   
  10.     @Override  
  11.     public Comparable<E> dequeue() {  
  12.         if(stack2.isEmpty()){  
  13.             stack2.push(stack1.pull());  
  14.         }//end if  
  15.           
  16.         return stack2.pull();  
  17.     }//end dequeue()  
  18.   
  19.     @Override  
  20.     public Comparable<E> peekMedian() {  
  21.         Comparable[] arr=null//用于存储队列中当前元素的数组  
  22.           
  23.         if(stack1.isEmpty()){  
  24.             arr=stack2.toArray();  
  25.         }//end if  
  26.         else if(stack2.isEmpty()){  
  27.             arr=stack1.toArray();  
  28.         }//end if  
  29.         else{  
  30.             arr=new Comparable[size()];  
  31.             Comparable[] arrE1=stack1.toArray();  
  32.             Comparable[] arrE2=stack2.toArray();  
  33.               
  34.             //将2个栈中的元素复制到一个数组中  
  35.             int i=0;  
  36.             for(;i<stack1.size();i++){  
  37.                 arr[i]=arrE1[i];  
  38.             }//end for  
  39.               
  40.             for(int j=0;j<stack2.size();j++){  
  41.                 arr[++i]=arrE2[j];  
  42.             }//end for  
  43.         }//end else  
  44.               
  45.         Arrays.sort(arr);  
  46.         return arr[arr.length/2];  
  47.     }//end peekMedian()  
  48.   
  49.     @Override  
  50.     public Comparable peekMaximum() {  
  51.         Comparable max1=stack1.getMax();  
  52.         Comparable max2=stack2.getMax();  
  53.           
  54.         if(max1.compareTo(max2)>0){  
  55.             return max1;  
  56.         }//end if  
  57.         else  
  58.             return max2;  
  59.     }  
  60.   
  61.     @Override  
  62.     public Comparable<E> peekMinimum() {  
  63.         Comparable min1=stack1.getMin();  
  64.         Comparable min2=stack2.getMin();  
  65.           
  66.         if(min1.compareTo(min2)>0){  
  67.             return min2;  
  68.         }//end if  
  69.         else  
  70.             return min1;  
  71.     }//end peekMinimum()  
  72.   
  73.     @Override  
  74.     public int size() {  
  75.         return (stack1.size()+stack2.size());  
  76.     }//end size()  
  77. }//end class  
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
提供的源码资源涵盖了安卓应用、小程序、Python应用和Java应用等多个领域,每个领域都包含了丰富的实例和项目。这些源码都是基于各自平台的最新技术和标准编写,确保了在对应环境下能够无缝运行。同时,源码中配备了详细的注释和文档,帮助用户快速理解代码结构和实现逻辑。 适用人群: 这些源码资源特别适合大学生群体。无论你是计算机相关专业的学生,还是对其他领域编程感兴趣的学生,这些资源都能为你提供宝贵的学习和实践机会。通过学习和运行这些源码,你可以掌握各平台开发的基础知识,提升编程能力和项目实战经验。 使用场景及目标: 在学习阶段,你可以利用这些源码资源进行课程实践、课外项目或毕业设计。通过分析和运行源码,你将深入了解各平台开发的技术细节和最佳实践,逐步培养起自己的项目开发和问题解决能力。此外,在求职或创业过程中,具备跨平台开发能力的大学生将更具竞争力。 其他说明: 为了确保源码资源的可运行性和易用性,特别注意了以下几点:首先,每份源码都提供了详细的运行环境和依赖说明,确保用户能够轻松搭建起开发环境;其次,源码中的注释和文档都非常完善,方便用户快速上手和理解代码;最后,我会定期更新这些源码资源,以适应各平台技术的最新发展和市场需求。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值