【Kuangbin简单DP】平整数组

news2025/1/11 3:58:37

4562. 平整数组 - AcWing题库

题意:

思路:

一开始想的是

设DP状态是:dp[i][0/1]表示阶段到 i ,然后前面是递增or递减的最小sum的集合

然后我是怎么转移的呢,现在看来感觉挺好笑的:

#include <bits/stdc++.h>
using namespace std;
#define int long long
const int mxn=2e3+10;
int n;
int a[mxn],dp[mxn][2];
signed main(){
	ios::sync_with_stdio(0),cin.tie(0),cout.tie(0);
	cin>>n;
	for(int i=1;i<=n;i++) cin>>a[i];
	for(int i=1;i<=n;i++){
		dp[i][0]=dp[i-1][0]+abs(a[i]-a[i-1])*(a[i-1]<a[i]);
		dp[i][1]=dp[i-1][1]+abs(a[i]-a[i-1])*(a[i-1]>a[i]);
	}
	int ans=min(dp[n][0],dp[n][1]);
	cout<<ans<<'\n';
}

 这尼玛....和贪心有区别嘛....

就是如果满足单调就不去动这个位置的数,否则就把它变成前一个数

这显然不能这样做啊qwq

状态转移确实是不重不漏了,但是枚举决策和枚举上一层的状态,显然也要满足不重不漏啊

要枚举到所有决策和上一层的所有状态,才能保证DP的正确性吧

我这样写,显然都没枚举决策,直接按照贪心策略去转移了,答案自然是错的

正解应该是这样的:

设dp[i][j]表示枚举到阶段 i ,然后第 i 位置选 b[j] 的最小sum的集合

其中 b 数组是 a 数组离散化之后的结果,说白了就是 a 数组这个集合

为什么这么设计?首先阶段 i 不用说,我们去看是什么影响了决策

是单调性

因此我们要去记录一下该位置放的是什么数

这也提示了我们,决策是可以放在数组里的

而且,在状态设计的时候,表示的是该阶段的参数,这样好转移

然后怎么转移?

for(int i=1;i<=n;i++){
        for(int j=1;j<=len;j++){
            for(int k=1;k<=j;k++){
                dp[i][j]=min(dp[i][j],dp[i-1][k]+abs(a[i]-b[j]));
            }
        }
    }

这个咋想到的

转移顺序肯定是满足拓扑序的嘛

即先去枚举阶段 ,再去枚举状态,最后去枚举决策

 第一层for循环就是枚举阶段

第二层for循环就是去枚举这个阶段的状态,即位置 i 的状态

然后就是去枚举决策,由于这道题决策也是状态的一部分,所以直接去枚举上一层状态 k

由于要满足单调性,这里先去考虑递增,所以上一层状态一定比这一层小,因此 k 在1到j之间

求完递增的,再求一遍递减的就好了

Code:

#include <bits/stdc++.h>
using namespace std;
const int mxn=2e3+10;
int n,ans=0x3f3f3f3f;
int a[mxn],b[mxn],dp[mxn][mxn];
int main(){
    ios::sync_with_stdio(0),cin.tie(0),cout.tie(0);
    cin>>n;
    for(int i=1;i<=n;i++) cin>>a[i],b[i]=a[i];
    sort(b+1,b+1+n);
    int len=unique(b+1,b+1+n)-(b+1);
    memset(dp,0x3f,sizeof(dp));
    for(int j=1;j<=len;j++) dp[0][j]=0;
    for(int i=1;i<=n;i++){
        for(int j=1;j<=len;j++){
            for(int k=1;k<=j;k++){
                dp[i][j]=min(dp[i][j],dp[i-1][k]+abs(a[i]-b[j]));
            }
        }
    }
    for(int j=1;j<=len;j++) ans=min(ans,dp[n][j]);
    memset(dp,0x3f,sizeof(dp));
    for(int j=1;j<=len;j++) dp[0][j]=0;
    for(int i=1;i<=n;i++){
        for(int j=1;j<=len;j++){
            for(int k=j;k<=len;k++){
                dp[i][j]=min(dp[i][j],dp[i-1][k]+abs(a[i]-b[j]));
            }
        }
    }
    for(int j=1;j<=len;j++) ans=min(ans,dp[n][j]);
    cout<<ans<<'\n';
}

但是这样复杂度是O(n^3)的,太暴力了

我们需要优化一下

这里复杂度优化的本质是,在枚举这一层阶段的状态时记录上一层状态的最值

用变量保存一下,然后直接转移就好了

Code:

