感觉动态规划非常模糊,怎么办呢???

狂刷题吧!!

!!

!!!

!!!

!!

。!!

。。!。!

PKU  PPt

动规解题的一般思路



1. 将原问题分解为子问题



                把原问题分解为若干个子问题,子问题和原问题形式同样或类似。仅仅只是规模变小了。

子问题都解决,原问题即解

决(数字三角形例)。

            子问题的解一旦求出就会被保存,所以每一个子问题仅仅需求解一次。

2. 确定状态



           在用动态规划解题时,我们往往将和子问题相关的各个变量的一组取值,称之为一个“状态”。

一个“状态”相应于一个或多个子问题,所谓某个“状态”下的“值”,就是这个“状态”所相应的子问题的解。

全部“状态”的集合。构成问题的“状态空间”。

“状态空间”的大小,与用动态规划解决这个问题的时间复杂度直接相关。

在数字三角形的样例里,一共同拥有N×(N+1)/2个数字。所以这个问题的状态空间里一共就有N×(N+1)/2个状态。

整个问题的时间复杂度是状态数目乘以计算每一个状态所需时间。

           在数字三角形里每一个“状态”仅仅须要经过一次。且在每一个状态上作计算所花的时间都是和N无关的常数。

           用动态规划解题,常常碰到的情况是,K个整型变量能构成一个状态(如数字三角形中的行号和列号这两个变量

构成“状态”)。

假设这K个整型变量的取值范围各自是N1, N2, ……Nk,那么。我们就能够用一个K维的数组array[N1] [N2]……[Nk]

来存储各个状态的“值”。

这个“值”未必就是一个整数或浮点数,可能是须要一个结构才干表示的。那么array就能够是一个结构数组。

一个“状态”下的“值”一般会是一个或多个子问题的解。

3. 确定一些初始状态(边界状态)的值

以“数字三角形”为例。初始状态就是底边数字,值就是底边数字值。

4. 确定状态转移方程



           定义出什么是“状态”,以及在该 “状态”下的“值”后,就要找出不同的状态之间怎样迁移――即怎样从一个或多个“值”

已知的“状态”,求出还有一个“状态”的“值”(“人人为我”递推型)。

状态的迁移能够用递推公式表示,此递推公式也可被称作“状态转移方程”。

数字三角形的状态转移方程:

能用动规解决的问题的特点



1) 问题具有最优子结构性质。

假设问题的最优解所包括的子问题的解也是最优的。我们就称该问题具有最优子结

构性质。

2) 无后效性。

当前的若干个状态值一旦确定。则此后过程的演变就仅仅和这若干个状态的值有关,和之前是採取哪

种手段或经过哪条路径演变到当前的这若干个状态,没有关系。

1、POJ 2479 Maximum sum

首刷水题!

!!

双向统计最大和。

AC代码例如以下:

#include<iostream>
#include<cstring>
#include<cstdio>
#define inf -1000000000
using namespace std;
int main()
{
int t;
scanf("%d",&t);
while(t--)
{
int n;
int i;
scanf("%d",&n);
int a[50005],dp[50005];
memset(dp,0,sizeof dp);
for(i=1;i<=n;i++)
scanf("%d",&a[i]);
int sum=0;
int maxx=inf;//须要取到负最小
for(i=1;i<n;i++)
{
sum+=a[i];
if(sum>maxx)
maxx=sum;
dp[i]=maxx;
if(sum<0)
sum=0;
}
int ans=inf;maxx=inf;sum=0;
for(i=n;i>1;i--)
{
sum+=a[i];
if(sum>maxx)
maxx=sum;
ans=max(ans,maxx+dp[i-1]);
if(sum<0)
sum=0;
}
cout<<ans<<endl;
}
return 0;
}

2、HDU  2059  龟兔赛跑

#include<iostream>
using namespace std;
#define M 10005
#define MAX 0xfffff; double min(double a,double b)
{
return a<b? a:b;
} int main()
{
double l,L;
int n,c;
double t,t1,t2,ti;
double vr,v1,v2;
double p[M],dp[M];
int i,j;
while(cin>>l)
{
cin>>n>>c>>t>>vr>>v1>>v2;
for(i=1,p[0]=0,p[n+1]=l;i<=n;i++)
cin>>p[i];
dp[0]=0;
t1=l/vr;t2=c/v1;
for(i=1;i<=n+1;i++)
{
dp[i]=MAX;
for(j=0;j<i;j++)
{
L=p[i]-p[j];
if(L>c)
ti=t2+(L-c)/v2;
else
ti=L/v1;
if(j)
ti+=t;
dp[i]=min(dp[i],dp[j]+ti);
}
}
if(dp[n+1]>t1)
cout<<"Good job,rabbit!"<<endl;
else
cout<<"What a pity rabbit!"<<endl; }
return 0;
}

