2245 2018-11-19 2020-06-25
前言:牛客网剑指Offer编程题25~30题。
一、复杂链表的复制
1、问题描述
考点分解让复杂问题简单,输入一个复杂链表(每个节点中有节点值,以及两个指针,一个指向下一个节点,另一个特殊指针指向任意一个节点),返回结果为复制后复杂链表的head(注意,输出结果中请不要返回参数中的节点引用,否则判题程序会直接返回空)。
2、我的代码
static class TwentyFive {
static class RandomListNode {
int label;
RandomListNode next = null;
RandomListNode random = null;
RandomListNode(int label) {
this.label = label;
}
}
public RandomListNode Clone(RandomListNode pHead) {
if (pHead == null) {
return null;
}
RandomListNode node = new RandomListNode(pHead.label);
if (pHead.random != null) {
node.random = new RandomListNode(pHead.random.label);
}
if (pHead.next != null) {
node.next = Clone(pHead.next);
}
return node;
}
}
3、python二刷
class RandomListNode:
def __init__(self, x):
self.label = x
self.next = None
self.random = None
class Solution:
# 返回 RandomListNode
def Clone(self, pHead):
if not pHead:
return
head = pHead
_list = list()
while head is not None:
root = RandomListNode(head.label)
if _list:
_list[-1].next = root
_list.append(root)
head = head.next
i = 0
head = pHead
while i < len(_list):
random_node = head.random
if random_node is not None:
p = pHead
j = 0
while p is not random_node:
p = p.next
j += 1
_list[i].random = _list[j]
i += 1
head = head.next
return _list[0]
二、二叉搜索树与双向链表
1、问题描述
考点分解让复杂问题简单,输入一棵二叉搜索树,将该二叉搜索树转换成一个排序的双向链表。要求不能创建任何新的结点,只能调整树中结点指针的指向。
2、我的代码
static class TwentySix {
static class TreeNode {
int val = 0;
TreeNode left = null;
TreeNode right = null;
public TreeNode(int val) {
this.val = val;
}
}
public TreeNode Convert(TreeNode pRootOfTree) {
if (pRootOfTree == null) {
return null;
}
TreeNode center = link(pRootOfTree);
while (center.left != null) {
center = center.left;
}
return center;
}
public TreeNode link(TreeNode pRootOfTree) {
TreeNode begin = null, end = null;
if (pRootOfTree.left != null) {
begin = link(pRootOfTree.left);
begin = begin.right == null ? begin : begin.right;
// System.out.println(pRootOfTree.left.val + "通过begin返回值为:" + begin.val);
}
if (begin != null) {
begin.right = pRootOfTree;
pRootOfTree.left = begin;
// System.out.println(begin.val + "连接" + pRootOfTree.val);
}
if (pRootOfTree.right != null) {
end = link(pRootOfTree.right);
end = end.left == null ? end : end.left;
// System.out.println(pRootOfTree.right.val + "通过end返回值为:" + end.val);
}
if (end != null) {
pRootOfTree.right = end;
end.left = pRootOfTree;
// System.out.println(pRootOfTree.val + "连接" + end.val);
}
return pRootOfTree;
}
private static void test() {
TreeNode node1 = new TreeNode(6);
TreeNode node2 = new TreeNode(4);
TreeNode node3 = new TreeNode(8);
TreeNode node4 = new TreeNode(3);
TreeNode node5 = new TreeNode(5);
TreeNode node6 = new TreeNode(7);
TreeNode node7 = new TreeNode(9);
node1.left = node2;
node1.right = node3;
node2.left = node4;
node2.right = node5;
node3.left = node6;
node3.right = node7;
TreeNode root = new TwentySix().Convert(node1);
TreeNode temp = null;
while (root != null) {
System.out.print(root.val + " ");
temp = root;
root = root.right;
}
System.out.println();
while (temp != null) {
System.out.print(temp.val + " ");
temp = temp.left;
}
}
期间把link方法错写成Convert,浪费了我将近一个小时查错,查了一遍又一遍,一直在说不科学啊。总的来说,还是采用递归的思想,大问题化小问题。
3、别人的代码
TreeNode head = null;
TreeNode realHead = null;
public TreeNode Convert1(TreeNode pRootOfTree) {
ConvertSub(pRootOfTree);
return realHead;
}
private void ConvertSub(TreeNode pRootOfTree) {
if(pRootOfTree==null) return;
ConvertSub(pRootOfTree.left);
if (head == null) {
head = pRootOfTree;
realHead = pRootOfTree;
} else {
head.right = pRootOfTree;
pRootOfTree.left = head;
head = pRootOfTree;
}
ConvertSub(pRootOfTree.right);
}
他采用的是中序遍历,不得不说这是一种好方法啊,我怎么没想到呢。
4、python二刷
class Solution:
def Convert(self, pRootOfTree):
if not pRootOfTree:
return
root = in_order(pRootOfTree, None)
while root.left:
root = root.left
return root
def in_order(root, right):
if not root:
return
if not root.left and not root.right:
return root
if root.left:
root.left = in_order(root.left, True)
root.left.right = root
if root.right:
root.right = in_order(root.right, False)
root.right.left = root
if right:
return root.right if root.right else root
else:
return root.left if root.left else root
三、字符串的排列
1、问题描述
考点分解让复杂问题简单,输入一个字符串,按字典序打印出该字符串中字符的所有排列。例如输入字符串abc,则打印出由字符a,b,c所能排列出来的所有字符串abc,acb,bac,bca,cab和cba。输入一个字符串,长度不超过9(可能有字符重复),字符只包括大小写字母。
2、别人的代码
static class TwentySeven {
public ArrayList<String> Permutation(String str) {
ArrayList<String> res = new ArrayList<>();
if (str != null && str.length() > 0) {
PermutationHelper(str.toCharArray(), 0, res);
Collections.sort(res);
}
return res;
}
public void PermutationHelper(char[] cs, int i, List<String> list) {
if (i == cs.length - 1) {
String val = String.valueOf(cs);
if (!list.contains(val)) {
list.add(val);
}
} else {
for (int j = i; j < cs.length; j++) {
swap(cs, i, j);
PermutationHelper(cs, i + 1, list);
swap(cs, i, j);
}
}
}
public void swap(char[] cs, int i, int j) {
char temp = cs[i];
cs[i] = cs[j];
cs[j] = temp;
}
private static void test() {
String str = "abc";
String str1 = "aabc";
ArrayList<String> list = new TwentySeven().Permutation(str);
System.out.println(list);
list = new TwentySeven().Permutation(str1);
System.out.println(list);
}
}
他的思路我理解了,但是需要多回顾一下,需要好好理解,过几天再来做一次。
3、python二刷
class Solution:
def Permutation(self, ss):
all_list = list()
if not ss:
return all_list
_str = ''
[i for i in ss].sort()
pailie(ss, _str, all_list)
return all_list
def pailie(ss, _str, all_list):
if len(_str) != len(ss):
for i, v in enumerate(ss):
if str(i) not in _str:
_str += str(i)
pailie(ss, _str, all_list)
_str = _str[0:len(_str) - 1]
if len(_str) == len(ss):
str_copy = ''
for i in _str:
str_copy += ss[int(i)]
if str_copy not in all_list:
all_list.append(str_copy)
还是觉得自己的代码思路清晰一些。这种问题就是典型的动态规划,非常类似八皇后问题。
四、数组中出现次数超过一半的数字
1、问题描述
考点时间效率,数组中有一个数字出现的次数超过数组长度的一半,请找出这个数字。例如输入一个长度为9的数组{1,2,3,2,2,2,5,4,2}。由于数字2在数组中出现了5次,超过数组长度的一半,因此输出2。如果不存在则输出0。
2、我的代码
static class TwentyEight {
public int MoreThanHalfNum_Solution(int [] array) {
int size = array.length / 2;
int[] arr = new int[array.length];
int index = 0;
for (int i = 0; i < array.length; i++) {
for (int j = 0; j <= index; j++) {
if (j == index) {
arr[index++]++;
break;
}
if (array[i] == array[j]) {
if (++arr[j] > size) {
return array[j];
}
break;
}
}
}
return size != 1 ? 0 : array[0];
}
private static void test() {
int[] arr = new int[]{1,2,3,2,4,2,5,2,3};
int[] arr1 = new int[]{1,3,4,5,2,2,2,2,2};
System.out.println(new TwentyEight().MoreThanHalfNum_Solution(arr1));
}
}
3、python二刷
class Solution:
def MoreThanHalfNum_Solution(self, numbers):
size = len(numbers) // 2
arr = []
i = 0
while i < len(numbers):
for j, v in enumerate(arr):
if numbers[i] == numbers[j]:
arr[j] += 1
if arr[j] > size:
return numbers[j]
continue
arr.append(1)
i += 1
return 0 if len(numbers) != 1 else numbers[0]
五、最小的K个数
1、问题描述
考点时间效率,输入n个整数,找出其中最小的K个数。例如输入4,5,1,6,2,7,3,8这8个数字,则最小的4个数字是1,2,3,4。
2、我的思路
一种简单的思路就是先排序,取前面k个。另外一种思路是设置一个容量为k的容器(如数组),使其有序排列。
剑指Offer上面提到了用最大堆和红黑树。
static class TwentyNine {
public ArrayList<Integer> GetLeastNumbers_Solution(int [] input, int k) {
ArrayList<Integer> list = new ArrayList<>();
if (input == null || input.length == 0 || k == 0 || k > input.length) {
return list;
}
quickSort(input, 0, input.length - 1);
for (int i = 0; i < k; i++) {
list.add(input[i]);
}
return list;
}
private void quickSort(int[] input, int start, int end) {
if (start == end || start > end) {
return;
}
int index = start;
for (int i = start; i <= end; i++) {
if (input[i] < input[index]) {
int temp = i;
int min = input[i];
while (temp != index) {
input[temp] = input[temp - 1];
temp--;
}
input[index++] = min;
}
}
quickSort(input, start, index - 1);
quickSort(input, index + 1, end);
}
private static void test() {
int[] arr = new int[]{5,8,7,6,9,4,3,2,1};
System.out.println(new TwentyNine().GetLeastNumbers_Solution(arr, arr.length));
}
}
哈哈哈,我的算法功底上来了,手撕快排,没有复习的哟。
3、python二撕快拍
# 手撕快排
def sort(arr):
quick_sort(arr, 0, len(arr) - 1)
return arr
def quick_sort(arr, p, q):
if p >= q:
return
i = p + 1
j = q
mid = arr[p]
mid_index = p
while i <= j:
if arr[i] < mid:
insert(arr, p, i)
mid_index += 1
i += 1
quick_sort(arr, p, mid_index - 1)
quick_sort(arr, mid_index + 1, q)
# 把位置j的值插入到位置i之前
def insert(arr, i, j):
t = arr[j]
while i < j:
arr[j] = arr[j - 1]
j -= 1
arr[i] = t
if __name__ == '__main__':
arr = sort([1, 3, 6, 2, 4, 5, 7, 9, 8])
print(arr)
六、连续子数组的最大和
1、问题描述
考点时间效率,输入一个整形数组,数组里有正数也有负数。数组中的一个或连续多个整数组成一个子数组。求所有子数组的和的最大值。例如输入数组{1, -2, 3, 10, -4, 7, 2, -5},和最大的子数组为{3, 10, -4, 7, 2},因此输出为该子数组的和18。
2、我的代码
我又放弃了,我是真的菜。
3、别人的代码
static class Thirty {
public int FindGreatestSumOfSubArray(int[] array) {
if (array == null || array.length == 0) {
return 0;
}
int max = array[0];
int sum = array[0];
for (int i = 1; i < array.length; i++) {
int cur = array[i];
// 当前元素大于连续子数组和加上元素本身 && 最大值比元素还小时
// 形如1, -2, 3抛弃前面的连续子数组,重新开始计算连续数组和
if (cur > (sum + cur) && max < cur) {
sum = cur;
max = cur;
} else if (sum + cur > max) {
// 加上当前元素后,数组和比最大值还大,则连续该元素
max = sum + cur;
sum += cur;
} else {
sum += cur;
}
System.out.println("经过第" + i + "个(" + cur + ")后,sum=" + sum + ",max=" + max);
}
return max;
}
public int FindGreatestSumOfSubArray1(int[] array) {
if (array == null || array.length == 0) {
return 0;
}
int curSum = array[0];
int maxSum = curSum;
for (int i = 1; i < array.length; i++) {
if (curSum <= 0) {
curSum = array[i];
} else {
curSum += array[i];
}
if (curSum > maxSum) {
maxSum = curSum;
}
}
return maxSum;
}
/**
* 动态规划解法
* 返回f(i)={f(i-1)+array[i],array[i]},两者的最大值,如果f(i-1)<0,则选的是array[i]
*/
public int FindGreatestSumOfSubArray2(int[] array) {
if (array == null || array.length == 0) {
return 0;
}
int result = array[0];
int currentResult = array[0];
for (int i = 1; i < array.length; i++) {
currentResult = (currentResult + array[i]) > array[i] ? currentResult + array[i] : array[i];
result = currentResult > result ? currentResult : result;
}
return result;
}
private static void test() {
int[] arr = new int[]{1, -2, 3, 10, -4, 7, 2, -5};
System.out.println(new Thirty().FindGreatestSumOfSubArray1(arr));
}
}
后面再来多多回顾吧,6道题花了两天。
4、python二刷
class Solution:
def FindGreatestSumOfSubArray(self, array):
if not array:
return 0
sum, max = 0, array[0]
for cur in array:
if sum < cur and sum < 0:
sum = cur
else:
sum += cur
if sum > max:
max = sum
return max
这次不用半个小时就搞定了哦。
总访问次数: 319次, 一般般帅 创建于 2018-11-19, 最后更新于 2020-06-25
欢迎关注微信公众号,第一时间掌握最新动态!