数组中重复的数字
在一个长度为n的数组里的所有数字都在0到n-1的范围内。 数组中某些数字是重复的,但不知道有几个数字是重复的。也不知道每个数字重复几次。请找出数组中任意一个重复的数字。 例如,如果输入长度为7的数组{2,3,1,0,2,5,3},那么对应的输出是第一个重复的数字2。
思路:
哈希表,时间复杂度为O(N),空间复杂度为O(N)
数组中数字都在0到n-1范围内。如果没有重复数字,则每个数字对应其下标。时间复杂度为O(N),空间复杂度为O(1)
1 | public boolean duplicate(int numbers[],int length,int [] duplication) { |
二维数组中的查找
在一个二维数组中(每个一维数组的长度相同),每一行都按照从左到右递增的顺序排序,每一列都按照从上到下递增的顺序排序。请完成一个函数,输入这样的一个二维数组和一个整数,判断数组中是否含有该整数。
思路:先检查数组左上角,若大于target,则排除最左列,若小于target,则排除最顶行。
1 | public boolean Find(int target, int [][] array) { |
替换空格
请实现一个函数,将一个字符串中的每个空格替换成“%20”。例如,当字符串为We Are Happy.则经过替换之后的字符串为We%20Are%20Happy。
思路:先遍历计算空格数量,从后到前移动字符(以保证所有字符只移动一次)。
1 | public String replaceSpace(StringBuffer str) { |
从尾到头打印链表
输入一个链表,按链表值从尾到头的顺序返回一个ArrayList。
思路:栈的特点是先进后出
1 | public ArrayList<Integer> printListFromTailToHead(ListNode listNode) { |
重建二叉树
输入某二叉树的前序遍历和中序遍历的结果,请重建出该二叉树。假设输入的前序遍历和中序遍历的结果中都不含重复的数字。例如输入前序遍历序列{1,2,4,7,3,5,6,8}和中序遍历序列{4,7,2,1,5,3,8,6},则重建二叉树并返回。
思路:前序遍历的第一个值为根节点的值,这个值将中序遍历结果分成两部分,递归。
具体细节:利用HashMap快速查找中序遍历数组的下标
1 | Map<Integer, Integer> inMap = new HashMap<>(); |
二叉树的下一个节点
给定一个二叉树和其中的一个结点,请找出中序遍历顺序的下一个结点并且返回。注意,树中的结点不仅包含左右子结点,同时包含指向父结点的指针。
思路:分两种情况:1.该节点的右节点不为null时,寻找该右节点的最左子节点/该节点的右节点。2.该节点的右节点为null时,寻找第一个左链接指向的节点包含该节点的祖先节点
1 | public TreeLinkNode GetNext(TreeLinkNode pNode){ |
用两个栈实现队列
用两个栈来实现一个队列,完成队列的Push和Pop操作。 队列中的元素为int类型。
思路:一个出栈,一个入栈。若出栈为空,则将入栈倒入到出栈中。
1 | Stack<Integer> in = new Stack<Integer>(); |
斐波那契数列
大家都知道斐波那契数列,现在要求输入一个整数n,请你输出斐波那契数列的第n项(从0开始,第0项为0,第1项为1)。
思路:递归效率太低,且占用大量栈内存,使用动态规划来做。
1 | public int Fibonacci(int n) { |
旋转数组的最小数字
把一个数组最开始的若干个元素搬到数组的末尾,我们称之为数组的旋转。 输入一个非减排序的数组的一个旋转,输出旋转数组的最小元素。 例如数组{3,4,5,1,2}为{1,2,3,4,5}的一个旋转,该数组的最小值为1。 NOTE:给出的所有元素都大于0,若数组大小为0,请返回0。
思路:二分法,若mid<high,后半为升序,最小值在前半。否则,最小值在后半。还要考虑low、mid、high位置的值相等的情况使用顺序查找。
1 | public int minNumberInRotateArray(int [] array) { |
矩阵中的路径
请设计一个函数,用来判断在一个矩阵中是否存在一条包含某字符串所有字符的路径。路径可以从矩阵中的任意一个格子开始,每一步可以在矩阵中向左,向右,向上,向下移动一个格子。如果一条路径经过了矩阵中的某一个格子,则之后不能再次进入这个格子。 例如 a b c e s f c s a d e e 这样的3 X 4 矩阵中包含一条字符串”bcced”的路径,但是矩阵中不包含”abcb”路径,因为字符串的第一个字符b占据了矩阵中的第一行第二个格子之后,路径不能再次进入该格子。
思路:通过二维数组{{0, -1}, {0, 1}, {-1, 0}, {1, 0}}来上下左右移动,boolean数组判断是否走过
1 | private final static int[][] next = {{0, -1}, {0, 1}, {-1, 0}, {1, 0}}; |
机器人的运动范围
地上有一个m行和n列的方格。一个机器人从坐标0,0的格子开始移动,每一次只能向左,右,上,下四个方向移动一格,但是不能进入行坐标和列坐标的数位之和大于k的格子。 例如,当k为18时,机器人能够进入方格(35,37),因为3+5+3+7 = 18。但是,它不能进入方格(35,38),因为3+5+3+8 = 19。请问该机器人能够达到多少个格子?
1 | static boolean[][] marked; |
剪绳子
把一根绳子剪成多段,并且使得每段的长度乘积最大。
思路:贪婪算法:尽可能多剪长度为 3 的绳子,并且不允许有长度为 1 的绳子出现。如果出现了,就从已经切好长度为 3 的绳子中拿出一段与长度为 1 的绳子重新组合,把它们切成两段长度为 2 的绳子。
1 | public int integerBreak(int n) { |
二叉树的镜像
思路:先递归遍历二叉树,再交换左右节点,返回当前根节点
1 | public TreeNode Mirror(TreeNode root) { |
链表中环的入口节点
思路:快慢指针,指针指向地址相等时,将其中一个指针指向头节点,两指针速度相等遍历,相遇时的节点就是环的入口节点
1 | public ListNode EntryNodeOfLoop(ListNode pHead){ |