3、POJ   1458  Common Subsequence

区间动规,最长公共子序列

#include<iostream>
#include<cstring>
#include<cstdio>
using namespace std; int dp[1005][1005]; int main()
{
int i,j;
int la,lb;
char a[1005],b[1005];
while (~scanf("%s%s",a,b))
{
la=strlen(a);
lb=strlen(b);
memset(dp,0,sizeof dp);
for(i=1;i<=la;i++)
for(j=1;j<=lb;j++)
{
if(a[i-1]==b[j-1])
{
dp[i][j]=dp[i-1][j-1]+1;
}
else {
dp[i][j]=max(dp[i-1][j],dp[i][j-1]);
}
} cout<<dp[la][lb]<<endl;
}
return 0;
}

4、POJ 3624 Charm Bracelet

背包水题。!

<pre name="code" class="cpp">#include<iostream>
#include<cstring>
#include<cstdio>
#define ll long long
using namespace std; int main()
{
int i,j;
int n,m;
int a[5000],b[5000];
int dp[20000];
while(~scanf("%d%d",&n,&m))
{
for(i=1;i<=n;i++)
scanf("%d%d",&a[i],&b[i]);
memset(dp,0,sizeof dp);
for(i=1;i<=n;i++)
for(j=m;j>=a[i];j--)
dp[j]=max(dp[j],dp[j-a[i]]+b[i]);
cout<<dp[m]<<endl;
} return 0;
}

5、POJ 2486  Apple Tree

经典树形dp!

#include<iostream>
#include<cstring>
#include<cstdio>
#include<algorithm>
using namespace std; int n,k;
int ap[205],cont[205];
int trees[205][205];
int dp[2][205][205];
int vis[205]; void dp_trees(int a)
{
int i,j,l;
int b;
for(i=0;i<=k;i++)
dp[0][a][i]=dp[1][a][i]=ap[a];
vis[a]=1;
for(i=1;i<=cont[a];i++)
{
b=trees[a][i];
if(!vis[b])
{
dp_trees(b);
for(j=k;j>=0;j--)
for(l=0;l<=j;l++)
{
dp[0][a][j+2]=max(dp[0][a][j+2],dp[0][b][l]+dp[0][a][j-l]);
dp[1][a][j+2]=max(dp[1][a][j+2],dp[0][b][l]+dp[1][a][j-l]);
dp[1][a][j+1]=max(dp[1][a][j+1],dp[1][b][l]+dp[0][a][j-l]);
}
}
}
} int main()
{
int i,j;
int a,b;
while(cin>>n>>k)
{
memset(ap,0,sizeof ap);
memset(cont,0,sizeof cont);
memset(trees,0,sizeof trees);
memset(vis,0,sizeof vis);
for(i=1;i<=n;i++)
cin>>ap[i];
for(i=1;i<n;i++)
{
cin>>a>>b;
trees[a][++cont[a]]=b;
trees[b][++cont[b]]=a;
}
memset(dp,0,sizeof dp);
dp_trees(1);
cout<<dp[1][1][k]<<endl;
}
return 0;
}

6、POJ  2533  Longest Ordered Subsequence

最长递增子序列。O(n*n)算法。。

#include<iostream>
#include<cstring>
#include<algorithm>
using namespace std; int main()
{
int i,j;
int t;
int n;
int dp[40005];
int a[40005];
while(cin>>n)
{
for(i=1;i<=n;i++)
{
cin>>a[i];
dp[i]=1;
}
for(i=2;i<=n;i++)
{
for(j=i-1;j>=1;j--)
{
if(a[i]>a[j])
dp[i]=max(dp[i],dp[j]+1);
}
}
int maxx=0;
for(i=1;i<=n;i++)
if(dp[i]>maxx)
maxx=dp[i];
cout<<maxx<<endl;
}
return 0;
}

7、POJ 1631  Bridging signals

最长递增子序列!

O(nlogn)算法!。

