0x51 线性DP
数据结构没什么好写的。。分块和整体二分还有点分学得很懂。。果然我还是比较适合这些东西
poj2279 奇怪题,我的想法就是五维记录最边上的一斜排,会M,结果的的确确是锻炼思维的,正解并不是DP2333
LCIS 这个。。。其实还挺常规的吧(脑子一抽),f[i][j]表示第一个匹配到i第二个匹配到j且选了j。
#include<cstdio>
#include<iostream>
#include<cstring>
#include<cstdlib>
#include<algorithm>
#include<cmath>
using namespace std; int a[],b[],f[][];
int main()
{
int n;
scanf("%d",&n);
for(int i=;i<=n;i++)scanf("%d",&a[i]);
for(int i=;i<=n;i++)scanf("%d",&b[i]); for(int i=;i<=n;i++)
{
int mx=;
for(int j=;j<=n;j++)
{
if(a[i]!=b[j])f[i][j]=f[i-][j];
else f[i][j]=mx+;
if(b[j]<a[i])mx=max(mx,f[i-][j]);
}
} int ans=;
for(int j=;j<=n;j++)ans=max(ans,f[n][j]);
printf("%d\n",ans);
return ;
}
LCIS
poj3666 首先要证明的是一定存在一种构造b的方案令所有b值都在a中出现过(我就不证了),f[i][j]表示枚举到i,且当前等于a数组中排名为j的那个数的最小值,和上题一样可以省掉一个for(也是常规操作吧)
#include<cstdio>
#include<iostream>
#include<cstring>
#include<cstdlib>
#include<algorithm>
#include<cmath>
using namespace std;
typedef long long LL; int n,m; LL a[],ls[];
void LSH()
{
m=;
for(int i=;i<=n;i++)ls[++m]=a[i];
sort(ls+,ls+m+);
m=unique(ls+,ls+m+)-ls-;
for(int i=;i<=n;i++)
a[i]=lower_bound(ls+,ls+m+,a[i])-ls;
} LL ans;
LL f[][];
LL myabs(LL x){return x>?x:-x;}
void DP1()
{
memset(f,,sizeof(f));
for(int j=;j<=n;j++)f[][j]=myabs(ls[a[]]-ls[j]);
for(int i=;i<=n;i++)
{
LL mn=(1LL<<);
for(int j=;j<=m;j++)
{
mn=min(mn,f[i-][j]);
f[i][j]=mn+myabs(ls[a[i]]-ls[j]);
}
}
for(int j=;j<=n;j++)ans=min(ans,f[n][j]);
}
void DP2()
{
memset(f,,sizeof(f));
for(int j=;j<=n;j++)f[][j]=myabs(ls[a[]]-ls[j]);
for(int i=;i<=n;i++)
{
LL mn=(1LL<<);
for(int j=m;j>=;j--)
{
mn=min(mn,f[i-][j]);
f[i][j]=mn+myabs(ls[a[i]]-ls[j]);
}
}
for(int j=;j<=n;j++)ans=min(ans,f[n][j]);
} int main()
{
scanf("%d",&n);
for(int i=;i<=n;i++)scanf("%lld",&a[i]);
LSH(); ans=(1LL<<);
DP1();DP2();
printf("%lld\n",ans);
return ;
}
poj3666
Mobile Service 这个省维的思想还挺好的,因为要修第i-1个位置,那么当要修第i个位置的时候肯定有个人在p[i-1]省去一维
#include<cstdio>
#include<iostream>
#include<cstring>
#include<cstdlib>
#include<algorithm>
#include<cmath>
using namespace std; int mp[][],p[];
int f[][][]; int main()
{
int n,Q;
scanf("%d%d",&n,&Q);
for(int i=;i<=n;i++)
for(int j=;j<=n;j++)
scanf("%d",&mp[i][j]);
p[]=;
for(int i=;i<=Q;i++)scanf("%d",&p[i]); memset(f,,sizeof(f));f[][][]=f[][][]=;
for(int i=;i<Q;i++)
{
for(int j=;j<=n;j++)
if(j!=p[i])
for(int k=;k<=n;k++)
if(k!=p[i]&&k!=j)
if(f[i][j][k]<=)
{
f[i+][j][k]=min(f[i+][j][k],f[i][j][k]+mp[p[i]][p[i+]]);
f[i+][p[i]][k]=f[i+][k][p[i]]=min(f[i+][p[i]][k],f[i][j][k]+mp[j][p[i+]]);
f[i+][p[i]][j]=f[i+][j][p[i]]=min(f[i+][p[i]][j],f[i][j][k]+mp[k][p[i+]]);
}
}
int ans=;
for(int j=;j<=n;j++)
for(int k=;k<=n;k++)
ans=min(ans,f[Q][j][k]);
printf("%d\n",ans);
return ;
}
Mobile Service
传纸条 表示我一眼插头DP 然而就是大力DP就好,只用三维,因为只能下或右走那么当前两条线的位置肯定是在一条右往左的斜线上,也就是x1+y1=x2+y2
#include<cstdio>
#include<iostream>
#include<cstring>
#include<cstdlib>
#include<algorithm>
#include<cmath>
using namespace std; int a[][];
int f[][][];
int main()
{
int n,m;
scanf("%d%d",&n,&m);
for(int i=;i<=n;i++)
for(int j=;j<=m;j++)
scanf("%d",&a[i][j]); memset(f,,sizeof(f));f[][][]=a[][]+a[][];
for(int i=;i<n+m-;i++)
{
for(int x1=;x1<=m;x1++)
{
int y1=i+-x1;
if(y1<=n)
{
for(int x2=;x2<=x1-;x2++)
{
int y2=i+-x2;
if(y2<=n)
{
if(y1+<=n&&y2+<=n)f[i+][x1][x2]=max(f[i+][x1][x2],f[i][x1][x2]+a[y1+][x1]+a[y2+][x2]);
if(x1+<=m&&x2+<=m)f[i+][x1+][x2+]=max(f[i+][x1+][x2+],f[i][x1][x2]+a[y1][x1+]+a[y2][x2+]);
if(y1+<=n&&x2+<=m&&(x2+<x1||i==n+m-))f[i+][x1][x2+]=max(f[i+][x1][x2+],f[i][x1][x2]+a[y1+][x1]+a[y2][x2+]);
if(x1+<=m&&y2+<=n)f[i+][x1+][x2]=max(f[i+][x1+][x2],f[i][x1][x2]+a[y1][x1+]+a[y2+][x2]);
}
}
}
}
}
printf("%d\n",f[n+m-][m][m]);
return ;
}
传纸条
I-country 超麻烦,需要考虑行数,左右边界,当前权,当前左右区间分别是上升还是下降
0x51 线性DP的更多相关文章
- 算法竞赛进阶指南0x51 线性DP
AcWing271. 杨老师的照相排列 思路 这是一个计数的题目,如果乱考虑,肯定会毫无头绪,所以我们从1号到最后一个依次进行安排. 经过反复实验,发现两个规律 每一行的同学必须是从左向右依次连续放置 ...
- LightOJ1044 Palindrome Partitioning(区间DP+线性DP)
问题问的是最少可以把一个字符串分成几段,使每段都是回文串. 一开始想直接区间DP,dp[i][j]表示子串[i,j]的答案,不过字符串长度1000,100W个状态,一个状态从多个状态转移来的,转移的时 ...
- Codeforces 176B (线性DP+字符串)
题目链接: http://acm.hust.edu.cn/vjudge/problem/viewProblem.action?id=28214 题目大意:源串有如下变形:每次将串切为两半,位置颠倒形成 ...
- hdu1712 线性dp
//Accepted 400 KB 109 ms //dp线性 //dp[i][j]=max(dp[i-1][k]+a[i][j-k]) //在前i门课上花j天得到的最大分数,等于max(在前i-1门 ...
- 动态规划——线性dp
我们在解决一些线性区间上的最优化问题的时候,往往也能够利用到动态规划的思想,这种问题可以叫做线性dp.在这篇文章中,我们将讨论有关线性dp的一些问题. 在有关线性dp问题中,有着几个比较经典而基础的模 ...
- POJ 2479-Maximum sum(线性dp)
Maximum sum Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 33918 Accepted: 10504 Des ...
- poj 1050 To the Max(线性dp)
题目链接:http://poj.org/problem?id=1050 思路分析: 该题目为经典的最大子矩阵和问题,属于线性dp问题:最大子矩阵为最大连续子段和的推广情况,最大连续子段和为一维问题,而 ...
- nyoj44 子串和 线性DP
线性DP经典题. dp[i]表示以i为结尾最大连续和,状态转移方程dp[i] = max (a[i] , dp[i - 1] + a[i]) AC代码: #include<cstdio> ...
- 『最大M子段和 线性DP』
最大M子段和(51nod 1052) Description N个整数组成的序列a[1],a[2],a[3],-,a[n],将这N个数划分为互不相交的M个子段,并且这M个子段的和是最大的.如果M &g ...
随机推荐
- HTTPS的中那些加密算法
密码学在计算机科学中使用非常广泛,HTTPS就是建立在密码学基础之上的一种安全的通信协议.HTTPS早在1994年由网景公司首次提出,而如今在众多互联网厂商的推广之下HTTPS已经被广泛使用在各种大小 ...
- intellij 创建java web项目(maven管理的SSH)
intellij 创建java web项目(maven管理的SSH) 环境intellij IDEA14.MAVEN.Spring.Struts2.Hibernate.Java Web.工程搭建. 1 ...
- jquery-pjax
项目介绍: Pjax是jQuery的一个插件,Pjax即pushState + Ajax,是实现无刷新Ajax加载并解决浏览器前进和后退问题的一个开源实现. 在2012年8月28日发布0.9版本. P ...
- Docker 数据卷重复挂载测试
没想到一年没写博客了,这中间都是记在自己的笔记本上,大部分网上都有,这个好像没有,所以发上来吧! 本文是测试Docker容器(相同目录/父子目录)同时挂载到宿主机(同目录/不同目录)时的情况,废话少说 ...
- 八进制、十进制、操作符(day04)
把二进制表示的数字从右向左每三个数位分成 一组,每组用一个0到7之间的数字替换. 这个替换结果叫做数字的八进制表示方式 (八进制) 可以直接在程序里用八进制方式表示数字, 这种数字必须以0做开头 可以 ...
- IE下自定义错误页面不显示的原因
如果一个 404 页面的内容小于 512B,IE 会认为该 404 页面不够友好,在 IE 下将不会成功返回该 404 错误页面
- Nginx面试中最常见的18道题 抱佛脚必备
Nginx的并发能力在同类型网页服务器中的表现,相对而言是比较好的,因此受到了很多企业的青睐,我国使用Nginx网站的知名用户包括腾讯.淘宝.百度.京东.新浪.网易等等.Nginx是网页服务器运维人员 ...
- Atcoder ARC 082C/D
C - Together 传送门:http://arc082.contest.atcoder.jp/tasks/arc082_a 本题是一个数学问题. 有一个长度为n的自然数列a[1..n],对于每一 ...
- delphi主i窗口中实现多页面管理效果
MainForm上加Panel,把Panel的DockSite设为True, 把窗口FB_orderManage的DragKind设为dkDock, DragMode设为dmAuromatic 在M ...
- Java并发之CAS与AQS简介
1,什么是CAS CAS(Compare And Swap),即比较并交换.是解决多线程并行情况下使用锁造成性能损耗的一种机制,CAS操作包含三个操作数——内存位置(V).预期原值(A)和新值(B). ...