Cracking the Coding Interview(Stacks and Queues)

1.Describe how you could use a single array to implement three stacks.

我的思路:一般堆栈的实现会利用一个数组,这里一个数组若实现3个堆栈,直接考虑把数组划分为3个部分,相当于3个独立的数组,所以就有以下的实现。

但是,这种实现方式的缺点在于均分了每个stack需要的space,但是事先无法确定每个stack是否需要更多的space相比于其他的stack。但是针对这个缺陷我没有想到解决的方案。但是书中提供了一种解决思路。

class stack_share_array
{
public:
    int top;
    int gap;
    stack_share_array(int initnum,int gap_dis):top(initnum),gap(gap_dis)
    {
 
    }
    void push(int num)
    {
        array[top] = num;
        top += gap;
    }
    int pop()
    {
        top -= gap;
        int result = array[top];
        return result;
    }
    static int array[15];
};

分成3个使用方法如下使用:

    stack_share_array* stackthree[3];
for (int i = 0;i < 3;i++)
{
    stackthree[i] = new stack_share_array(i,3);
}
int test1 = 10;
int test2 = 20;
int test3 = 30;
stackthree[0]->push(test1++);
stackthree[0]->push(test1++);
stackthree[0]->push(test1++);
stackthree[0]->push(test1++);
stackthree[0]->push(test1++);
 
stackthree[1]->push(test2++);
stackthree[1]->push(test2++);
stackthree[1]->push(test2++);
stackthree[1]->push(test2++);
stackthree[1]->push(test2++);
 
stackthree[2]->push(test3++);
stackthree[2]->push(test3++);
stackthree[2]->push(test3++);
stackthree[2]->push(test3++);
stackthree[2]->push(test3++);

3种划分方式多种多样,不限于上限的实现。也可以不等分的划分。

另一种思路:一种在数组里面形成链表的思路,及每个数组单元重新定义,除了保存堆栈的存储信息之外保存前一个元素的信息,方便堆栈顶部的向前移动。这样就能够比较自由的分配3个堆栈所占数组的空间。

2.How would you design a stack which,in addition to push and pop,also has a function min which returns the minimum element?Push,pop and min should all operate in O(1) time.

我的思路:

因为min要能够在O(1)能够返回结果,所以min的操作必定之前保存的应该有记录。我第一直觉是用一个int保存最小值,但是细细想一下,如果pop()之后无法确保这个最小值是否被pop(),如果pop()了,无法更新余下的最小值,思考了很久之后发现可以保存一个根据当前top值来返回min的min数组,从开始堆栈内容为0的时候进行记录。所以这个数组的长度应该与设计堆栈的数组的长度相同。

class stack_min
{
public:
    int array[15];
    int min_array[16];
    int top;
 
    stack_min():top(0)
    {
        min_array[0] = BIG;
    }
    void push(int num)
    {
        array[top] = num;
        top ++;
        if (num < min_array[top - 1])
        {
            min_array[top] = num;
        }
        else
        {
            min_array[top] = min_array[top - 1];
        }
    }
    int pop()
    {
        top --;
        int result = array[top];
        return result;
    }
    int min()
    {
        return min_array[top];
    }
};

这样空间复杂度有些高,书中提供另外一种能够节约空间复杂度的算法。

思路是将min数组设计为一个堆栈,这样就能够不需要重复保存很多冗余的最小值。因为利用数组会重复保存同样的最小值多次。

3.Imagine a (Literal) stack of plates.If the stack gets too high,it might topple.Therefore,in real life,we would likely start a new stack when the previous stack exceeds some threshold, Implement a data structure SetOfStacks that mimics this.SetOfStacks should be composed of several stacks, and should create a new stack once the previous one exceeds capacity.SetOfStacks.push() and SetOfStacks.pop() should behave identically to a single stack (that is,pop() should return the same values as it would if there were just a single stack).

Follow up

Implement a function popAt(int index) which performs a  pop operation on a specific sub-stack.

class Stack
{
public:
    int top;
    int array[10];
 
