本篇延续上一篇,介绍《剑指offer》第二版中的四个题目:从尾到头打印链表、用两个栈实现队列、旋转数组的最小数字、二进制中1的个数。

5、从尾到头打印链表

题目:输入一个链表的头结点,从尾到头反过来打印出每个节点的值。
        链表节点定义的类如下:
      

解答:这里提供两种方式:用栈和递归。
        第一种方式,用栈。因为单向链表一般在表头插入一个新元素,最早插入的会在链表表尾,新插入的会在链表表头。如果是从头到尾(正向)打印一个链表会很容易,直接从头结点开始一步步往表尾打印即可;而反过来从尾到头(反向)打印就没那么容易了,因为表尾元素不容易得到。
        栈是一种后进先出的线性表,我们可以用栈来解决这个问题。先将链表元素从头到尾压入一个栈中,这样链表表尾的元素会在栈顶的位置,然后依次出栈,即可将链表元素从尾到头打印出来。

代码如下:
      

第二种方式,递归。递归在本质上就是一个栈结构,因此可以用递归实现。要实现反向输出链表,每访问一个结点的时候,先递归地输出其后面的结点,再输出该节点自身,这样链表元素的输出顺序就反过来了。代码如下:
     

可以看到基于递归方式的代码看起来很简洁,但可能会产生一个问题:当链表非常长的时候,就会导致方法调用的层级很深,从而可能导致方法调用的栈溢出。因此推荐第一种方式。

6、用两个栈实现队列

题目:
        用两个栈实现一个队列。请实现它的两个方法appendTail和deleteHead,分别完成在队列尾部插入结点和在队列头部删除结点的功能。
        解答:
        栈是一种“后进先出”数据结构,而队列是一种“先进先出”的数据结构。本题整体思路是先用一个栈存放在队列尾部插入的结点,在队列头部删除结点时,将第一个栈中元素全部出栈并压入第二个栈中,然后第二个栈出栈,这样就能实现队列“先进先出”的特点。
        定义两个栈:stack1和stack2。stack1存放在队列尾部插入的结点,因此appendTail方法中只会涉及对stack1的入栈操作。在队列头部删除结点时,将stack1中的元素全部出栈并压入stack2中,然后stack2出栈操作。因此deleteHead方法涉及stack1的出栈、stack2的入栈、stack2的出栈三个主要操作。每次删除结点时都要先判断stack2是否为空,如果不为空则直接stack2出栈,为空时则stack1出栈并且stack2入栈。代码如下:

7、旋转数组的最小数字

题目:
        把一个数组最开始的若干个元素搬到数组的末尾,我们称之为数组的旋转。输入一个递增排序的数组的一个旋转,输出旋转数组的最小元素。例如,数组{3,4,5,1,2}为{1,2,3,4,5}的一个旋转,该数组的最小值为1。
        解答:
        本题最直接的方法是从头到尾遍历数组,找出其最小元素,这种方式的时间复杂度为O(n)。这是一种方式,但这种方式只是单纯地从头到尾遍历数组,并没有利用旋转数组的特性。
        有没有更好的方式呢?我们注意到旋转之后的数组可以分为两个排序的子数组,前面子数组元素都大于或等于后面子数组元素,最小元素正好是两个子数组的分界线。在已排序的数组中,可以用二分法查找元素,时间复杂度为O(logn),本题数组在一定程度上是排序的,因此可以尝试用二分法来寻找最小元素。
        设置两个索引i和j来分别指向数组的第一个元素和最后一个元素位置,再用一个索引mid来指向数组中间的元素位置。接下来进行判断,如果mid位置元素大于i位置元素,则说明mid位置在前一个子数组中,最小元素应该在mid和j之间,因此令i=mid,这样查找范围缩小了一半;如果mid位置元素小于j位置元素,则说明mid位置在后一个子数组中,最小元素应该在i和mid之间,因此令j=mid,查找范围又缩小了一半……以此类推,直到i和j位置相邻(即j-i==1),此时j位置元素就是最小元素。
        此外还有两种特殊情况:第一种是元素已经有序,此时直接输出第一个元素即可;第二种情况是存在重复元素,即i、j、mid位置元素相等,此时只能从头到尾遍历数组查找。
        代码:

8、二进制中1的个数

题目:
        请实现一个方法,输入一个整数,输出该数二进制表示中1的个数。例如9的二进制表示是1001,有2个1,因此如果输入数字9,则方法输出2。
        解答:
        把一个整数减去1,都是把其二进制中最右边的1变成0,如果它右边还有0,则所有0都变成1,而它左边的所有数字都保持不变。例如将12的二进制数1100减去1之后,变成了1011。而将其减去1之后的二进制数与它自身进行按位“与”运算(&),结果变成了1000,即1100&1011==1000,就相当于把1100最右边的数字1变成了0。
        基于上面的规律,得出结论:把一个整数减去1,再与原整数进行按位“与”(&)运算,会把该整数二进制中最右边的1变成0。因此一个整数的二进制中有多少个1,就可以进行多少次这样的运算,直到所有1都变成0,统计出运算次数就是1的个数。
        代码:

转载请注明出处 http://www.cnblogs.com/Y-oung/p/8877901.html

工作、学习、交流或有任何疑问,请联系邮箱:yy1340128046@163.com  微信:yy1340128046