#include <bits/stdc++.h>
using namespace std;
#define int long long
const int mxn=2e3+10;
int n,ans=1e18;
int a[mxn],b[mxn],dp[mxn][mxn];
signed main(){
    ios::sync_with_stdio(0),cin.tie(0),cout.tie(0);
    cin>>n;
    for(int i=1;i<=n;i++) cin>>a[i],b[i]=a[i];
    sort(b+1,b+1+n);
    int len=unique(b+1,b+1+n)-(b+1);
    memset(dp,127,sizeof(dp));
    for(int j=1;j<=len;j++) dp[0][j]=0;
    for(int i=1;i<=n;i++){
        int mi=1e18;
        for(int j=1;j<=len;j++){
            mi=min(mi,dp[i-1][j]);
            dp[i][j]=min(dp[i][j],mi+abs(a[i]-b[j]));
        }
    }
    for(int j=1;j<=len;j++) ans=min(ans,dp[n][j]);
    memset(dp,127,sizeof(dp));
    reverse(b+1,b+1+len);
    for(int j=1;j<=len;j++) dp[0][j]=0;
    for(int i=1;i<=n;i++){
        int mi=1e18;
        for(int j=1;j<=len;j++){
            mi=min(mi,dp[i-1][j]);
            dp[i][j]=min(dp[i][j],mi+abs(a[i]-b[j]));
        }
    }
    for(int j=1;j<=len;j++) ans=min(ans,dp[n][j]);
    cout<<ans<<'\n';
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/134598.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Golang 函数使用的注意事项和细节

来自&#xff1a;尚硅谷-韩老师 尚硅谷 1&#xff09;函数的形参列表可以是多个&#xff0c;返回值列表也可以是多个 2&#xff09;形参列表和返回值列表的数据类型可以是值类型和引用类型 3&#xff09;函数的命名遵循标识符命名规范&#xff0c;首字母不能是数字&#xf…

京东购物成功订单已开具个人发票不能报销怎么申请更换重新开具企业发票用于报销?

原文来源&#xff1a;https://www.caochai.com/article-4109.html 京东购物订单的发票开成个人了可以更换成企业发票吗&#xff1f; 可以&#xff0c;通过京东购物完成的订单默认开个人发票&#xff0c;如果对发票有要求需要开企业发票的可以申请更换重新开企业发票&#xff1…

一文读懂JVM虚拟机:JVM虚拟机的内存管理(万字详解)

JVM虚拟机的内存管理 文章目录JVM虚拟机的内存管理JVM与操作系统Java虚拟机规范和 Java 语言规范的关系java虚拟机的内存管理JVM整体架构一、PC 程序计数器二、虚拟机栈三、本地方法栈四、堆Java 堆概念年轻代和老年代对象分配过程堆GC元空间为什么要废弃永久代&#xff0c;引入…

动手学习深度学习-《矩阵运算》

标量导数 常用求导&#xff1a; yyyaaaxnx^nxnexp(x)exp(x)exp(x)log(x)log(x)log(x)sin(x)sin(x)sin(x)dydx\frac{dy}{dx}dxdy​000nxn−1nx^{n-1}nxn−1exp(x)1x\frac{1}{x}x1​cos(x)cos(x)cos(x) 求导公式&#xff1a; yyyuvuvuvuvuvuvyf(u),ug(x)yf(u),ug(x)yf(u),ug(x)d…

植物大战僵尸:学会使用人造指针

通过向游戏中注入一段特殊的汇编代码&#xff0c;实现自动获取动态地址&#xff0c;省略找基址的麻烦。该方法适用于游戏基址层数过多无法直接获取到基址&#xff0c;游戏根本无法找到基址。 1.打开CE工具并附加游戏进程&#xff0c;首先通过遍历的方式找到阳光的动态地址&…

zookeeper之master选举代码实现

master选举的基本概念 &#xff08;1&#xff09;假设有一个系统A,它向外提供了一个服务&#xff0c;叫做服务B。并且这个服务需要24小时持续不断的向外提供。也就是提供服务的机器不能够有单点故障。于是我们考虑使用集群。 &#xff08;2&#xff09;我们采用的是master-sla…

傻白入门芯片设计,如何做文献笔记(十九)

Article: 文献出处&#xff08;方便再次搜索&#xff09; 作者文献题目文献时间Data: 文献数据&#xff08;总结归纳&#xff0c;方便理解&#xff09; 这篇文章的目的结论背景介绍结果方法&#xff08;可选&#xff09;Comments: 对文献的想法 &#xff08;强迫自己思考&#…

Java人脸识别相册分类按时间分类相册按城市分类相册app源码

简介 后台Java&#xff0c;前台mui开发的android app&#xff1b;主要是按拍摄时间&#xff0c;人脸&#xff0c;城市进行相册照片的分类。 演示视频 https://www.bilibili.com/video/BV1XP4y187rA/?share_sourcecopy_web&vd_sourceed0f04fbb713154db5cc611225d92156 技…

MySQL调优-Explain详解和索引最佳实践

目录 Explain工具介绍 Explain分析示例 explain 两个变种 explain中的列 1.id列 2.select_type列 3. table列 4.type列 5. possible_keys列 6. key列 7. key_len列 8. ref列 9. rows列 10.Extra列 索引最佳实践 1.全值匹配 2.最左前缀法则 3.不在索引列上做任何操…

2D 平台动作冒险游戏

本文实现比较流畅的跑和跳跃 跑的动画需要从idle经历到walk再到run的过程&#xff0c;这个过程可以用融合树实现 也可以让玩家在按下按键时先固定播放完一个walk的动画&#xff0c;然后再自动切换到run的状态。 只不过在任何状态时&#xff0c;只要玩家松开了按键&#xff0c;…

钧瓷产业数字化,将促使禹州走向更高级的社会形态——钧共体

讲好钧瓷产业的故事 有深度的故事,有温度的内容 有态度的文字,有立场的思考 版权声明:钧瓷内参独家发布,侵权必究 第334期 钧瓷内参 2023年1月1日 这里的上市指沪深的主板,创业板和科创板,区域的挂牌不算。 这个数据是根据禹州钧瓷产业2022…

GPU服务器之Colab配置及使用教程

在进行深度学习过程中&#xff0c;不可避免需要进行炼丹的操作&#xff0c;而博主的电脑本身性能就很拉跨&#xff0c;一旦再运行其他的应用便是寸步难行&#xff0c;最好的办法便是再买一台&#xff0c;哈哈哈&#xff0c;当然也可以使用一些云服务器来进行&#xff0c;博主这…

MySQL调优-深入理解MySQL索引底层数据结构与算法

目录 索引的定义 B-Tree (1) 非叶子节点不存储data数据&#xff0c;只进行存储索引(这个索引是冗余的索引&#xff0c;多余的)&#xff0c;可以放更多的索引 (2) 叶子节点包含所有的索引字段 (3) 叶子节点用指针相互连接&#xff0c;提高区间访问的性能 Hash (1) 对索引的…

基于pybind11的c++开发cuda算子用于python调用

一、环境 win10VS2022python39pybind11最新版git下载源码,后边会用:https://github.com/pybind/pybind11同时通过pip install pybind11,python环境下也要安装cuda环境配置参考:https://zhuanlan.zhihu.com/p/488518526二、cuda核函数动态库生成 步骤一:创建cuda项目 创…

VueElement

一、vue 1.1 概述 接下来我们学习一款前端的框架&#xff0c;就是VUE。 Vue 是一套前端框架&#xff0c;免除原生JavaScript中的DOM操作&#xff0c;简化书写。 我们之前也学习过后端的框架 Mybatis &#xff0c; Mybatis 是用来简化 jdbc 代码编写的&#xff1b;而 VUE 是前…

绝对路径和相对路径

目录 1说明 2.相对路径的优势 3.注意 4./说明 1说明 绝对路径&#xff1a;文件真实的存放位置 例&#xff1a;D:\data\applogs\xxl-job\xxl-job-admin.log 不需要知道其他任何信息就可以根据绝对路径判断出文件的位置 相对路径&#xff1a;相对于当前位置的路径 前端开…

QT学 控件(四)步长调节框(QSpinBox + QDateTimeEdit)

文章目录QSpinBoxQDateTimeEditQSpinBox 允许用户通过单击向上/向下按钮或按键盘上的上/下来选择一个值来增加/减少当前显示的值。用户还可以手动键入值 常用于处理 选择 &#xff1a; 【1-99】 【星期1一 - 星期天】【1月-12月】 样式&#xff1a; 继承自&#xff1a; QAb…

网络协议(二):MAC地址、IP地址、子网掩码、子网和超网

网络协议系列文章 网络协议(一)&#xff1a;基本概念、计算机之间的连接方式 网络协议(二)&#xff1a;MAC地址、IP地址、子网掩码、子网和超网 目录一、MAC地址二、IP地址1、IP地址的组成2、IP地址的分类三、子网划分1、等长子网划分2、变长子网划分四、超网五、判断一个网段…

DIY NAS服务器之OMV 5.6入坑指南(四) -安装docker和Portainer

系列文章目录 DIY NAS服务器之OMV 5.6入坑指南&#xff08;一&#xff09;-openmediavalut 5.6安装DIY NAS服务器之OMV 5.6入坑指南&#xff08;二&#xff09;- 安装omv-extras插件DIY NAS服务器之OMV 5.6入坑指南&#xff08;三&#xff09;- 切换系统源​​​​​​​ ​​…

CentOS安装Vscode-yum+Vscode前端开发必安装的插件

CentOS安装Vscode-yum Visual Studio Code 是 Microsoft 产品和开源跨平台代码编辑器。 它提供内置调试支持、代码完成、集成终端语法高亮、嵌入式 Git 控制、代码重构和代码片段。 在本教程中&#xff0c;我们将介绍如何使用 CentOS 7 机器上的官方代码存储库安装最新版本的…