    Stack():top(0)
    {
 
    }
    void push(int num)
    {
        array[top] = num;
        top ++;
    }
    int pop()
    {
        top --;
        int result = array[top];
        return result;
    }
};
class setofstacks
{
public:
    Stack* stackArray[10];
    int x;
    setofstacks():x(0)
    {
        for (int i = 0;i < 10;i++)
        {
            stackArray[i] = new Stack();
        }
    }
    void push(int num)
    {
        if (stackArray[x]->top < 10)
        {
            stackArray[x]->push(num);
        }
        else
        {
            x++;
            push(num);
        }
    }
    int pop()
    {
        if (stackArray[x]->top > 0)
        {
            return stackArray[x]->pop();
        }
        else
        {
            x--;
            return pop();
        }
    }
    int popAt(int index)
    {
        return stackArray[index]->pop();
    }
};

上面我的实现有这些问题。

首先,数组长度固定,如果最后一个堆栈用完,无法扩展数组长度。

其次,popAt(index)之后也许需要把后面的元素向前挪动,不然就会出现类似内存碎片一样的无法利用的空间。

4.Write a program to move the disks from the first rod to the last using Stacks.

关于汉诺塔的问题一直都是递归的经典问题,但是一般都是计算时间复杂度的举例,利用堆栈模拟这个过程还真不太会...

模拟的过程实习也是一个实际问题转换为编程问题,需要进行一些抽象化。

class Tower
{
public:
    //
    stack<int>* disks;
    int num;
    Tower(int i):num(i)
    {
        disks = new stack<int>();
    }
    int index()
    {
        return num;
    }
    void add(int d)
    {
        if (!disks->empty() && disks->top() <= d)
        {
            return;
        }
        else
        {
            disks->push(d);
        }
    }
    void movetopto(Tower* t)
    {
        int top = disks->top();
        disks->pop();
        t->add(top);
        cout<<"Move disk"<<top<<"from"<<index()<<"to"<<t->index()<<endl;
    }
    void moveDisks(int n,Tower* dst,Tower* buffer)
    {
        if (n>0)
        {
            moveDisks(n-1,buffer,dst);
            movetopto(dst);
            buffer->moveDisks(n-1,dst,this);
        }
    }
};

5.Implement a MyQueue class which implements a queue using two stacks.

class queue
{
public:
    Stack* stacktwo[2];
    queue()
    {
        for (int i = 0;i < 2;i++)
        {
            stacktwo[i] = new Stack();
        }
    }
    void push(int num)
    {
        stacktwo[0]->push(num);
    }
    int pop()
    {
        if (stacktwo[1]->top == 0)
        {
            while (stacktwo[0]->top != 0)
            {
                stacktwo[1]->push(stacktwo[0]->pop());
            }
        }
        return stacktwo[1]->pop();
    }
};

一个堆栈存放push()进来的元素,pop()的时候直接pop()另外一个堆栈,如果空了则将第一个堆栈中的元素添加进来,再执行pop()操作,这样两次之后就是先进先出的队列。

6.Write a program to sort a stack in ascending order.You should not make any assumptions about how the stack is implemented.The following are the only functions that should be used to write this program:push|pop|peek|isEmpty.

copy code from the book:

public static Stack<Interger> sort(Stack<Integer> s)
{
    Stack<Integer> r = new Stack<Integer>();
    while(!s.isEmpty())
    {
        int tmp = s.pop();
        while(!r.isEmpty() && r.peek() > tmp)
            s.push(r.pop());
        r.push(tmp)
    }
    return r;
}

