【算法刷题】—7.15动态规划[DP],体验动态规划的魅力

简介: ✨今日算法三题1.连续子数组的最大和2.最长递增子序列3.统计放置房子的方式数

✨今日算法三题


1.连续子数组的最大和

2.最长递增子序列

3.统计放置房子的方式数


文章目录


1.连续子数组的最大和


题目描述

思路详解


本题的思路较为简单,本题只询问最大子数组的值,并没有说具体求出是哪个子数组,那么我们就可以动态规划来解题。

我们遍历数组进行累加,判断加上后边的这个数是否比之前的子数组大,两者取较大的,然年再比较之前保存的最大结果与当前所得子数组值的大小,取最大值就可以了。


代码与结果

class Solution {
    public int maxSubArray(int[] nums) {
        int pre = 0, maxAns = nums[0];
        for (int x : nums) {
            pre = Math.max(pre + x, x);
            maxAns = Math.max(maxAns, pre);
        }
        return maxAns;
    }
}


2.最长递增子序列


题目描述

思路详解


这里的思路,我们维护一个与数组nums等长的数组dp,用来保存与之对应的最大增序列的结果,我们来两层循环,第一层我们遍历数组 i ,同时把dp对应的也赋值为1,进入第二层循 环 j ,本层循环从nums[0]开始到nums[i]的前一个为止。我们进行比较第二层循环的值是否小于nums[i],如果小的话就把dp[j]的值加1,与dp[i]进行比较,取大值。同时我们保持结果一直是最大就好。