#include<iostream>
#include<cstring>
#include<algorithm>
using namespace std; int dp[40005];
int a[40005]; int dich(int l,int r,int i)
{
int mid;
while(r>l)
{
mid = (l+r)/2;
if(a[i]<=dp[mid]) r=mid;
else l=mid+1;
}
return l;
} int main()
{
int i,j;
int t;
int n;
cin>>t;
while(t--)
{
cin>>n;
for(i=1;i<=n;i++)
{
cin>>a[i];
}
dp[1]=a[1];
int top=1;
for(i=2;i<=n;i++)
{
if(a[i]>dp[top])
{
dp[++top]=a[i];
}
else
{
int ans=dich(1,top,i);
dp[ans]=a[i];
}
}
cout<<top<<endl;
}
return 0;
}

8、POJ  1836  Alignment

双向找最长递增子序列就可以!

#include<iostream>
#include<algorithm>
#include<cstring>
using namespace std;
int main()
{
int n;
int i,j;
double a[1005];
int l[1005],r[1005];
while(cin>>n)
{
for(i=1;i<=n;i++)
cin>>a[i];
memset(l,0,sizeof l);
memset(r,0,sizeof r);
for(i=1;i<=n;i++)
{
l[i]=1;
r[i]=1;
}
for(i=2;i<=n;i++)
{
for(j=i-1;j>=1;j--)
{
if(a[j]<a[i])
l[i]=max(l[i],l[j]+1);
}
}
for(i=1;i<=n;i++)
if(l[i]<l[i-1])
l[i]=l[i-1];
for(i=n-1;i>=1;i--)
{
for(j=i+1;j<=n;j++)
if(a[j]<a[i])
r[i]=max(r[i],r[j]+1);
}
for(i=n;i>=1;i--)
if(r[i]<r[i+1])
r[i]=r[i+1];
int ans=0;
for(i=0;i<=n+1;i++)
ans=max(ans,l[i]+r[i+1]); cout<<n-ans<<endl;
}
return 0;
}

9、POJ  1080   Human Gene Functions

#include<iostream>
#include<cstring>
using namespace std; int map[10][10]={{0,-3,-4,-2,-1},
{-3,5,-1,-2,-1},
{-4,-1,5,-3,-2},
{-2,-2,-3,5,-2},
{-1,-1,-2,-2,5}};
int main()
{
int t;
cin>>t;
while (t--)
{
int i,j;
char a[105],b[105];
int aa[105],bb[105];
int la,lb;
cin>>la>>a>>lb>>b;
for(i=0;i<la;i++)
{
if(a[i]=='A')
aa[i]=1;
if(a[i]=='C')
aa[i]=2;
if(a[i]=='G')
aa[i]=3;
if(a[i]=='T')
aa[i]=4;
}
for(i=0;i<lb;i++)
{
if(b[i]=='A')
bb[i]=1;
if(b[i]=='C')
bb[i]=2;
if(b[i]=='G')
bb[i]=3;
if(b[i]=='T')
bb[i]=4;
}
int dp[105][105];
memset(dp,0,sizeof dp);
for(i=1;i<=la;i++)
dp[i][0]=dp[i-1][0]+map[aa[i-1]][0];
for(i=1;i<=lb;i++)
dp[0][i]=dp[0][i-1]+map[0][bb[i-1]];
for(i=1;i<=la;i++)
{
for(j=1;j<=lb;j++)
{
dp[i][j]=max(dp[i-1][j]+map[aa[i-1]][0],dp[i][j-1]+map[0][bb[j-1]]);
dp[i][j]=max(dp[i][j],dp[i-1][j-1]+map[aa[i-1]][bb[j-1]]);
}
}
cout<<dp[la][lb]<<endl;
}
return 0;
}

10、HDU 1024  Max Sum  Plus Plus

#include<iostream>
#include<cstring>
#include<cstdio>
#include<algorithm>
#define inf -1000000000
using namespace std; int dp[1000005],pr[1000005];
int a[1000005]; int main ()
{
int n,m;
int i,j;
int maxx;
while(~scanf("%d%d",&m,&n))
{
for(i=1;i<=n;i++)
{
scanf("%d",&a[i]);
}
memset(dp,0,sizeof dp );
memset(pr,0,sizeof pr);
for(i=1;i<=m;i++)
{
maxx=inf;
for(j=i;j<=n;j++)
{
dp[j]=max(dp[j-1]+a[j],pr[j-1]+a[j]);
pr[j-1]=maxx;
maxx=max(maxx,dp[j]);
}
}
printf("%d\n",maxx);
}
return 0;
}

11、HDU 1231  最大连续子序列

