版权声明:最后一年,加油~ https://blog.csdn.net/zzti_xiaowei/article/details/82219395
- 经典dp:最大M字段和
- dp[i][j]:表示以a[j]结尾的前j个元素分成i组的最大和。
- 它的取值有两种可能:
- (x1,y1) (x2,y2) … (xi,yi,a[j]) // a[j]加前i组合为第i组
- (x1,y1) (x2,y2) … (xi-1,yi-1) (a[j]) // a[j]单独属于第i组
- 故状态转移方程:dp[i][j]=max(dp[i][j-1],dp[i-1][k])+a[j] i<=k<=j-1
- 优化:用pre[j-1]记录dp[i-1]k的最大值 。
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
const int inf=0x3f3f3f3f;
const int Max_n=1e6+10;
int n,m;
int a[Max_n],pre[Max_n];
int dp[Max_n];
int main()
{
while(~scanf("%d%d",&m,&n)){
for(int i=1;i<=n;i++)scanf("%d",&a[i]);
memset(dp,0,sizeof(dp));
memset(pre,0,sizeof(pre));
int tmp;
for(int i=1;i<=m;i++){
tmp=-inf;
for(int j=i;j<=n;j++){
dp[j]=max(dp[j-1],pre[j-1])+a[j];
pre[j-1]=tmp;
tmp=max(tmp,dp[j]);
}
}
printf("%d\n",tmp);
}
return 0;
}
Hdu-1029 Ignatius and the Princess IV
// 简单桶排序,dp[i]记录数字i出现的次数
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
const int Max_n=1e6+10;
int n;
int dp[Max_n];
int main()
{
while(~scanf("%d",&n)){
memset(dp,0,sizeof(dp));
int c,f;
for(int i=0;i<n;i++){
scanf("%d",&c);
dp[c]++;
if(dp[c]>=(n+1)/2)f=c;
}
printf("%d\n",f);
}
return 0;
}
// 先求出每个积木的六种摆放方式,然后排序后求最大递增子序列。
// dp[i]:表示第i个积木可搭的最大高度。
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
const int Max_n=1100;
int n,t=0;
struct edge{
int x,y,z;
bool operator<(const edge& e)const{
if(x!=e.x)return x<e.x;
return y<e.y;
}
}e[Max_n];
int dp[Max_n];
int main()
{
while(~scanf("%d",&n)&&n){
int a,b,c;
for(int i=0;i<n;i++){
scanf("%d%d%d",&a,&b,&c);
e[i*6].x=a,e[i*6].y=b,e[i*6].z=c;
e[i*6+1].x=a,e[i*6+1].y=c,e[i*6+1].z=b;
e[i*6+2].x=b,e[i*6+2].y=a,e[i*6+2].z=c;
e[i*6+3].x=b,e[i*6+3].y=c,e[i*6+3].z=a;
e[i*6+4].x=c,e[i*6+4].y=a,e[i*6+4].z=b;
e[i*6+5].x=c,e[i*6+5].y=b,e[i*6+5].z=a;
}
sort(e,e+(n*6));
int Max=0;
for(int i=0;i<n*6;i++){ //求最大递增子序列
dp[i]=e[i].z;
for(int j=0;j<i;j++){
if(e[i].x>e[j].x&e[i].y>e[j].y)
dp[i]=max(dp[i],dp[j]+e[i].z);
}
Max=max(Max,dp[i]);
}
printf("Case %d: maximum height = %d\n",++t,Max);
}
return 0;
}
Hdu-1087 Super Jumping! Jumping! Jumping!
//最大递增子序列
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
const int Max_n=1100;
int n;
int a[Max_n];
int dp[Max_n];
int main()
{
while(~scanf("%d",&n)&&n){
for(int i=1;i<=n;i++)scanf("%d",&a[i]);
int Max=0;
for(int i=1;i<=n;i++){
dp[i]=a[i];
for(int j=1;j<i;j++){
if(a[i]>a[j])dp[i]=max(dp[i],dp[j]+a[i]);
Max=max(Max,dp[i]);
}
}
printf("%d\n",Max);
}
return 0;
}
//完全背包
//dp[i][j]:表示前i中硬币重量达到j时的最小面值
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
const int inf=0x3f3f3f3f;
const int Max_n=1e4+10;
int t;
int e,f,n,W;
int p[Max_n],w[Max_n];
int dp[Max_n];
int main()
{
scanf("%d",&t);
while(t--){
scanf("%d%d%d",&e,&f,&n);
W=f-e;
for(int i=1;i<=n;i++)scanf("%d%d",&p[i],&w[i]);
memset(dp,0x3f,sizeof(dp));
dp[0]=0;
for(int i=1;i<=n;i++){
for(int j=w[i];j<=W;j++)
dp[j]=min(dp[j],dp[j-w[i]]+p[i]);
}
if(dp[W]!=inf)printf("The minimum amount of money in the piggy-bank is %d.\n",dp[W]);
else printf("This is impossible.\n");
}
return 0;
}
//dp[i][j]:表示第i秒落在j位置的最大馅饼
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
const int Max_n=1e5+10;
int n;
struct node{
int x,t;
bool operator<(const node& no){
return t<no.t;
}
}no[Max_n];
int dp[Max_n][22];
int main()
{
while(~scanf("%d",&n)&&n){
for(int i=0;i<n;i++){
scanf("%d%d",&no[i].x,&no[i].t);
no[i].x++;
}
sort(no,no+n);
memset(dp,-1,sizeof(dp));
dp[0][6]=0;
int k=0;
for(int i=1;i<Max_n;i++){
for(int j=1;j<=11;j++){
if(dp[i-1][j-1]!=-1&&dp[i-1][j-1]>dp[i][j])dp[i][j]=dp[i-1][j-1];
if(dp[i-1][j]!=-1&&dp[i-1][j]>dp[i][j])dp[i][j]=dp[i-1][j];
if(dp[i-1][j+1]!=-1&&dp[i-1][j+1]>dp[i][j])dp[i][j]=dp[i-1][j+1];
}
while(k<n&&no[k].t==i){
if(dp[i][no[k].x]!=-1)dp[i][no[k].x]++;
k++;
}
// for(int j=1;j<=11;j++)cout<<dp[i][j]<<' ';
// cout<<endl;
if(k==n){
int Max=0;
for(int j=1;j<=11;j++)
if(dp[i][j]>Max)Max=dp[i][j];
printf("%d\n",Max);
break;
}
}
}
return 0;
}
//dp[i][0]:表示第i个人单独买票的最少时间
//dp[i][1]:表示第i个人和前一个人一起买票的最短时间
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
const int Max_n=2e3+10;
int t,n;
int a[Max_n],b[Max_n];
int dp[Max_n][2];
int main()
{
scanf("%d",&t);
while(t--){
scanf("%d",&n);
for(int i=1;i<=n;i++)scanf("%d",&a[i]);
for(int i=1;i<=n-1;i++)scanf("%d",&b[i]);
memset(dp,0x3f,sizeof(dp));
dp[1][0]=a[1];
for(int i=2;i<=n;i++){
dp[i][0]=min(dp[i-1][0],dp[i-1][1])+a[i];
dp[i][1]=dp[i-1][0]-a[i-1]+b[i-1];
}
int ans=min(dp[n][0],dp[n][1]);
int s,m,h;
s=ans%60;ans/=60;
m=ans%60;h=ans/60;
h=(h+8)%24;
if(h>12)printf("%02d:%02d:%02d pm\n",h%12,m,s);
else printf("%02d:%02d:%02d am\n",h,m,s);
}
return 0;
}
//最长上升子序列
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
const int inf=0x3f3f3f3f;
const int Max_n=1e6+10;
int n;
int dp[Max_n];
int main()
{
while(~scanf("%d",&n)){
int c;
memset(dp,0x3f,sizeof(dp));
for(int i=0;i<n;i++){
scanf("%d",&c);
*lower_bound(dp,dp+Max_n,c)=c;
}
printf("%d\n",lower_bound(dp,dp+Max_n,inf)-dp);
}
return 0;
}