Cracking the Coding Interview(Stacks and Queues)的更多相关文章

  1. Cracking the Coding Interview(Trees and Graphs)

    Cracking the Coding Interview(Trees and Graphs) 树和图的训练平时相对很少,还是要加强训练一些树和图的基础算法.自己对树节点的设计应该不是很合理,多多少少 ...

  2. 二刷Cracking the Coding Interview(CC150第五版)

    第18章---高度难题 1,-------另类加法.实现加法. 另类加法 参与人数:327时间限制:3秒空间限制:32768K 算法知识视频讲解 题目描述 请编写一个函数,将两个数字相加.不得使用+或 ...

  3. Cracking the Coding Interview(linked list)

    第二章的内容主要是关于链表的一些问题. 基础代码: class LinkNode { public: int linknum; LinkNode *next; int isvisit; protect ...

  4. Cracking the Coding Interview(String and array)

    1.1实现一个算法判断一个字符串是否存在重复字符.如果不能利用另外的数据结构又该如何实现? My solution: /** *利用类似一个hash table的计数 *然后检查这个hash tabl ...

  5. Cracking the coding interview

    写在开头 最近忙于论文的开题等工作,还有阿里的实习笔试,被虐的还行,说还行是因为自己的水平或者说是自己准备的还没有达到他们所需要人才的水平,所以就想找一本面试的书<Cracking the co ...

  6. Cracking the coding interview 第一章问题及解答

    Cracking the coding interview 第一章问题及解答 不管是不是要挪地方,面试题具有很好的联系代码总用,参加新工作的半年里,做的大多是探索性的工作,反而代码写得少了,不高兴,最 ...

  7. 《Cracking the Coding Interview》读书笔记

    <Cracking the Coding Interview>是适合硅谷技术面试的一本面试指南,因为题目分类清晰,风格比较靠谱,所以广受推崇. 以下是我的读书笔记,基本都是每章的课后习题解 ...

  8. Cracking the coding interview目录及资料收集

    前言 <Cracking the coding interview>是一本被许多人极力推荐的程序员面试书籍, 详情可见:http://www.careercup.com/book. 第六版 ...

  9. Cracking the Coding Interview 150题(二)

    3.栈与队列 3.1 描述如何只用一个数组来实现三个栈. 3.2 请设计一个栈,除pop与push方法,还支持min方法,可返回栈元素中的最小值.pop.push和min三个方法的时间复杂度必须为O( ...

随机推荐

  1. 【Leetcode】Sort List (Sorting)

    这个问题需要与归并排序排两个名单,基本思路分为切割与合并 合并后的代码Merge Two Sorted List里已经讲得非常清楚了. 所以这里直接给出代码. public ListNode merg ...

  2. Python美女[从新手到高手]--阅读&quot;见个面问题 HashMap 储存方法&quot;联想

    今伯乐在线 上看到一篇文章.一道面试题看 HashMap 的存储方式.也就是问: 在 HashMap 中存放的一系列键值对,当中键为某个我们自己定义的类型.放入 HashMap 后,我们在外部把某一个 ...

  3. openSUSE 安装

    https://lug.ustc.edu.cn/sites/opensuse-guide/installation.php 开始 1. 简介2. 改用 GNU/Linux3. 获取 openSUSE4 ...

  4. iOS8自适应布局视频教程

    联系:http://www.elsyy.com/course/6480 这是颐和园最近录制iOS8视频课程.简介iOS8出现在自适应布局. 本教程的书,颐和园<ios8 swift编程指南> ...

  5. (转)javabean操作文件正确,但是Jsp调用javabean时文件路径出错问题解决之JavaBean访问本地文件实现路径无关实现方法

        在JSP中,页面链接是使用web路径的,但如果JavaBean要访问本地文件读取配置信息的话,是需要文件的本地路径的.如果你在写 Bean的时候直接将本地路径写进去,那网站的路径就不能变化,丧 ...

  6. 让Sqlite脱离VC++ Runtime独立执行

    前段时间在开发OrayTalk(傲瑞通企业即时通信系统)的聊天记录模块时用到了Sqlite,这是我第一次接触和使用Sqlite,整体感觉还是很不错的.这里把我使用Sqlite的经验跟大家分享一下. 一 ...

  7. href 做导航 特效

    <div> <div> <%for (int i = 0; i < 200; i++) { %><%=i.ToString() %> <br ...

  8. C语言库函数大全及应用实例三

    原文:C语言库函数大全及应用实例三 [编程资料]C语言库函数大全及应用实例三 函数名: ecvt 功 能: 把一个浮点数转换为字符串 用 法: char ecvt(double value, int ...

  9. Installshield 64位操作系统下拷贝文件,如何重定向到32位的系统文件夹下

    原文:Installshield 64位操作系统下拷贝文件,如何重定向到32位的系统文件夹下 64位操作系统下拷贝文件重定向问题,在在复制代码前加上Disable(WOW64FSREDIRECTION ...

  10. 表单验证的3个函数ISSET()、empty()、is_numeric()的使用方法

    原文:表单验证的3个函数ISSET().empty().is_numeric()的使用方法 本文就简单讲一下php中表单验证的三个函数,应该比较常用吧,最后给一些示例,请看下文. ISSET();—— ...