[SCOI2009]粉刷匠(动态规划,序列dp,背包)

分别对每块木板做区间dp,设\(g[i][j]\)表示前i个格子,刷恰好j次,并且第i格是合法的最多合法的格子数.从前往后枚举断点来转移就好了.

这样处理再出来\(g[i][j]\)每一块木板i刷j次的最大合法格子数.

最后再合并每块木板的答案,用\(dp[i][j]\)表示前i块木板,一共恰好刷了k次的最大合法格子数,用刷表法暴力背包合并就好了.

很详细的注释.

#include<iostream>
#include<cstring>
#include<cstdio>
#define maxn 55
using namespace std;
int n,m,T,ans,sum[maxn],dp[maxn][maxn*maxn];
int f[maxn][maxn],g[maxn][maxn];
char s[maxn];
//分别对每块木板区间dp
//再用背包来合并
int main()
{
    cin>>n>>m>>T;
    for(int i=1;i<=n;i++)
    {
        scanf("%s",s+1);
        memset(g,0x8f,sizeof(g));
        g[0][0]=0;
        for(int j=1;j<=m;j++)sum[j]=sum[j-1]+(s[j]=='1');//蓝色的前缀和
        //g[j][k]当前木板表示前j个格子,刷k次的最多合法的格子数,并且j是合法的格子
        for(int j=1;j<=m;j++)
            for(int k=1;k<=j;k++)
                for(int l=0;l<j;l++)
                {
                    if(s[j]=='1')g[j][k]=max(g[j][k],g[l][k-1]+sum[j]-sum[l]);
                    else g[j][k]=max(g[j][k],g[l][k-1]+j-l-sum[j]+sum[l]);
                }
        //f[j][k]表示第i块木板,恰好刷k次的最多合法格子数
        for(int j=0;j<=m;j++)
            for(int k=0;k<=j;k++)
                f[i][k]=max(f[i][k],g[j][k]);
    }
    //dp[i][j]表示前i个块木板,恰好刷j次的最多的合法格子数
    for(int i=1;i<=n;i++)
        for(int j=0;j<=T&&j<=i*m-m;j++)
            for(int k=0;k<=m;k++)
                dp[i][j+k]=max(dp[i][j+k],dp[i-1][j]+f[i][k]);
    for(int i=0;i<=T;i++)ans=max(ans,dp[n][i]);
    cout<<ans<<endl;
    return 0;
}

原文地址:https://www.cnblogs.com/terribleterrible/p/9879111.html

时间: 2024-10-11 10:40:11

[SCOI2009]粉刷匠(动态规划,序列dp,背包)的相关文章

BZOJ 1296 SCOI2009 粉刷匠 动态规划

题目大意:给定n*m的木板,每个点需要刷成1和0两种颜色之一,每次只能刷一行中连续的一段,一个点只能刷一次,求T刷子最多能刷对多少个点 首先对每行拆开处理 令f[i][j]为用i刷子刷前j个格子最多刷对多少个点 动规处理出这一行刷i刷子最多能刷对多少个点 然后分组背包即可 #include<cstdio> #include<cstring> #include<iostream> #include<algorithm> #define M 60 using n

BZOJ 1296: [SCOI2009]粉刷匠 分组DP

1296: [SCOI2009]粉刷匠 Description windy有 N 条木板需要被粉刷. 每条木板被分为 M 个格子. 每个格子要被刷成红色或蓝色. windy每次粉刷,只能选择一条木板上一段连续的格子,然后涂上一种颜色. 每个格子最多只能被粉刷一次. 如果windy只能粉刷 T 次,他最多能正确粉刷多少格子? 一个格子如果未被粉刷或者被粉刷错颜色,就算错误粉刷. Input 输入文件paint.in第一行包含三个整数,N M T. 接下来有N行,每行一个长度为M的字符串,'0'表示

1296: [SCOI2009]粉刷匠[多重dp]

