剑指offer
用两个栈实现队列
入栈需要考虑stackA,但是一开始是空数组,一直在插入和删除,不需要考虑大小。
出栈需要考虑stackB,同时stackB的出栈需要依赖stackA的状态。
push进去不需要返回,pop出去需要返回。
1 | var CQueue = function() { |
30包含min函数的栈
定义栈的数据结构,请在该类型中实现一个能够得到栈的最小元素的 min 函数在该栈中,调用 min、push 及 pop 的时间复杂度都是 O(1)。
用空间换时间,制造了一个minStack和Stack同Push同pop,不过push进去的的时候放入每次放入的最小值。这是时候栈从下到上,依次表示最小值,而且也表示这最小值变换的位置。
Infinity 正无穷大
1 | var MinStack = function{ |
06从尾到头打印链表
输入一个链表的头节点,从尾到头反过来返回每个节点的值(用数组返回)。
//利用了数组把链表的数据放进去,然后调用reverse()方法得到每个节点的值。
1 | var reversePrint = function(head){ |
//反转链表
1 | function reverseLink(head){ |
24反转链表
定义一个函数,输入一个链表的头节点,反转该链表并输出反转后链表的头节点
对于没有的值先赋值,最后两个操作肯定是要复位。
1 | var reverseList = funtion(head){ |
35复杂链表的复制
请实现 copyRandomList 函数,复制一个复杂链表。在复杂链表中,每个节点除了有一个 next 指针指向下一个节点,还有一个 random 指针指向链表中的任意节点或者 null。
使用map记录新节点,然后再来操作、把map中value设置为new Node
很新奇,以后记得。
1 | function Node(val, next, random) { |
05替换空格
请实现一个函数,把字符串 s
中的每个空格替换成”%20”。
string split() join
1 | //分割 |
58 II 左旋转字符串
字符串的左旋转操作是把字符串前面的若干个字符转移到字符串的尾部。请定义一个函数实现字符串左旋转操作的功能。比如,输入字符串”abcdefg”和数字2,该函数将返回左旋转两位得到的结果”cdefgab”。
string返回一个新的字符处啊你,并且不会改动原字符串 [)
1 | var reverseLeftWords = function(s,n){ |
查找(二分 或者 map)
03数组中重复的数字
找出数组中重复的数字。在一个长度为 n 的数组 nums 里的所有数字都在 0~n-1 的范围内。数组中某些数字是重复的,但不知道有几个数字重复了,也不知道每个数字重复了几次。请找出数组中任意一个重复的数字。
(map)
map[c] === undefined也能查找这个数是不是在里面
1 | var findRepeatNumber = function(nums){ |
53I在排序数组中查找数字I
统计一个数字在排序数组中出现的次数。 示例中数组是有序的。
(二分)
1 | var search = function(nums,target){ |
53 II 0~n-1中确实的数字
一个长度为n-1的递增排序数组中的所有数字都是唯一的,并且每个数字都在范围0~n-1之内。在范围0~n-1内的n个数字中有且只有一个数字不在该数组中,请找出这个数字。
1 | 自己做的,题库没有 |
04 二维数组中的查找
在一个 n * m 的二维数组中,每一行都按照从左到右 非递减 的顺序排序,每一列都按照从上到下 非递减 的顺序排序。请完成一个高效的函数,输入这样的一个二维数组和一个整数,判断数组中是否含有该整数。
Array.prototype.flat() 会按照一个可指定的深度递归遍历数组,并将所有元素与遍历到的子数组中的元素合并为一个新数组返回。
1 | //写起来简单,但是复杂度高 |
11旋转数组的最小数字
把一个数组最开始的若干个元素搬到数组的末尾,我们称之为数组的旋转。
给你一个可能存在 重复 元素值的数组 numbers ,它原来是一个升序排列的数组,并按上述情形进行了一次旋转。请返回旋转数组的最小元素。例如,数组 [3,4,5,1,2] 为 [1,2,3,4,5] 的一次旋转,该数组的最小值为 1。
注意,数组 [a[0], a[1], a[2], …, a[n-1]] 旋转一次 的结果为数组 [a[n-1], a[0], a[1], a[2], …, a[n-2]] 。
1 | //投机 |
50第一只出现一次的字符
在字符串 s 中找出第一个只出现一次的字符。如果没有,返回一个单空格。 s 只包含小写字母。
Array.prototype.indexOf()indexOf()
方法返回在数组中可以找到给定元素的第一个索引,如果不存在,则返回 -1。
1 | indexOf(searchElement, fromIndex)开始查找的位置。如果该索引值大于或等于数组长度,意味着不会在数组里查找,返回 -1。如果参数中提供的索引值是一个负值,则将其作为数组末尾的一个抵消,即 -1 表示从最后一个元素开始查找,-2 表示从倒数第二个元素开始查找,以此类推。注意:如果参数中提供的索引值是一个负值,并不改变其查找顺序,查找顺序仍然是从前向后查询数组。如果抵消后的索引值仍小于 0,则整个数组都将会被查询。其默认值为 0。 |
利用循环找出第一个数的位置,再次查询这个数位置之后是否还存在相同的数字,存在则继续,不存在则输出
1 | var firstUniqChar = function(s){ |
搜索与回溯算法
32I 从上到下打印二叉树
从上到下打印出二叉树的每个节点,同一层的节点按照从左到右的顺序打印。
Array.prototype.shift() 会转移并回传队列的第一个元素。此方法会改变队列的长度。
**unshift()
**方法会增加一个或多个元素至队列的开头,并与回传队列的新长度。
1 | // 按照同一层从左到右的顺序打印,因此使用广度优先遍历,先将同一层的遍历完,再遍历下一层的 |
32 II从上到下打印二叉树II
从上到下按层打印二叉树,同一层的节点按从左到右的顺序打印,每一层打印到一行。
运用了一个tmp来进行对某一层的元素的存储
1 | var levelOrder = function(root){ |
32III从上到下打印二叉树III
请实现一个函数按照之字形顺序打印二叉树,即第一行按照从左到右的顺序打印,第二层按照从右到左的顺序打印,第三行再按照从左到右的顺序打印,其他行以此类推。
采用不同level中对tmp的push()还是unshift()来进行区分
1 | var levelOrder = function(root){ |
026 树的子结构
输入两棵二叉树A和B,判断B是不是A的子结构。(约定空树不是任意一个树的子结构)
B是A的子结构, 即 A中有出现和B相同的结构和节点值。
1 | 记录下js解法 |
27二叉树的镜像
请完成一个函数,输入一个二叉树,该函数输出它的镜像。
1 | var mirrorTree = function(root){ |
动态规划
10 I 斐波那契数列
写一个函数,输入 n
,求斐波那契(Fibonacci)数列的第 n
项(即 F(N)
)。斐波那契数列的定义如下:
F(0) = 0, F(1) = 1
F(N) = F(N - 1) + F(N - 2), 其中 N > 1.
斐波那契数列由 0 和 1 开始,之后的斐波那契数就是由之前的两数相加而得出。
答案需要取模 1e9+7(1000000007),如计算初始结果为:1000000008,请返回 1。
1 | //1 |
10 II 青蛙跳台问题
一只青蛙一次可以跳上1级台阶,也可以跳上2级台阶。求该青蛙跳上一个 n 级的台阶总共有多少种跳法。
答案需要取模 1e9+7(1000000007),如计算初始结果为:1000000008,请返回 1。
1 | var numWays = function(n){ |
63股票的最大利润
假设把某股票的价格按照时间先后顺序存储在数组中,请问买卖该股票一次可能获得的最大利润是多少?
1 | var maxProfit = function(prices){ |
42连续子数组的最大和
输入一个整型数组,数组中的一个或连续多个整数组成一个子数组。求所有子数组的和的最大值。
要求时间复杂度为O(n)。
1 | var manSubArray = function(nums){ |
47礼物的最大价值
在一个 m*n 的棋盘的每一格都放有一个礼物,每个礼物都有一定的价值(价值大于 0)。你可以从棋盘的左上角开始拿格子里的礼物,并每次向右或者向下移动一格、直到到达棋盘的右下角。给定一个棋盘及其上面的礼物的价值,请计算你最多能拿到多少价值的礼物?
1 | //只求最后价值所以把原数组破坏了,如果不想就初始化一个新的全是0的dp数组。 |
46 把数字翻译成字符串
给定一个数字,我们按照如下规则把它翻译为字符串:0 翻译成 “a” ,1 翻译成 “b”,……,11 翻译成 “l”,……,25 翻译成 “z”。一个数字可能有多个翻译。请编程实现一个函数,用来计算一个数字有多少种不同的翻译方法。
数字每每一位或者两位判断,转换成字符串,然后进行处理。
1 | var translateNum = function(num){ |
48最长不含重复字符的子字符串
请从字符串中找出一个最长的不包含重复字符的子字符串,计算该最长子字符串的长度。
该**lastIndexOf()
**方法给定一个参数:要搜索的子字符串,搜索整个调用字符串,并返回指定子字符串最后一次出现的索引。给定第二个参数:一个数字,该方法返回指定子字符串在小于或等于指定数字的索引处的最后一次出现。
关于字符串的不重复字符串,需要不重复字符串的首位索引,本题最后只要求字符串的长度,所以直接pop。
1 | var lengthOfLongestSubstring = function(s) { |
双指针
18删除链表的节点
给定单向链表的头指针和一个要删除的节点的值,定义一个函数删除该节点。
返回删除后的链表的头节点。
注意:此题对比原题有改动
1 | function ListNode(val){ |
22 链表中倒数第k个节点
输入一个链表,输出该链表中倒数第k个节点。为了符合大多数人的习惯,本题从1开始计数,即链表的尾节点是倒数第1个节点。
例如,一个链表有 6 个节点,从头节点开始,它们的值依次是 1、2、3、4、5、6。这个链表的倒数第 3 个节点是值为 4 的节点。
1 | var getKthFromEnd = function(head,k){ |
25合并两个排序的链表
输入两个递增排序的链表,合并这两个链表并使新链表中的节点仍然是递增排序的。
1 | //普通方法 |
052两个链表的第一个公共节点
输入两个链表,找出它们的第一个公共节点。
1 | //A前 + 公共 + B前 = B前 + 公共 + A前 |
21调整数组顺序使奇数位于偶数前面
输入一个整数数组,实现一个函数来调整该数组中数字的顺序,使得所有奇数在数组的前半部分,所有偶数在数组的后半部分。
1 | //双指针:头尾遍历 |
057 和为s的两个数字
输入一个递增排序的数组和一个数字s,在数组中查找两个数,使得它们的和正好是s。如果有多对数字的和等于s,则输出任意一对即可。
1 | //双指针 |
058反转单词顺序
输入一个英文句子,翻转句子中单词的顺序,但单词内字符的顺序不变。为简单起见,标点符号和普通字母一样处理。例如输入字符串”I am a student. “,则输出”student. a am I”。
1 | var reverseWords = function(s){ |
搜索与回溯算法(中等)
12矩阵中的路径
给定一个 m x n 二维字符网格 board 和一个字符串单词 word 。如果 word 存在于网格中,返回 true ;否则,返回 false 。
单词必须按照字母顺序,通过相邻的单元格内的字母构成,其中“相邻”单元格是那些水平相邻或垂直相邻的单元格。同一个单元格内的字母不允许被重复使用。
1 | var exist = function(board,word){ |
13机器人的运动范围
地上有一个m行n列的方格,从坐标 [0,0] 到坐标 [m-1,n-1] 。一个机器人从坐标 [0, 0] 的格子开始移动,它每次可以向左、右、上、下移动一格(不能移动到方格外),也不能进入行坐标和列坐标的数位之和大于k的格子。例如,当k为18时,机器人能够进入方格 [35, 37] ,因为3+5+3+7=18。但它不能进入方格 [35, 38],因为3+5+3+8=19。请问该机器人能够到达多少个格子?
1 | //BFS |
34二叉树中和维某一值的路径
给你二叉树的根节点 root 和一个整数目标和 targetSum ,找出所有 从根节点到叶子节点 路径总和等于给定目标和的路径。
叶子节点 是指没有子节点的节点。
1 | var pathSum = function(root,target){ |
36二叉搜索树与双向链表
输入一棵二叉搜索树,将该二叉搜索树转换成一个排序的循环双向链表。要求不能创建任何新的节点,只能调整树中节点指针的指向。
1 | var treeToDoublyList = funciton(root){ |
54二叉搜索树的第k大节点
给定一颗二叉搜索树,请找出其中第k大的节点的值。
1 | var kehLargest = function(root,k){ |
排序
45把数组排成最小的数
输入一个非负整数数组,把数组里所有数字拼接起来排成一个数,打印能拼接出的所有数字中最小的一个。
1 | var minNumber = function(nums){ |
61扑克牌中的顺子
从若干副扑克牌中随机抽 5 张牌,判断是不是一个顺子,即这5张牌是不是连续的。2~10为数字本身,A为1,J为11,Q为12,K为13,而大、小王为 0 ,可以看成任意数字。A 不能视为 14。
1 | //常规方法, 相邻两个的差 和 0 的情况 |
40最小的k个数
输入整数数组 arr
,找出其中最小的 k
个数。例如,输入4、5、1、6、2、7、3、8这8个数字,则最小的4个数字是1、2、3、4。
1 | //最简单 |
41数据流中的中位数(困难) 没做————————
搜索与回溯算法(中等)
55 I 二叉树的深度
输入一棵二叉树的根节点,求该树的深度。从根节点到叶节点依次经过的节点(含根、叶节点)形成树的一条路径,最长路径的长度为树的深度。
1 | //1递归 |
55 II 平衡二叉树
输入一棵二叉树的根节点,判断该树是不是平衡二叉树。如果某二叉树中任意节点的左右子树的深度相差不超过1,那么它就是一棵平衡二叉树。
1 | //遍历到当前结点的左右子节点,AB为根节点的树的左右子树深度差值不超过一 |
64 求1+2+…+n
求 1+2+...+n
,要求不能使用乘除法、for、while、if、else、switch、case等关键字及条件判断语句(A?B:C)。
1 | //一行 前面的n是作为条件判断的 |
68I 二叉搜索树的最近公共祖先(二叉搜索树,有序)
给定一个二叉搜索树, 找到该树中两个指定节点的最近公共祖先。
百度百科中最近公共祖先的定义为:“对于有根树 T 的两个结点 p、q,最近公共祖先表示为一个结点 x,满足 x 是 p、q 的祖先且 x 的深度尽可能大(一个节点也可以是它自己的祖先)。”
1 | // |
68 II 二叉树的最近公共祖先(无序值的二叉树)
给定一个二叉树, 找到该树中两个指定节点的最近公共祖先。
1 | var lowestCommonAncestor = functino(root,p,q){ |
分治法(中等)
07重建二叉树
输入某二叉树的前序遍历和中序遍历的结果,请构建该二叉树并返回其根节点。
1 | //1 |
16 数值的整数次方
实现 pow(x, n) ,即计算 x 的 n 次幂函数(即,xn)。不得使用库函数,同时不需要考虑大数问题。
1 | var myPow = function(x,n){ |
33二叉搜索树的后序遍历
输入一个整数数组,判断该数组是不是某二叉搜索树的后序遍历结果。如果是则返回 true
,否则返回 false
。假设输入的数组的任意两个数字都互不相同。
1 | //二叉搜索树特点是右子树值永远大于左子树 |
位运算
15二进制中1的个数
编写一个函数,输入是一个无符号整数(以二进制串的形式),返回其二进制表达式中数字位数为 ‘1’ 的个数(也被称为 汉明重量).)。
1 | //按位与&,对应位数都为1则为1;,循环的条件:只要存在有1的情况n就不会为0;,退出循环时,说明n==0;,没循环一次count++; |
65 不用加减乘除做加法
写一个函数,求两个整数之和,要求在函数体内不得使用 “+”、“-”、“*”、“/” 四则运算符号。
1 | // |
56 I 数组中数字出现的次数——————
一个整型数组 nums
里除两个数字之外,其他数字都出现了两次。请写程序找出这两个只出现一次的数字。要求时间复杂度是O(n),空间复杂度是O(1)。
1 |
56 II 数组中数字出现的次数——————
1 |
数学(简单)
39数组中出现次数超过一半的数字
数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字。
1 | //方法1:快排后,返回nums[n/2] |
66. 构建乘积数组
给定一个数组 A[0,1,…,n-1],请构建一个数组 B[0,1,…,n-1],其中 B[i] 的值是数组 A 中除了下标 i 以外的元素的积, 即 B[i]=A[0]×A[1]×…×A[i-1]×A[i+1]×…×A[n-1]。不能使用除法。
1 | var constructArr = function(a){ |
数学(中等)
14-I 剪绳子
给你一根长度为 n 的绳子,请把绳子剪成整数长度的 m 段(m、n都是整数,n>1并且m>1),每段绳子的长度记为 k[0],k[1]…k[m-1] 。请问 k[0]k[1]…*k[m-1] 可能的最大乘积是多少?例如,当绳子的长度是8时,我们把它剪成长度分别为2、3、3的三段,此时得到的最大乘积是18。
1 | // |
57-II和为s的连续正数序列
输入一个正整数 target ,输出所有和为 target 的连续正整数序列(至少含有两个数)。
序列内的数字由小到大排列,不同序列按照首个数字从小到大排列。
1 | var findCotinuousSequence = function(target){ |
62 圆圈中最后剩下的数字
0,1,···,n-1这n个数字排成一个圆圈,从数字0开始,每次从这个圆圈里删除第m个数字(删除后从下一个数字开始计数)。求出这个圆圈里剩下的最后一个数字。
例如,0、1、2、3、4这5个数字组成一个圆圈,从数字0开始每次删除第3个数字,则删除的前4个数字依次是2、0、4、1,因此最后剩下的数字是3。
1 | var lastRemaining = function(n,m){ |
模拟(中等)
29 顺时针打印矩阵
输入一个矩阵,按照从外向里以顺时针的顺序依次打印出每一个数字。
1 | var spiralOrder = function(matrix){ |
剑指offerII
001 整数除法
给定两个整数 a
和 b
,求它们的除法的商 a/b
,要求不得使用乘号 '*'
、除号 '/'
以及求余符号 '%'
。
注意:
整数除法的结果应当截去(truncate)其小数部分,例如:truncate(8.345) = 8 以及 truncate(-2.7335) = -2
假设我们的环境只能存储 32 位有符号整数,其数值范围是 [−2^31, 2^31−1]。本题中,如果除法结果溢出,则返回 2^31 − 1
先把特殊情况排除,然后最先判断正负号,然后把负负情况改为正。创建结果。 用减法代替除法进行处理。
1 | var divide = function(a,b){ |