【专章】dp基础
知识储备:dp入门。
好了,完成了dp入门,我们可以做一些稍微不是那么裸的题了。
dp基础,主要是做题,只有练习才能彻底掌握。
洛谷P1417 烹调方案
分析:由于时间的先后会对结果有影响,所以c[i]*b[j]>c[j]*b[i]为条件排序,然后再01背包。
洛谷P1387 最大正方形
分析:用dp[i][j]来表示以a[i][j](正方形数组)为右下角最后一个数的正方形边长,a数组可以直接用dp数组代替掉。
#include <cstdio>
#include <algorithm>
using namespace std;
const int maxn=;
int dp[maxn][maxn];
int min_a(int a,int b,int c)
{
return min(min(a,b),min(b,c));
}
int main()
{
int n,m,ans=-;
scanf("%d%d",&n,&m);
for(int i=;i<=n;i++)
for(int j=;j<=m;j++)
scanf("%d",&dp[i][j]);
for(int i=;i<=n;i++)
{
for(int j=;j<=m;j++)
{
if(dp[i][j]>=)
{
dp[i][j]=min_a(dp[i-][j],dp[i-][j-],dp[i][j-])+;
}
}
}
for(int i=;i<=n;i++)
{
for(int j=;j<=m;j++)
ans=max(ans,dp[i][j]);
}
printf("%d",ans);
return ;
}
洛谷P1006 传纸条
分析:拿到题我不假思索地写了一份 单次最大好心值+dfs路径还原标记+单次最大好心值,然而发现,第一次最大好心值的路线把第二次堵死了。。。
好吧,正解是双线程,dp[k][i][j]表示走了k步,两份纸条横坐标分别是i,j时的好心最大值。
安利一发别人博客,讲得很透彻:http://www.cnblogs.com/OIerShawnZhou/p/7492555.html
#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
const int maxn=;
int dp[*maxn][maxn][maxn],a[maxn][maxn],go[maxn][maxn];
//dp[k][i][j]走了k步,两份纸条横坐标分别是i,j时的好心最大值
int max_a(int a,int b,int c,int d)
{
return max(max(a,b),max(c,d));
}
int main()
{
int m,n,ans=;
scanf("%d%d",&m,&n);
for(int i=;i<=m;i++)
for(int j=;j<=n;j++)
scanf("%d",&a[i][j]);
for(int k=;k<n+m;k++)//步数
{
for(int i=;i<=k;i++)
{
for(int j=;j<=k;j++)
{
dp[k][i][j]=max_a(dp[k-][i][j],dp[k-][i][j-],dp[k-][i-][j],dp[k-][i-][j-])+a[i][k-i+]+a[j][k-j+];
if(i==j) dp[k][i][j]-=a[i][k-i+];//重复了
}
}
}
printf("%d",dp[n+m-][m][m]);
return ;
}
洛谷P1282 多米诺骨牌
分析:拿到题有点懵,但仔细想想,这好像也算是01背包,可以把翻转每个骨牌得到的点数差看做一件物品(记住,点数不带绝对值),但会出现数组下标为负数的情况,这个时候把 N 看做为0,N-1就是-1,N+1就是1
dp[i][j]:前i张牌使得上一行点数之和为j的最小方法数
普通版:
#include <cstdio>
#include <cmath>
#include <algorithm>
using namespace std;
const int maxn=;
const int N=maxn*,INF=;
int dp[maxn][*N],a[maxn],b[maxn];//dp[i][j]:前i张牌使得上一行点数之和为j的最小方法数
int main()
{
int n,minnum=INF,k=;
scanf("%d",&n);
for(int i=;i<=n;i++)
{
scanf("%d%d",&a[i],&b[i]);
}
for(int i=;i<maxn;i++)
for(int j=;j<*N;j++)
dp[i][j]=INF;
dp[][N]=;
for(int i=;i<=n;i++)
{
for(int j=-N;j<=N;j++)
{
int dis=a[i]-b[i];//有正有负,故而开数组要开2*N,下面j+N同理
dp[i][j+N]=min(dp[i-][j+dis+N]+,dp[i-][j-dis+N]);//翻或不翻
}
}
for(int i=;i<=N;i++)
{
minnum=min(dp[n][N-i],dp[n][N+i]);
if(minnum!=INF)
{
printf("%d",minnum);
return ;
}
}
return ;
}
滚动数组优化:
#include <cstdio>
#include <cmath>
#include <algorithm>
using namespace std;
const int maxn=;
const int N=maxn*,INF=;
int dp[][*N],a[maxn],b[maxn];//dp[i][j]:前i张牌使得上一行点数之和为j的最小方法数
int main()
{
int n,minnum=INF,k=;
scanf("%d",&n);
for(int i=;i<=n;i++)
{
scanf("%d%d",&a[i],&b[i]);
}
for(int i=;i<;i++)
for(int j=;j<*N;j++)
dp[i][j]=INF;
dp[][N]=;
for(int i=;i<=n;i++)
{
for(int j=-N;j<=N;j++)
{
int dis=a[i]-b[i];//有正有负,故而开数组要开2*N,下面j+N同理
dp[i&][j+N]=min(dp[(i-)&][j+dis+N]+,dp[(i-)&][j-dis+N]);//翻或不翻
}
}
for(int i=;i<=N;i++)
{
minnum=min(dp[n&][N-i],dp[n&][N+i]);
if(minnum!=INF)
{
printf("%d",minnum);
return ;
}
}
return ;
}
洛谷P1880 石子合并
分析:曾经的noi系列。石子合并问题分好几种情况。
这题是环形的,且只能合并相邻的两堆,那么我们把石子数组展开,如[1,2,3,4,5] -> [1,2,3,4,5,1,2,3,4,5],这样进行处理,就能将环形转换成直线。
转换完之后,就可以用dp做了,dp的解释详见代码。这题数据过得去,时间是 O(n3) 不然要用平行四边形优化。
安利一发acdreamer的博客,把石子合并问题讲的很清楚http://blog.csdn.net/acdreamers/article/details/18039073
#include <cstdio>
#include <algorithm>
using namespace std;
int sum[],mins[][],maxs[][],s[][];//dp[i][j]=i~j的最大得分
int main()
{
int a,n,minnum=,maxnum=-minnum;
scanf("%d",&n);
for(int i=;i<=n;i++)
{
scanf("%d",&sum[i]);
sum[i+n]=sum[i];
s[i][i]=i;
s[i*][i*]=i;
}
for(int i=;i<=*n;i++) sum[i]+=sum[i-]; //前缀和
for(int l=;l<=n;l++)//区间长度
{
for(int i=;i+l<=*n;i++)//开始的端点
{ int j=i+l;//结束的端点
maxs[i][j]=-;mins[i][j]=-*maxs[i][j];
for(int k=i;k<j;k++)
{
maxs[i][j]=max(maxs[i][j],maxs[i][k]+maxs[k+][j]+sum[j]-sum[i-]);
mins[i][j]=min(mins[i][j],mins[i][k]+mins[k+][j]+sum[j]-sum[i-]);
}
}
}
for(int i=;i<=n;i++)//因为是环形的,可从任意两堆间分开,所以需要枚举一遍起点从1~n
{
maxnum=max(maxs[i][i+n-],maxnum);
minnum=min(mins[i][i+n-],minnum);
}
printf("%d\n%d",minnum,maxnum);
return ;
}
洛谷P1108 低价购买
分析:第一问还是很简单的,求最长下降子序列。第二问要计算方案数量,还要判重。
用t[i]表示前i个股票的不同方案个数,可以得出,如果存在dp[j]==dp[i] && a[j]==a[i]方案就是重复的,所以就把t[i]赋为0,如果f[i]==f[j]+1,那么i可以从j转移,所以t[i]+=t[j]。最后统计和即可。
#include <cstdio>
#include <algorithm>
using namespace std;
int dp[],a[],t[];//t,计算出现次数的dp
int main()
{
int ans=-,n,k=,res=;
scanf("%d",&n);
for(int i=;i<=n;i++) scanf("%d",&a[i]);
for(int i=;i<=n;i++)
{
dp[i]=;
for(int j=;j<i;j++)
{
if(a[j]>a[i]) dp[i]=max(dp[i],dp[j]+);
}
ans=max(ans,dp[i]);
}
for(int i=;i<=n;i++)
{
if(dp[i]==) t[i]=;//如果无法转移则为1
for(int j=;j<i;j++)
{
if(dp[j]==dp[i]- && a[i]<a[j])//判前继 判可以为下一个数
{
t[i]+=t[j];//转移
}
else if(dp[i]==dp[j]&&a[i]==a[j]) t[j]=;//判重
}
}
for(int i=;i<=n;i++)
{
if(dp[i]==ans) res+=t[i];
}
printf("%d %d",ans,res);
return ;
}
【专章】dp基础的更多相关文章
- 【学习笔记】dp基础
知识储备:dp入门. 好了,完成了dp入门,我们可以做一些稍微不是那么裸的题了. dp基础,主要是做题,只有练习才能彻底掌握. 洛谷P1417 烹调方案 分析:由于时间的先后会对结果有影响,所以c[i ...
- hdu 2089 不要62 (数位dp基础题)
不要62 Time Limit: 1000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total Submis ...
- poj 2955 Brackets (区间dp基础题)
We give the following inductive definition of a “regular brackets” sequence: the empty sequence is a ...
- DP基础(线性DP)总结
DP基础(线性DP)总结 前言:虽然确实有点基础......但凡事得脚踏实地地做,基础不牢,地动山摇,,,嗯! LIS(最长上升子序列) dp方程:dp[i]=max{dp[j]+1,a[j]< ...
- 树形dp基础
今天来给大家讲一下数形dp基础 树形dp常与树上问题(lca.直径.重心)结合起来 而这里只讲最最基础的树上dp 1.选课 题目描述 在大学里每个学生,为了达到一定的学分,必须从很多课程里选择一些课程 ...
- poj2642 The Brick Stops Here(DP基础题)
比基础的多一点东西的背包问题. 链接:POJ2642 大意:有N种砖,每种花费p[i],含铜量c[i],现需要用M种不同的砖融成含铜量在Cmin到Cmax之间(可等于)的砖,即这M种砖的含铜量平均值在 ...
- UVA103 dp基础题,DAG模型
1.UVA103 嵌套n维空间 DAG模型记忆化搜索,或者 最长上升子序列. 2.dp[i]=max( dp[j]+1),(第i个小于第j个) (1) //DAG模型记忆化搜索 #include< ...
- hdu 1561 The more, The Better(树形dp,基础)
The more, The Better Time Limit: 6000/2000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Oth ...
- hdu1114 Piggy-Bank (DP基础 完全背包)
链接:Piggy-Bank 大意:已知一只猪存钱罐空的时候的重量.现在的重量,已知若干种钱的重量和价值,猪里面装着若干钱若干份,求猪中的钱的价值最小值. 题解: DP,完全背包. g[j]表示组成重量 ...
随机推荐
- (转载)理解Java中的引用传递和值传递
关于Java传参时是引用传递还是值传递,一直是一个讨论比较多的话题,有论坛说Java中只有值传递,也有些地方说引用传递和值传递都存在,比较容易让人迷惑.关于值传递和引用传递其实需要分情况看待,今天 ...
- selenium 对chrome浏览器操作
参照http://www.testwo.com/blog/6931博客内容 1.下载ChromeDriver驱动包(下载地址: http://chromedriver.storage.googleap ...
- CVE-2016-10190 FFmpeg Http协议 heap buffer overflow漏洞分析及利用
作者:栈长@蚂蚁金服巴斯光年安全实验室 -------- 1. 背景 FFmpeg是一个著名的处理音视频的开源项目,非常多的播放器.转码器以及视频网站都用到了FFmpeg作为内核或者是处理流媒体的工具 ...
- 数据绑定技术一:GridView控件
在网站或应用程序中,要显示数据信息,可用到ASP.NET提供的数据源控件和能够显示数据的控件. 一.数据源控件 数据源控件用于连接数据源.从数据源中读取数据以及把数据写入数据源. 1.数据源控件特点 ...
- [我所理解的REST] 2.REST用来干什么的?
笔者每当遇到一个新事物的想去了解的时候,总是会问上自己第一个问题,这个新事物是干什么用的?在解释我所理解的REST这个过程中也不例外,这篇博客我们先关注一下REST是干什么用的,然后后续再解释REST ...
- [我所理解的REST] 3.基于网络应用的架构
上篇中解释到什么是架构风格和应该以怎样的视角来理解REST(Web的架构风格).本篇来介绍一组自洽的术语,用它来描述和解释软件架构:以及列举下对于基于网络的应用来说,哪些点是需要我们重点关注的. 1 ...
- 团队作业8——Beta 阶段冲刺6th day
一.当天站立式会议 二.每个人的工作 (1)昨天已完成的工作(具体在表格中) 完善订单功能 (2)今天计划完成的工作(具体如下) 完善支付功能 (3) 工作中遇到的困难(在表格中) 成员 昨天已完成的 ...
- 201521123040 《Java程序设计》第6周学习总结
1.本章学习总结 1.1 面向对象学习暂告一段落,请使用思维导图,以封装.继承.多态为核心概念画一张思维导图,对面向对象思想进行一个总结. 2. 书面作业 clone方法1.1 Object对象中的c ...
- 201521123059 《Java程序设计》第六周学习总结
1. 本周学习总结 1.1 面向对象学习暂告一段落,请使用思维导图,以封装.继承.多态为核心概念画一张思维导图,对面向对象思想进行一个总结. 注1:关键词与内容不求多,但概念之间的联系要清晰,内容覆盖 ...
- 201521123098 《Java程序设计》 第5周学习总结
1. 本周学习总结 1.1 尝试使用思维导图总结有关多态与接口的知识点. 1.2 可选:使用常规方法总结其他上课内容. 1. 对接口这一定义有了初步的了解: 2. 学习了如何定义实现类和如何实现一些接 ...