解题思路:

对于每家店,有选和不选两种选择

那选还是不选取决于我当前是否能取得最大价值

第一步:画搜索树

屏幕截图 2024-02-19 145854.png
第二步:暴力DFS

关键:找到递归边界值和递归公式

递归公式:dfs(n)=max(dfs(n+1),dfs(n+2)+v[n])

递归边界:dfs(n+1)=dfs(n+2)=0

参考代码:

import java.io.*;
public class Main {
    static BufferedReader br=new BufferedReader(new InputStreamReader(System.in));
    static PrintWriter pw=new PrintWriter(new OutputStreamWriter(System.out));
    static int shop;
    static long values[];
    public static void main(String[] args) throws IOException {
        int test=Integer.parseInt(br.readLine());
        while(test>0) {
            shop = Integer.parseInt(br.readLine());
            values = new long[shop + 1];
            String[] s = br.readLine().split(" ");
            for (int i = 1; i shop) {
            return 0;//递归的底
        }
         return Math.max(dfs(n+1),dfs(n+2)+values[n]);
    }
}

运行结果:超时

那肯定得优化

由搜索树发现,红色框内的店铺不需要继续深搜,因为在左支中已经搜过了


第三步:记忆化搜索

关键:记录之前搜索店铺i的结果

用map[x]记录从1到x店铺的最大价值sum,若map[x]!=0,返回map[x]

参考代码:

import java.io.*;
 
public class Main {
    static BufferedReader br=new BufferedReader(new InputStreamReader(System.in));
    static PrintWriter pw=new PrintWriter(new OutputStreamWriter(System.out));
    static int shop;
    static long[]map;
    static long values[];
    public static void main(String[] args) throws IOException {
        int test=Integer.parseInt(br.readLine());
        while(test>0) {
            shop = Integer.parseInt(br.readLine());
            values = new long[shop + 1];
             map= new long[3 + shop];
            String[] s = br.readLine().split(" ");
            for (int i = 1; i shop) {
            sum=0;
        }else{
            sum=Math.max(dfs(n+1),dfs(n+2)+values[n]);
        }
        map[n]=sum;
        return sum;
    }
}

运行结果:运行错误

代码没问题,其他平台能跑


第四步:递推

关键:找到递推公式,由小推到大

递推公式:dp[i]=max(dp[i-1],dp[i-2]+values[i])

映射:因为遍历从1开始,防止下标越界,则下标偏移2

公式变成:dp[i+2]=max(dp[i+1],dp[i]+values[i])

参考代码:

import java.io.*;
 
public class Main {
    static BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
    static PrintWriter pw = new PrintWriter(new OutputStreamWriter(System.out));
    static int shop;
    static long values[];
    public static void main(String[] args) throws IOException {
        int test = Integer.parseInt(br.readLine());
        while (test > 0) {
            shop = Integer.parseInt(br.readLine());
            values = new long[shop + 1];
            String[] s = br.readLine().split(" ");
            for (int i = 1; i <= shop; i++) {
                values[i] = Integer.parseInt(s[i - 1]);
            }
            long []dp=new long[shop+3];//dp[i]存1-i家店洗劫后取得的最大价值
            for (int i = 1; i<=shop; i++) {
                dp[i+2]=Math.max(dp[i+1],dp[i]+values[i]);
            }
            pw.println(dp[shop+2]);
            pw.flush();
            test--;
        }
    }
}

运行结果:100分  运行时间: 1585ms    消耗内存: 50880KB


进一步优化:

因为递推公式只涉及i,i-1,i-2三个数

每次循环只需要三个值就可以得出此次循环的答案,不需要用dp数组记录每次循环的结果

用三个变量newf,tmp1,tmp2记录每次循环的结果,并在循环里更新newf,tmp1,tmp2方便下次计算

主要优化空间:数组→变量

参考代码:

import java.io.*;
 
/**
 * @Author:杨雨彤
 * @date:2024/2/18 10:00
 */
public class Main {
    static BufferedReader br=new BufferedReader(new InputStreamReader(System.in));
    static PrintWriter pw=new PrintWriter(new OutputStreamWriter(System.out));
    static int shop;
    static long values[];
    public static void main(String[] args) throws IOException {
        int test=Integer.parseInt(br.readLine());
        while(test>0) {
            shop = Integer.parseInt(br.readLine());
            values = new long[shop + 1];
            String[] s = br.readLine().split(" ");
            for (int i = 1; i <= shop; i++) {
                values[i] = Integer.parseInt(s[i - 1]);
            }
            long newf = 0, temp1 = 0, temp2 = 0;
            for (int i = 1; i <= shop; i++) {
                newf = Math.max(temp1, temp2 + values[i]);
                temp2 = temp1;
                temp1 = newf;
            }
            pw.println(newf);
            pw.flush();
            test --;
        }
    }
 
}

运行结果:100分 运行时间: 1858ms    消耗内存: 46252KB


最后:这道题是跟着B站上一位up主写的。

每次想学dp都无所适从,到底从哪开始呢?状态转移方程怎么推出来的?

在无数次探索中终于找到了答案

那就是暴力DFS→记忆化搜索→递推(DP)

动态规划也分很多种,线性,树形..但没关系只要找到门还怕什么!!

最后的最后:

其实我一开始的思路不是这样的,对每个店铺都有选和不选

套用01背包的思想,店铺dp[i][0]表示第i家店不偷时,洗劫前i家店铺所能取的最大价值

dp[i][1]表示第i家店偷时,洗劫前i家店能取得的最大价值

最后的答案就在,dp[n][0]和dp[n][1]中取较大的

对于dp[i][0]递推公式为:dp[i][0]=max(dp[i-1][1],dp[i-1][0])

理解:如果第i家店不偷,那么第i-1家店偷不偷都可以,最后取决于dp[i-1][0]和dp[i-1][1]谁大

对于dp[i][1]递推公式为:dp[i][1]=max(dp[i-1][0],dp[i-2][1])

理解:如果第i家店偷,要么第i-1家店不能偷,要么第i-2家店被偷,从而只能偷i了


注意事项:

dp[i][1]既然偷了第i家店,那么dp[i][1]初始值就是该店铺的价值values[i]

递推的边界:dp[1][0]=0,dp[1][1]=values[1];

参考代码:

import java.beans.Visibility;
import java.io.*;
 
/**
 * @Author:杨雨彤
 * @date:2024/2/18 10:00
 */
public class Main {
    static BufferedReader br=new BufferedReader(new InputStreamReader(System.in));
    static PrintWriter pw=new PrintWriter(new OutputStreamWriter(System.out));
    static int shop;
    static long values[];
    public static void main(String[] args) throws IOException {
        int test=Integer.parseInt(br.readLine());
        while(test>0) {
            shop = Integer.parseInt(br.readLine());
            values = new long[shop + 1];
            String[] s = br.readLine().split(" ");
            for (int i = 1; i <= shop; i++) {
                values[i] = Integer.parseInt(s[i - 1]);
            }
        long [][]dp=new long[shop+1][2];//dp[i][0]表示第i家商店不偷时,前i个店铺的最大价值
            //dp[i][1]表示第i家店偷时,前i家店铺的最大价值,初始值就是第i家店铺偷时所取得的价值
        dp[1][0]=0;//第1家店不偷
        for (int i = 1; i <=shop ; i++) {
            dp[i][1]=values[i];
        }
        for (int i = 2; i <=shop ; i++) {
           dp[i][0]+=Math.max(dp[i-1][0],dp[i-1][1]);
           dp[i][1]+=Math.max(dp[i-1][0],dp[i-2][1]);
        }
        pw.println(Math.max(dp[shop][0],dp[shop][1]));
        pw.flush();
            test --;
        }
    }
}

运行结果:100分 运行时间: 1714ms    消耗内存: 52512KB

点赞(0)
 

0.0分

1 人评分

C语言网提供由在职研发工程师或ACM蓝桥杯竞赛优秀选手录制的视频教程,并配有习题和答疑,点击了解:

一点编程也不会写的:零基础C语言学练课程

解决困扰你多年的C语言疑难杂症特性的C语言进阶课程

从零到写出一个爬虫的Python编程课程

只会语法写不出代码?手把手带你写100个编程真题的编程百练课程

信息学奥赛或C++选手的 必学C++课程

蓝桥杯ACM、信息学奥赛的必学课程:算法竞赛课入门课程

手把手讲解近五年真题的蓝桥杯辅导课程

评论列表 共有 0 条评论

暂无评论