#include<iostream>
#include<cstring>
#include<algorithm>
#include<cstdio>
#define inf -1000000000
using namespace std;
int main()
{
int n;
int i,j;
int a[10005];
while(~scanf("%d",&n)&&n)
{
for(i=1;i<=n;i++)
{
scanf("%d",&a[i]);
}
int l,r,bj=1;
l=1;r=1;bj=1;
int sum=0;
int maxx=inf;
int cont=0;
for(i=1;i<=n;i++)
{
sum+=a[i]; if(sum>maxx)
{
maxx=sum;
l=bj;
r=i;
}
if(sum<0)
{
cont++;
sum=0;
bj=i+1;
}
}
if(sum>maxx)
{
maxx=sum;
l=bj;
r=i;
}
if(cont==n)
printf("0 %d %d\n",a[1],a[n]);
else
printf("%d %d %d\n",maxx,a[l],a[r]); }
return 0;
}

12、HDU 1503  Advanced Fruits

最长公共序列的路径记忆!!!

#include<iostream>
#include<cstring>
#include<algorithm>
#include<cstdio>
using namespace std; char a[105],b[105];
int bj[105][105];
int dp[105][105]; void pri(int l,int r)
{
if(l==0&&r==0) return ;
if(bj[l][r]==1) {pri(l,r-1);cout<<b[r];}
else if(bj[l][r]==3) {pri(l-1,r);cout<<a[l];}
else {pri(l-1,r-1);cout<<a[l];}
} int main()
{
int i,j;
while(~scanf("%s%s",a+1,b+1))
{
int la=strlen(a+1);
int lb=strlen(b+1);
memset(dp,0,sizeof dp);
memset(bj,0,sizeof bj);
for(i=0;i<=la;i++)
bj[i][0]=3;
for(i=0;i<=lb;i++)
bj[0][i]=1;
for(i=1;i<=la;i++)
{
for(j=1;j<=lb;j++)
{ if(a[i]==b[j])
{
dp[i][j]=dp[i-1][j-1]+1;
bj[i][j]=2;
}
else
{
if(dp[i][j-1]>dp[i-1][j])
{
dp[i][j]=dp[i][j-1];
bj[i][j]=1;
}
else {
dp[i][j]=dp[i-1][j];
bj[i][j]=3;
}
}
}
}
pri(la,lb);
cout<<endl; }
return 0;
}

13、HDU 1058 Humble Numbers

数位水题!

!。

#include <iostream>
#include <cstdio>
#include <string>
#include <cstring>
#include <cstdlib>
#include <algorithm>
#define inf 2000000000
#define mod 1000000007
using namespace std; int main()
{
int a,b,c,d;
int n,i;
int s[10005];
a=b=c=d=1;
s[1]=1;
for(i=2;i<=5842;i++)
{
s[i]=min(s[a]*2,min(s[b]*3,min(s[c]*5,s[d]*7)));
if(s[i]==s[a]*2)
a++;
if(s[i]==s[b]*3)
b++;
if(s[i]==s[c]*5)
c++;
if(s[i]==s[d]*7)
d++;
}
while(~scanf("%d",&n)&&n)
{
if(n%10==1&&n%100!=11)
printf("The %dst humble number is %d.\n",n,s[n]);
else if(n%10==2&&n%100!=12)
printf("The %dnd humble number is %d.\n",n,s[n]);
else if(n%10==3&&n%100!=13)
printf("The %drd humble number is %d.\n",n,s[n]);
else
printf("The %dth humble number is %d.\n",n,s[n]);
}
return 0;
}

14、HDU 3466 Proud Merchants

背包好题!

。!

!!值得一做!!!!

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<queue>
#define M 10010
using namespace std; struct H
{
int p,q,v;
}a[505]; bool cmp(H a,H b)
{
return a.q-a.p<b.q-b.p;
} int dp[5005]; int main()
{
int i,j;
int n,m;
while(~scanf("%d%d",&n,&m))
{
memset(dp,0,sizeof dp);
for(i=0;i<n;i++)
scanf("%d%d%d",&a[i].p,&a[i].q,&a[i].v);
sort(a,a+n,cmp);
for(i=0;i<n;i++)
for(j=m;j>=a[i].p&&j>=a[i].q;j--)
dp[j]=max(dp[j],dp[j-a[i].p]+a[i].v);
printf("%d\n",dp[m]);
}
return 0;
}

14、HDU 3555 Bomb

数位基础训练開始。!

!!!

!!

我待水似流年,流年带我似水

15、HDU 2086 不要62

我待水似流年,流年待我似水

16、ACdream  1064  完美数

到这我都用了经典dp和记忆化搜索做,后面的记忆化搜索好理解一些

我待水似流年,流年待我似水

17、HDU  3652   B-number

数位DP!

