1. 打家劫舍问题
1.1 题目一:198. 打家劫舍
- 题目描述
- 解题思路
1)设定dp数组,明确含义:在第i号屋进行作案时,手上能够偷到的最大金额
2)dp在对应的每个房子时存在两种可能性
(1):偷第i户房子:dp[ i ] = dp[ i-2 ]+ nums[ i ];因为前面一户肯定不能偷,故加上前面两户的地方。
(2):不偷第i户房子:dp[ i ] = dp[ i-1 ];
(3):比较两者取最大值:dp[ i ] =Math.max(dp[ i-2 ]+ nums[ i ], dp[ i-1 ])
- 注意点:
(1):对于i=1的情况,需要提前讨论
(2):首先由于在迭代中有-2操作,故i需要从2开始,即i=0和1需要进行初始化操作
(3):在最后输出的时候,返回的是dp[nums.length-1] ,即最后一个位置的金额大小。
- 代码题解
public int rob(int[] nums) {
if(nums.length==1) return nums[0];
int[] dp=new int[nums.length];
dp[0]=nums[0];
dp[1]=Math.max(nums[0],nums[1]);
for(int i=2;i<nums.length;i++){
dp[i]=Math.max(dp[i-2]+nums[i],dp[i-1]);
}
return dp[nums.length-1];
}
1.2 题目二:213. 打家劫舍 II
- 题目描述
- 解题思路
1)这一题和上一题的区别是,房间围成了一个环,如果还是采用上一题的思路会出现一种问题
(1):如果头部被偷了,尾部也被偷了,那么就不符合要求
2)针对这种问题,正面去解答,满足要求的有三种:
(1):首偷尾不偷,
(2):首不偷尾偷,
(3):首尾都不偷,其中前面说的偷是存在偷的可能性,不是一定要偷,故该情况是包含在前面两种情况中的
3)解答方法
(1):首先是写一个通用的方法,根据数组,能够得到对应偷得的最大金额。
(2):把首偷尾不偷,首不偷尾偷两种情况考虑一下,得到最大值 - 代码
public int rob(int[] nums) {
if(nums.length==1) return nums[0];
return Math.max( method(nums,0,nums.length-1), method(nums,1,nums.length) );
}
public static int method(int[] nums,int head, int end){
int[] data=new int[nums.length-1];
for(int i=head,j=0;i<end;i++) data[j++]=nums[i];
System.out.println("data="+Arrays.toString(data));
int[] dp=new int[data.length+1];
dp[1]=data[0];
for(int i=2;i<=data.length;i++){
dp[i]=Math.max(dp[i-2]+data[i-1],dp[i-1]);
}
return dp[data.length];
}
1.3 题目三:237. 打家劫舍 III
- 题目描述
- 解题思路
1)这一题的核心是一个树层的遍历???