用PHP解决的一个栈的面试题


Posted in PHP onJuly 02, 2014

前言

遇到一道面试题,题目大概意思如下:

使用两个普通栈实现一个特殊栈,使得pop、push、min三个函数的都是复杂度为O(1)的操作,min函数是获得当前栈的最小值。

初步想法

1.要实现min函数为(1)操作,当时第一想法是事先需要算好当前最小值,于是会想到用一个值来保存当前栈中最小值元素,然后push和pop操作的时候维护这个值。这样min,push都是O(1)了,但pop可不是,如果当前弹出的是最小值,需要从新寻找当前元素的最小值,这个就不是o(1)了。

2.而且上面方法没有用到另外一个栈,于是又想到:在一个栈中存储排好序的元素,同样在push和pop操作中维护这个有序堆栈,如图:

用PHP解决的一个栈的面试题

但是这样的话min操作是O(1),但是push、pop操作因为要维护这个有序栈,怎么也想不到一个方法可以O(1)的复杂度。

当时觉得肯定是在另一个栈中缓存最小值信息,但是不知道是因为没吃饭还是怎么地,思维就此僵住了。

正确解法

遇到问题解决不了,感觉心里很不爽,于是吃饭的时候又开始想怎么充分理由栈的特性,有效的缓存最小值信息,以便min操作使用。

栈操作最大的特性是只能操作栈顶元素,想到那用一个辅助栈缓存每次栈操作时的最小值,不是刚刚好。这样每次pop操作的时候,两边一起弹出就可以;因为辅助栈的栈顶元素最当前栈中的最小值,push操作是也只需要比较入栈元素和辅助栈栈顶元素就可以。这样push、pop、min都都O(1)操作了。如图:

用PHP解决的一个栈的面试题

文字可能没说清楚,上代码,下面是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,满足要求。

你是否有其他更好方法实现,如果有,请告诉我^_^

PHP 相关文章推荐
在同一窗体中使用PHP来处理多个提交任务
Oct 09 PHP
php 无限分类的树类代码
Dec 03 PHP
深入apache host的配置详解
Jun 09 PHP
php使用curl检测网页是否被百度收录的示例分享
Jan 31 PHP
Yii中render和renderPartial的区别
Sep 03 PHP
php判断类是否存在函数class_exists用法分析
Nov 14 PHP
php中ob_get_length缓冲与获取缓冲长度实例
Nov 20 PHP
php递归创建目录的方法
Feb 02 PHP
php 使用array函数实现分页
Feb 13 PHP
浅谈PHP中Stream(流)
Jun 08 PHP
基于PHP实现的多元线性回归模拟曲线算法
Jan 30 PHP
PHP使用标准库spl实现的观察者模式示例
Aug 04 PHP
函数中使用require_once问题深入探讨 优雅的配置文件定义方法推荐
Jul 02 #PHP
PHP中遇到BOM、编码导致json_decode函数无法解析问题
Jul 02 #PHP
php foreach正序倒序输出示例代码
Jul 01 #PHP
浅析ThinkPHP的模板输出功能
Jul 01 #PHP
ThinkPHP中的系统常量和预定义常量集合
Jul 01 #PHP
ThinkPHP实现多数据库连接的解决方法
Jul 01 #PHP
ThinkPHP快速入门实例教程之数据分页
Jul 01 #PHP
You might like
php笔记之:文章中图片处理的使用
2013/04/26 PHP
Fatal error: Allowed memory size of 134217728 bytes exhausted (tried to allocate 2611816 bytes)
2014/11/08 PHP
php生成shtml类用法实例
2014/12/09 PHP
PHP 7的一些引人注目的新特性简单介绍
2015/11/08 PHP
PHP扩展框架之Yaf框架的安装与使用
2016/05/18 PHP
JavaScript函数、方法、对象代码
2008/10/29 Javascript
jQuery ajax serialize()方法的使用以及常见问题解决
2013/01/27 Javascript
利用jquery动画特效和css打造的侧边弹出垂直导航
2014/04/04 Javascript
基于jquery实现图片相关操作(重绘、获取尺寸、调整大小、缩放)
2015/12/25 Javascript
jQuery+Ajax请求本地数据加载商品列表页并跳转详情页的实现方法
2017/07/12 jQuery
javascript cookie的基本操作(添加和删除)
2017/07/24 Javascript
微信小程序实现倒计时60s获取验证码
2020/04/17 Javascript
基于ES6 Array.of的用法(实例讲解)
2017/09/05 Javascript
vue移动端UI框架实现QQ侧边菜单组件
2018/03/09 Javascript
使用JavaScript中的lodash编写双色球效果
2018/06/24 Javascript
微信小程序实现签到功能
2018/10/31 Javascript
Vue.js特性Scoped Slots的浅析
2019/02/20 Javascript
JavaScript使用localStorage存储数据
2019/09/25 Javascript
vue项目中openlayers绘制行政区划
2020/12/24 Vue.js
[05:26]2014DOTA2西雅图国际邀请赛 iG战队巡礼
2014/07/07 DOTA
python检测lvs real server状态
2014/01/22 Python
用python + hadoop streaming 分布式编程(一) -- 原理介绍,样例程序与本地调试
2014/07/14 Python
Python生成密码库功能示例
2017/05/23 Python
深入了解Django中间件及其方法
2019/07/26 Python
python字典排序的方法
2019/10/12 Python
python实现贪吃蛇双人大战
2020/04/18 Python
python 实现IP子网计算
2021/02/18 Python
工程师求职简历的自我评价分享
2013/10/10 职场文书
2013年入党人员的自我鉴定
2013/10/25 职场文书
优秀员工表扬信
2014/01/17 职场文书
2014年教师节活动总结
2014/08/29 职场文书
不服从上级领导安排的检讨书
2014/09/14 职场文书
2014年前台接待工作总结
2014/12/05 职场文书
离婚协议书怎么写
2015/01/26 职场文书
2016年教师党员公开承诺书
2016/03/24 职场文书
Python中itertools库的四个函数介绍
2022/04/06 Python