注意,这里可能会有小伙伴有问题,其实呢,我们每次循环都能找出当前以 i 为结尾的增长子序列的最大值并且保存到dp[i]中,那么当二层循环的时候,如果nums[j] < nums[i] 那么不就是以 i 为结尾的增长子序列又长了 1 嘛,因为 i 在 j 的后面,j 本身就是那么长,nums[i] > nums[j] ,那么自然 以 i 为结尾子序列最大长度 就是 dp[j] + 1 了嘛,(记得是(dp[i] , dp[i] + 1)取最大值哦


代码与结果

class Solution {
    public int lengthOfLIS(int[] nums) {
        if (nums.length == 0) {
            return 0;
        }
        int[] dp = new int[nums.length];
        dp[0] = 1;
        int maxans = 1;
        for (int i = 1; i < nums.length; i++) {
            dp[i] = 1;
            for (int j = 0; j < i; j++) {
                if (nums[i] > nums[j]) {
                    dp[i] = Math.max(dp[i], dp[j] + 1);
                }
            }
            maxans = Math.max(maxans, dp[i]);
        }
        return maxans;
    }
}


3.统计放置房子的方式数


题目描述

思路详解


单独考虑一侧的房子,定义f[i] 表示前 i 个地块的放置方案数,其中第 i 个地块可以放房子,也可以不放房子。

考虑第 i 个地块:

若不放房子,那么第i−1 个地块可放可不放,则有 f[i] = f[i-1];

若放房子,那么第 i−1 个地块无法放房子,第i−2 个地块可放可不放,则有 f[i]=f[i−2]。

因此

f[i]=f[i−1]+f[i−2]

边界为

f[0]=1,空也是一种方案;

f[1]=2,放与不放两种方案。

由于两侧的房屋互相独立,根据乘法原理,答案为f[n] ** 2。


代码与结果

class Solution {
    static final int MOD = (int) 1e9 + 7, MX = (int) 1e4 + 1;
    static final int[] f = new int[MX];
    static {
        f[0] = 1;
        f[1] = 2;
        for (var i = 2; i < MX; ++i)
            f[i] = (f[i - 1] + f[i - 2]) % MOD;
    }
    public int countHousePlacements(int n) {
        return (int) ((long) f[n] * f[n] % MOD);
    }
}


✨总结


今天练习了动态规划,动态规划一开始就是用f[i]来表示以i为结尾的满足条件的结果数,之后根据条件修改结果。想熟练运用的话,还是要多多练习哦!!!


相关文章
|
2月前
|
算法 Python
在Python编程中,分治法、贪心算法和动态规划是三种重要的算法。分治法通过将大问题分解为小问题,递归解决后合并结果
在Python编程中,分治法、贪心算法和动态规划是三种重要的算法。分治法通过将大问题分解为小问题,递归解决后合并结果;贪心算法在每一步选择局部最优解,追求全局最优;动态规划通过保存子问题的解,避免重复计算,确保全局最优。这三种算法各具特色,适用于不同类型的问题,合理选择能显著提升编程效率。
69 2
|
3月前
|
算法
动态规划算法学习三:0-1背包问题
这篇文章是关于0-1背包问题的动态规划算法详解,包括问题描述、解决步骤、最优子结构性质、状态表示和递推方程、算法设计与分析、计算最优值、算法实现以及对算法缺点的思考。
131 2
动态规划算法学习三:0-1背包问题
|
3月前
|
算法
动态规划算法学习四:最大上升子序列问题(LIS:Longest Increasing Subsequence)
这篇文章介绍了动态规划算法中解决最大上升子序列问题(LIS)的方法,包括问题的描述、动态规划的步骤、状态表示、递推方程、计算最优值以及优化方法,如非动态规划的二分法。
87 0
动态规划算法学习四:最大上升子序列问题(LIS:Longest Increasing Subsequence)
|
3月前
|
算法
动态规划算法学习二:最长公共子序列
这篇文章介绍了如何使用动态规划算法解决最长公共子序列(LCS)问题,包括问题描述、最优子结构性质、状态表示、状态递归方程、计算最优值的方法,以及具体的代码实现。
201 0
动态规划算法学习二:最长公共子序列
|
3月前
|
存储 算法
动态规划算法学习一:DP的重要知识点、矩阵连乘算法
这篇文章是关于动态规划算法中矩阵连乘问题的详解,包括问题描述、最优子结构、重叠子问题、递归方法、备忘录方法和动态规划算法设计的步骤。
211 0
|
3月前
|
数据可视化 搜索推荐 Python
Leecode 刷题笔记之可视化六大排序算法:冒泡、快速、归并、插入、选择、桶排序
这篇文章是关于LeetCode刷题笔记,主要介绍了六大排序算法(冒泡、快速、归并、插入、选择、桶排序)的Python实现及其可视化过程。
27 0
|
3月前
|
算法 C++
【算法解题思想】动态规划+深度优先搜索(C/C++)
【算法解题思想】动态规划+深度优先搜索(C/C++)
|
4天前
|
算法 数据安全/隐私保护
室内障碍物射线追踪算法matlab模拟仿真
### 简介 本项目展示了室内障碍物射线追踪算法在无线通信中的应用。通过Matlab 2022a实现,包含完整程序运行效果(无水印),支持增加发射点和室内墙壁设置。核心代码配有详细中文注释及操作视频。该算法基于几何光学原理,模拟信号在复杂室内环境中的传播路径与强度,涵盖场景建模、射线发射、传播及接收点场强计算等步骤,为无线网络规划提供重要依据。
|
5天前
|
机器学习/深度学习 数据采集 算法
基于GA遗传优化的CNN-GRU-SAM网络时间序列回归预测算法matlab仿真
本项目基于MATLAB2022a实现时间序列预测,采用CNN-GRU-SAM网络结构。卷积层提取局部特征,GRU层处理长期依赖,自注意力机制捕捉全局特征。完整代码含中文注释和操作视频,运行效果无水印展示。算法通过数据归一化、种群初始化、适应度计算、个体更新等步骤优化网络参数,最终输出预测结果。适用于金融市场、气象预报等领域。
基于GA遗传优化的CNN-GRU-SAM网络时间序列回归预测算法matlab仿真
|
5天前
|
算法
基于龙格库塔算法的锅炉单相受热管建模与matlab数值仿真
本设计基于龙格库塔算法对锅炉单相受热管进行建模与MATLAB数值仿真,简化为喷水减温器和末级过热器组合,考虑均匀传热及静态烟气处理。使用MATLAB2022A版本运行,展示自编与内置四阶龙格库塔法的精度对比及误差分析。模型涉及热传递和流体动力学原理,适用于优化锅炉效率。