前言
遇到一道面试题,题目大概意思如下:
使用两个普通栈实现一个特殊栈,使得pop、push、min三个函数的都是复杂度为O(1)的操作,min函数是获得当前栈的最小值。
初步想法
1.要实现min函数为(1)操作,当时第一想法是事先需要算好当前最小值,于是会想到用一个值来保存当前栈中最小值元素,然后push和pop操作的时候维护这个值。这样min,push都是O(1)了,但pop可不是,如果当前弹出的是最小值,需要从新寻找当前元素的最小值,这个就不是o(1)了。
2.而且上面方法没有用到另外一个栈,于是又想到:在一个栈中存储排好序的元素,同样在push和pop操作中维护这个有序堆栈,如图:
但是这样的话min操作是O(1),但是push、pop操作因为要维护这个有序栈,怎么也想不到一个方法可以O(1)的复杂度。
当时觉得肯定是在另一个栈中缓存最小值信息,但是不知道是因为没吃饭还是怎么地,思维就此僵住了。
正确解法
遇到问题解决不了,感觉心里很不爽,于是吃饭的时候又开始想怎么充分理由栈的特性,有效的缓存最小值信息,以便min操作使用。
栈操作最大的特性是只能操作栈顶元素,想到那用一个辅助栈缓存每次栈操作时的最小值,不是刚刚好。这样每次pop操作的时候,两边一起弹出就可以;因为辅助栈的栈顶元素最当前栈中的最小值,push操作是也只需要比较入栈元素和辅助栈栈顶元素就可以。这样push、pop、min都都O(1)操作了。如图:
文字可能没说清楚,上代码,下面是PHP的实现,通过数组来模拟堆栈。
<?php /** * 使用一个辅助栈,O(1)复杂度求出栈中的最小数 * @hack 类中通过数组来模拟堆栈 * * @author laiwenhui */ class strack{ /** * 数据栈,存储栈数据; * * @var array */ private $_arrData = array(); /** * 辅助栈,存储数据组栈中每层的最下值信息; * * @var array */ private $_arrMin = array(); /** * 栈顶所在单元 * * @var int */ private $_top=-1; /** * 出栈 * @return bool|int */ public function pop(){ if ($this->_top === -1){ return false; } array_pop($this->_arrMin); $this->_top--; return array_pop($this->_arrData); } /** * 入栈 * @param int $element * @return bool */ public function push($element){ $element = intval($element); //如果栈为空,直接入栈 if ($this->_top === -1){ array_push($this->_arrData, $element); array_push($this->_arrMin, $element); $this->_top++; return true; } //不为空,判断入栈的值是否比最小栈栈顶小 $min = $this->_arrMin[$this->_top]; //比较求出最小值 $currentMin = $element < $min ? $element : $min; //当前栈中最小值入栈 array_push($this->_arrMin, $currentMin); //数据入栈 array_push($this->_arrData, $element); $this->_top++; return true; } /** * 求当前栈空间的最小值 * @return bool|int */ public function min(){ if ($this->_top === -1){ return false; } return $this->_arrMin[$this->_top]; } }
使用如下:
$obj = new strack(); $obj->push(12); $obj->push(56); $obj->push(23); $obj->push(89); $obj->push(4); var_dump($obj->min()); $obj->pop(); var_dump($obj->min()); $obj->push(8); var_dump($obj->min());
输出为:
int(4) int(12) int(8)
OK,满足要求。
你是否有其他更好方法实现,如果有,请告诉我^_^
本文向大家介绍解决vue多个路由共用一个页面的问题,包括了解决vue多个路由共用一个页面的问题的使用技巧和注意事项,需要的朋友参考一下 在日常的vue开发中我们可能会遇见多个路由需要共用一个页面的需求,特别是当路由是通过动态添加的,不同的路由展示的东西只是数据不同其他没有变化。例如: 这种情况的时候,我们发现,其实我们的页面在第一次加载成功后就不会再加载了。所以页面一直显示第一次加载的数据,给人的
问题是如何用Java中的codingBat解决这个问题。 问题陈述: 给定一个int的非空数组,返回一个新数组,其中包含原始数组中位于原始数组中最后4个之后的元素。原始数组将至少包含一个4。请注意,在Java中创建长度为0的数组是有效的。 post4({2,4,1,2})→ {1, 2} post4({4,1,4,2})→ {2} post4({4,4,1,2,3})→ {1, 2, 3} 以下是
问题是如何用Java中的codingBat解决这个问题。 问题陈述: 返回一个数组,该数组包含与给定数组完全相同的数字,但要重新排列,以便所有偶数都排在所有奇数之前。除此之外,数字可以是任意顺序。您可以修改并返回给定数组,也可以创建新数组。 偶数({1,0,1,0,0,1,1})→ {0, 0, 0, 1, 1, 1, 1} 偶数奇数({3,3,2})→{2,3,3} 偶数奇数({2,2,2})→
本文向大家介绍10个值得深思的PHP面试题,包括了10个值得深思的PHP面试题的使用技巧和注意事项,需要的朋友参考一下 文章所罗列的问题虽然看似简单,但是每个背后都涵盖了一个或几个大家容易忽视的基础知识点,希望能够帮助到你的面试和平时工作。 Q1 第一个问题关于弱类型 正确运行的输出结果: "yabadabadoo" does not contain "yaba" strpos是返回字符串str2
"var链接=driver.FindElement(By.XPath("/html/body/div[2]/div[1]/div/div/div/div[1]/div/nav/ol/li[3]"));" 我很难点击li标签中的selenium链接(var links ),尽管它可以用手点击。我已经尝试了许多方法,但还没有找到一个工作;彻底检查stackoverflows已经存在的问题,也无济于事。
本文向大家介绍用js提交表单解决一个页面有多个提交按钮的问题,包括了用js提交表单解决一个页面有多个提交按钮的问题的使用技巧和注意事项,需要的朋友参考一下 用js提交表单解决一个页面有多个提交按钮的问题,主要是判断是否为提交文本,然后再执行相应的动作,比较简单。