问题

  用两个栈来实现一个队列,完成队列的Push和Pop操作。 队列中的元素为int类型。

思路

  将入队的元素,加入stack1,如果有出队操作,则将stack1中的元素依次压入stack2,从stack2中弹出元素就等于出队操作,直到从stack2中弹出所有元素,则才又从stack1中加入元素。

package offer007;

import java.util.Stack;

/** 
 * @Title: CQueue.java
 * @Package: offer007
 * @Description 两个栈实现一个队列
 * @author Han
 * @date 2016-4-17 上午11:24:17 
 * @version V1.0
 * @param <T>
 */ 
      
public class CQueue<T> {
    
    /**
     * 第一个栈:用于存放输入队列的元素
     */
    private Stack<T> stack1;
    /**
     * 第二个栈:用于存放输出队列的元素
     */
    private Stack<T> stack2;
    
    public CQueue() {
        
        stack1 = new Stack<T>();
        stack2 = new Stack<T>();
    }
    
    /**
     * @Description 输入队列
     * @author Han
     * @param item
     * @return
     */
    
    public T push(T item){
        
        stack1.push(item);
        
        return item;
    }
    
    /**
     * @Description 输出对头元素
     * @author Han
     * @return
     */
    
    public T pop(){
        
        /**
         * 堆栈2中没有元素
         */
        if (stack2.size() <= 0){
            
            /**
             * 将堆栈1中的元素压入堆栈2
             */
            while(stack1.size() > 0){
                
                stack2.push(stack1.pop());
            }
        }
        
        if(stack2.size() <= 0){
            
            throw new RuntimeException("The Queue Is Empty!!");
        }
        
        /**
         * 弹出堆栈2的头元素
         */
        return stack2.pop();
    }
}

测试

    @Test
    public void testCQueue() throws Exception {

        CQueue<Integer> queue = new CQueue<Integer>();

        try {
            
            queue.push(3);
            queue.push(4);
            System.out.println(queue.pop());
            System.out.println(queue.pop());
            System.out.println(queue.pop());
            queue.push(4);
            queue.push(5);
            System.out.println(queue.pop());
            System.out.println(queue.pop());

        } catch (Exception e) {

            System.out.println(e.getMessage());
        }
    }

结果

扩展

  用两个队列来实现栈

思路

  入栈操作时,哪个队列为空,则将哪个元素加入空队列,出栈操作时,将queue1中的元素(除了队尾元素)依次加入queue2中,并将queue1队尾元素当成栈顶元素弹出,下次出栈,又将queue2中的元素(除了队尾元素)依次加入queue1中,并将queue2队尾元素当成栈顶元素弹出。

package offer007;

import java.util.LinkedList;
import java.util.Queue;

/** 
 * @Title: CStack.java
 * @Package: offer007
 * @Description 使用两个队列实现栈
 * @author Han
 * @date 2016-4-17 下午2:35:38 
 * @version V1.0
 */ 
      
public class CStack<T> {

    /**
     * 队列一
     */
    private Queue<T> queue1;    
    
    /**
     * 队列二
     */
    private Queue<T> queue2;
    
    public CStack() {
        
        queue1 = new LinkedList<T>();
        queue2 = new LinkedList<T>();
    }
    
     
    /** 
     * @Description 向栈内压入元素
     * @author Han
     * @param item
     * @return  
     */
          
    public T push(T item){
        
        /**
         * 如果队列一为空,则向队列二中添加元素
         */
        if(queue1.size() == 0){

            queue2.offer(item);
        }else{
            
            queue1.offer(item);
        }
        
        return item;
    }
    
     
    /** 
     * @Description 将栈顶元素弹出
     * @author Han
     * @return  
     */
          
    public T pop(){
        
        /**
         * 哪个队列不为空,则把哪个队列中作为output队列,另一个作为input队列
         */
        if(queue1.size() != 0){
            
            return getTop(queue1, queue2);
        }else if(queue2.size() != 0){
            
            return getTop(queue2, queue1);
        }
        
        throw new RuntimeException("栈顶无元素!!");
    }

     
    /** 
     * @Description 得到栈顶元素
     * @author Han
     * @param output
     * @param input
     * @return  
     */
          
    private T getTop(Queue<T> output, Queue<T> input) {
        
        /**
         * 将output队列(除队尾元素)依次加入input队列,并将队尾元素当成栈顶元素返回
         */
        while(output.size() != 1){
            
            input.offer(output.poll());
        }
        
        return output.poll();
    }
}

测试

    @Test
    public void testCStack() throws Exception {
        
        CStack<Integer> queue = new CStack<Integer>();

        try {
            
            queue.push(3);
            queue.push(4);
            System.out.println(queue.pop());
            System.out.println(queue.pop());
            queue.push(4);
            queue.push(5);
            System.out.println(queue.pop());
            System.out.println(queue.pop());
            System.out.println(queue.pop());

        } catch (Exception e) {

            System.out.println(e.getMessage());
        }
    }

结果

总结

  清楚知道关于队列(FIFO)和栈(LIFO)的相关特点。

 

 posted on 2016-04-17 15:07  韩思明  阅读(298)  评论(0编辑  收藏  举报