记忆化搜索,优化非常水,500MS!


贴自己博客的代码也审核。

,无奈了!!发链接吧!!!

我待水似流年,流年待我似水

18、HDU 4389  X mod f(x)

数位dp!。

感觉这个题比較好!

。。。

我待水似流年,流年待我似水

动态规划 is beginning。。。。。。。。。的更多相关文章

  1. hdu 1159:Common Subsequence(动态规划)

    Common Subsequence Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Other ...

  2. 动态规划——区间dp

    在利用动态规划解决的一些实际问题当中,一类是基于区间上进行的,总的来说,这种区间dp是属于线性dp的一种.但是我们为了更好的分类,这里仍将其单独拿出进行分析讨论. 让我们结合一个题目开始对区间dp的探 ...

  3. 动态规划——线性dp

    我们在解决一些线性区间上的最优化问题的时候,往往也能够利用到动态规划的思想,这种问题可以叫做线性dp.在这篇文章中,我们将讨论有关线性dp的一些问题. 在有关线性dp问题中,有着几个比较经典而基础的模 ...

  4. VF(动态规划)

    VF 时间限制:1000 ms  |  内存限制:65535 KB 难度:2 描述 Vasya is the beginning mathematician. He decided to make a ...

  5. HDU 5009 Paint Pearls (动态规划)

    Paint Pearls Problem Description Lee has a string of n pearls. In the beginning, all the pearls have ...

  6. 动态规划 Common Subsequence

    描述 A subsequence of a given sequence is the given sequence with some elements (possible none) left o ...

  7. HDU 4247 Pinball Game 3D(cdq 分治+树状数组+动态规划)

    Pinball Game 3D Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) ...

  8. PAT 甲级 1033 To Fill or Not to Fill (25 分)(贪心,误以为动态规划,忽视了油量问题)*

    1033 To Fill or Not to Fill (25 分)   With highways available, driving a car from Hangzhou to any oth ...

  9. HDU 1159 Common Subsequence (动态规划、最长公共子序列)

    Common Subsequence Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Other ...

随机推荐

  1. 51nod建设国家

    小C现在想建设一个国家.这个国家中有一个首都,然后有若干个中间站,还有若干个城市. 现在小C想把国家建造成这样的形状:选若干(可以是0个)的中间站把他们连成一条直线,然后把首都连在这一条直线的左端.然 ...

  2. WHU 1572 Cyy and Fzz (AC自动机 dp )

    题意: 给出n个串,求任意长度为m的字符串包含串的个数的期望.(n<=8,m<=14,给定串的长度不超过12). Solution: 首先可以想到应该用概率DP,我们需要至少3维,dp[i ...

  3. Jquery异步请求数据实例

    一.Jquery向aspx页面请求数据 前台页面JS代码: $("#Button1").bind("click", function () { $.ajax({ ...

  4. 使用Thinkphp框架开发移动端接口

     本文给大家分享的是使用thinkphp框架开发移动端接口的2种方法,一种是开发API,另外一种是实现移动端访问自动切换移动主题模板,从而实现伪app访问,下面我们就来详细看下如何实现吧. 方案一:给 ...

  5. nginx 一般配置实例 静态页面

    # 使用的用户和组 user www www; # 指定工作衍生进程数(一般等于CPU的总核数或总核数的两倍,例如两个四核CPU,则总核数为8) worker_processes 8; # 指定错误日 ...

  6. html框架集 js刷新页面方法大全

    一.先来看一个简单的例子: 下面以三个页面分别命名为frame.html.top.html.bottom.html为例来具体说明如何做. frame.html 由上(top.html)下(bottom ...

  7. Linux on ASUS N550JK4700

    实际上,ASUS N550JK对Ubuntu 14.04的兼容性是相当好的,包括无线网卡.蓝牙.键盘背光的调节.触摸板的开关.音量的键盘调节都是安装后无需配置直接可以使用的,这是出乎意料的,因为这些功 ...

  8. Centos yum install

    http://wiki.centos.org/TipsAndTricks/BrokenVserver centos mirror:  http://mirror.centos.org/centos/6 ...

  9. 练习2 D 题- 第几天?

    Time Limit:1000MS     Memory Limit:32768KB     64bit IO Format:%I64d & %I64u   Description 给定一个日 ...

  10. 提高Order by语句查询效率的两个思路

    提高Order by语句查询效率的两个思路 2011-03-01 13:07 水太深 ITPUB 字号:T | T 在MySQL数据库中,Order by语句的使用频率是比较高的.但是众所周知,在使用 ...