2019-12-11 刷题-3(栈)

155-最小栈

  • 解法一:此题最简单的做法就是实现两个栈,一个记录当前元素,一个记录当前序列的最小元素。
    代码:
class MinStack {
public:
    stack<int> s;
    stack<int> m;
    /** initialize your data structure here. */
    MinStack() {
    }
    
    void push(int x) {
        s.push(x);
        if (m.empty())
            m.push(x);
        else{
            int cur_min = m.top();
            if (x < cur_min)
                m.push(x);
            else
                m.push(cur_min);
        }
    }
    
    void pop() {
        s.pop();
        m.pop();
    }
    
    int top() {
        return s.top();
    }
    
    int getMin() {
        return m.top();
    }
};
  • 解法二:更厉害一点的做法是,只用一个栈,当前最小值用一个int型记录。如果新压栈的数比当前最小值小,就要把当前最小值先压栈,然后更新当前最小值。
    不过,当新压栈的数与当前最小值相等时,比较容易出错。需要注意:1)当新压栈的数与当前最小值最小时,也要将最小值先压栈,否则出栈时会出错;2)当栈为空时,第一个元素要压栈两次,否则出栈最后一个元素时,会报段错误。
    代码:
class MinStack {
public:
    stack<int> s;
    int cur_min;
    /** initialize your data structure here. */
    MinStack() {
    }
    
    void push(int x) {
        if (s.empty()){
            s.push(x);   //第一个元素要压栈两次
            s.push(x);
            cur_min = x;
        }
        else{
            if (x <= cur_min){   //等号很重要
                s.push(cur_min);
                s.push(x);
                cur_min = x;
            }
            else{
                s.push(x);
            }
        }
    }
    
    void pop() {
        if (s.top() == cur_min){
            s.pop();
            cur_min = s.top();
            s.pop();
        }
        else{
            s.pop();
        }
    }
    
    int top() {
        return s.top();
    }
    
    int getMin() {
        return cur_min;
    }
};

225-用队列实现栈

  • 解法一:本题最直观的做法还是搞两个队列,一个平时存数据,一个pop的时候临时存放。
    push:O(1),pop:O(n)。
  • 解法二:另一种做法是只使用一个队列,在push的时候就把数据的顺序调整成栈的存放顺序。
    比如push a, b, c:
    1)push a,队列中只有a
    2)push b,此时队列为b a,右边为队头,左边为队尾。将a出队后再push,队列变为a b。
    3)push c,此时队列为c a b,将a和b分别出队再入队,队列变为a b c.
    push:O(n), pop: O(1)。
    代码:
class MyStack {
public:
    /** Initialize your data structure here. */
    queue <int> q1;
    int top_ele;
    MyStack() {

    }

    /** Push element x onto stack. */
    void push(int x) {
        if (q1.empty())
            q1.push(x);
        else {
            int q_len = q1.size();
            q1.push(x);
            while (q_len--) {
                int tmp = q1.front();
                q1.pop();
                q1.push(tmp);
            }
        }
    }

    /** Removes the element on top of the stack and returns that element. */
    int pop() {
        int rm_ele = q1.front();
        q1.pop();
        return rm_ele;
    }

    /** Get the top element. */
    int top() {
        return q1.front();
    }

    /** Returns whether the stack is empty. */
    bool empty() {
        return q1.empty();
    }
};

232-用栈实现队列
双栈法,一个主栈,一个辅助栈。push的时候只push到主栈中。

  • 解法一:维护一个top值,pop的时候将主栈的元素都倒到辅助栈中,完成pop再倒回主栈中。
    时间复杂度:push O(1), pop O(n), top O(1)
    这种做法比较简单,就不放代码了。

  • 解法二:分别维护一个top_all和top_s1用来存放模拟队列的队头和主栈的栈底。pop的时候将主栈的元素都倒到辅助栈中,完成pop后也不倒回,之后每次pop都从辅助栈中取,直到辅助栈为空,再将主栈元素倒入辅助栈中。平均下来,每个元素pop的代价也是O(1)。
    时间复杂度:push O(1), pop O(1), top O(1)。
    代码:

class MyQueue {
public:
    /** Initialize your data structure here. */
    stack <int> s1;  // store elements with stack order
    stack <int> s2;  // store elements with queue order
    int top_all, top_s1;
    MyQueue() {
    }

    /** Push element x to the back of queue. */
    void push(int x) {
        if (s1.empty() && s2.empty()) {
            top_all = x;
        }
        if (s1.empty() && !s2.empty()) {
            top_s1 = x;   // s1栈底元素,用于s2空时更新队头元素
        }
        s1.push(x);
    }

    /** Removes the element from in front of queue and returns that element. */
    int pop() {
        int rm_ele;
        if (!s2.empty()) {
            // pop element from s2
            rm_ele = s2.top();
            s2.pop();
            // update top_all
            if (s2.empty())
                top_all = top_s1;
            else
                top_all = s2.top();
            return rm_ele;
        }
        // transmit all elements from s1 to s2
        while (!s1.empty()) {
            int tmp = s1.top();
            s2.push(tmp);
            s1.pop();
        }
        rm_ele = s2.top();
        s2.pop();
        if (!s2.empty())
            // update top_all
            top_all = s2.top();
        return rm_ele;
    }

    /** Get the front element. */
    int peek() {
        return top_all;
    }

    /** Returns whether the queue is empty. */
    bool empty() {
        return s1.empty() && s2.empty();
    }
};

不过,这两种解法提交后运行时间都是0ms,反而是如果不维护top元素,每次取队头元素都要倒一遍数据的做法用时比较多,可能测试样例中比较多peek操作。而维护top元素的代价很小,所以还是很值得的。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 217,185评论 6 503
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,652评论 3 393
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 163,524评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,339评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,387评论 6 391
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,287评论 1 301
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,130评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,985评论 0 275
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,420评论 1 313
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,617评论 3 334
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,779评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,477评论 5 345
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,088评论 3 328
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,716评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,857评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,876评论 2 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,700评论 2 354

推荐阅读更多精彩内容