转载于:https://www.cnblogs.com/Y-oung/p/8877901.html

剑指offer题目系列二相关推荐

  1. LeetCode Algorithm 剑指 Offer II 056. 二叉搜索树中两个节点之和

    剑指 Offer II 056. 二叉搜索树中两个节点之和 Ideas 这题有点类似一个组合题,首先是通过二叉树遍历得到一个序列,然后再通过LeetCode Algorithm 1. 两数之和的方法查 ...

  2. 【LeetCode】剑指 Offer 68 - I. 二叉搜索树的最近公共祖先

    [LeetCode]剑指 Offer 68 - I. 二叉搜索树的最近公共祖先 文章目录 [LeetCode]剑指 Offer 68 - I. 二叉搜索树的最近公共祖先 一.迭代 二.递归 祖先的定义 ...

  3. 剑指 Offer II 055. 二叉搜索树迭代器

    链接:剑指 Offer II 055. 二叉搜索树迭代器 题解: /*** Definition for a binary tree node.* struct TreeNode {* int val ...

  4. 【前端js】实现剑指offer|leetcode(二)——数组题目集合

    文章目录 一.数组去重 1. 对排序数组去重(leetcode 26. 删除排序数组中的重复项) 2. 检查是否存在重复元素(leetcode 217. 存在重复元素) 3. 检查相邻k个元素是否存在 ...

  5. 【剑指offer题解】二维数组中的查找

    前言 众所周知,对于面试而言,<剑指offer>是一本"好书". 如果你和我一样是个算法菜鸡,那么最推荐的是先把剑指offer的题目搞明白,其次再去刷LeetCode等 ...

  6. java剑指offer_剑指offer题目java实现

    Problem2:实现Singleton模式 题目描述:设计一个类,我们只能生成该类的一个实例 1 packageProblem2;2 3 public classSingletonClass {4 ...

  7. 剑指offer:2.二维数组的查找(Java版)

    备注:本文参照<剑指offer第二版> 题目: 在一个二维数组中,每一行都按照从左到右递增的顺序排序,每一列都按照从上到下递增的顺序排序.请完成一个函数, 输入这样的一个二维数组和一个整数 ...

  8. 《剑指Offer》51. 二叉搜索树的第k个结点

    题目:51. 二叉搜索树的第k个结点 知识点:二叉搜索树 题目描述: 给定一棵二叉搜索树,请找出其中的第k小的结点.例如, (5,3,7,2,4,6,8)    中,按结点数值大小顺序第三小结点的值为 ...

  9. 剑指Offer题目详解(CPP、JAVA)

    2019.10.11更新 最近开始学习c++,在本文中增加c++的解法,并更新一部分题目的解法. 前言(2018.6.1) 虽然已经度过了艰辛的找实习的日子,到了公司实习.但依然非常怀念那段准备面试. ...

  10. 剑指offer题目及答案

    剑指offer 最近在牛客网上刷剑指offer的题目,现将题目和答案总结如下: 1. 二维数组的查找 2. 替换空格 3. 从尾到头打印链表 4. 重建二叉树 5. 用两个栈实现队列 6. 旋转数组的 ...

最新文章

  1. 关于logrotate工具的日志切割
  2. MFC多视图与重绘效率
  3. 【BZOJ 1449】 1449: [JSOI2009]球队收益 (最小费用流)
  4. BufferedWriter 和 BufferedReader 的基本用法,附演示程序。以及一个复制文本文件的程序
  5. apache cxf java_Java-Apache CXF Web服务问题
  6. UI设计师必备的APP设计素材
  7. java 不同类数据的传递_java 数据在不同类之间的传递
  8. linux 内核之美,Linux内核 | 系统调用
  9. python基础学习
  10. 专业PDF转换Word/PPT/Excel工具:AnyBizSoft PDF Converter绿色便携版
  11. Xcode13运行iPhone14模拟器暨低版本Xcode运行高版本模拟器
  12. strcmp, strcasecmp, memcmp
  13. 网络编程中同步与异步,IO阻塞与非阻塞总结
  14. 骑行听音乐用什么耳机,盘点几款适合在出行佩戴的耳机
  15. 如何使用vitis_历时5年Vitis正式面世,赛灵思转型软件平台公司
  16. 【转】iOS开发人员必看的精品资料(100个)
  17. 3dvary灯光材质为什么不亮_为什么老司机都不把卤素车灯升级?听老司机说完,后悔升级了...
  18. Java Security:Java加密框架(JCA)简要说明
  19. 大炮打蚊子(已AC)
  20. 净水行业首家,安吉尔新水效国标检测能力获CNAS认可

热门文章

  1. UBUNTU18禁用锁屏功能
  2. LINUX检查一个进程内存增长的脚本
  3. WINDOWS补丁不装更好
  4. 管理感悟:宁可五个阶段做三个月,绝不一个阶段做二个月
  5. 使用jpeglib直接把NV12转为jpeg文件的代码
  6. f5 web服务器 位置,用Ctrl+F5向Web服务器索要最新鲜的信息!
  7. 开始菜单单如何打开计算机程序,什么是开始菜单
  8. dx 游戏 hook 画面截取 鼠标_关于DX-30E 个人简评
  9. arcgis水文分析_值得收藏!用DEM做水文分析前,你还需要储备的知识
  10. java表述环形链表_数据结构环形链表(java实现)