1296: [SCOI2009]粉刷匠 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 1919  Solved: 1099[Submit][Status][Discuss] Description windy有 N 条木板需要被粉刷. 每条木板被分为 M 个格子. 每个格子要被刷成红色或蓝色. windy每次粉刷,只能选择一条木板上一段连续的格子,然后涂上一种颜色. 每个格子最多只能被粉刷一次. 如果windy只能粉刷 T 次,他最多能正确粉刷多

1296: [SCOI2009]粉刷匠

1296: [SCOI2009]粉刷匠 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 916  Solved: 532[Submit][Status] Description windy有 N 条木板需要被粉刷. 每条木板被分为 M 个格子. 每个格子要被刷成红色或蓝色. windy每次粉刷,只能选择一条木板上一段连续的格子,然后涂上一种颜色. 每个格子最多只能被粉刷一次. 如果windy只能粉刷 T 次,他最多能正确粉刷多少格子? 一个格子如果

BZOJ 1296: [SCOI2009]粉刷匠

BZOJ 1296: [SCOI2009]粉刷匠 Description windy有 N 条木板需要被粉刷. 每条木板被分为 M 个格子. 每个格子要被刷成红色或蓝色. windy每次粉刷,只能选择一条木板上一段连续的格子,然后涂上一种颜色. 每个格子最多只能被粉刷一次. 如果windy只能粉刷 T 次,他最多能正确粉刷多少格子? 一个格子如果未被粉刷或者被粉刷错颜色,就算错误粉刷. Input 输入文件paint.in第一行包含三个整数,N M T. 接下来有N行,每行一个长度为M的字符串,

BZOJ1296: [SCOI2009]粉刷匠

1296: [SCOI2009]粉刷匠 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 844  Solved: 486[Submit][Status] Description windy有 N 条木板需要被粉刷. 每条木板被分为 M 个格子. 每个格子要被刷成红色或蓝色. windy每次粉刷,只能选择一条木板上一段连续的格子,然后涂上一种颜色. 每个格子最多只能被粉刷一次. 如果windy只能粉刷 T 次,他最多能正确粉刷多少格子? 一个格子如果

【Dp】Bzoj1296 [SCOI2009] 粉刷匠

Description windy有 N 条木板需要被粉刷. 每条木板被分为 M 个格子. 每个格子要被刷成红色或蓝色. windy每次粉刷,只能选择一条木板上一段连续的格子,然后涂上一种颜色. 每个格子最多只能被粉刷一次. 如果windy只能粉刷 T 次,他最多能正确粉刷多少格子? 一个格子如果未被粉刷或者被粉刷错颜色,就算错误粉刷. Input 输入文件paint.in第一行包含三个整数,N M T. 接下来有N行,每行一个长度为M的字符串,'0'表示红色,'1'表示蓝色. Output 输

[bzoj1296][SCOI2009]粉刷匠(泛化背包)

http://www.lydsy.com:808/JudgeOnline/problem.php?id=1296 分析: 首先预处理出每一行的g[0..T]表示这一行刷0..T次,最多得到的正确格子数.这个做法只要把每一行连续的个数分别求出来,然后从大到小累加就行 然后整体来看,就是对n个物品做泛化背包了.

【luogu2657】【bzoj1026】 [SCOI2009]windy数 [动态规划 数位dp]

P2657 [SCOI2009]windy数 bzoj1026 一本通说这是一道数位dp模板题 emmmmm 就是逐位确定 f[i][j]表示填了i位数其最高位数字为j 然后就去求可能方案数 分为 不满足x的位数的严格小于x的全部情况 和x的位数相同 但最高位小于x的最高为的全部方案数 和x的位数相同 有一位比x的对应位数小的全部方案数 其余位数对应数字都相同(这是数位dp常用的一个性质:对于一个小于n的数 它从高位到低位一定会出现某一位上的数字小于n所对应这一位上的数字) PS 因为x不一定为