LG4158 「SCOI2009」粉刷匠 线性DP

问题描述

LG4158


题解

设\(opt[i][j][k]\)代表到\((i,k)\)刷了\(j\)次的方案数。

一开始DP顺序有点问题,调了很长时间。

务必考虑清楚DP顺序问题


\(\mathrm{Code}\)

#include<bits/stdc++.h>
using namespace std;

template <typename Tp>
void read(Tp &x){
    x=0;char ch=1;int fh;
    while(ch!='-'&&(ch<'0'||ch>'9')) ch=getchar();
    if(ch=='-'){
        fh=-1;ch=getchar();
    }
    else fh=1;
    while(ch>='0'&&ch<='9'){
        x=(x<<1)+(x<<3)+ch-'0';
        ch=getchar();
    }
    x*=fh;
}

const int maxn=53;

int n,m,t;
int a[maxn][maxn],s[maxn][maxn*maxn],opt[maxn][maxn*maxn][maxn];
int f[maxn][maxn*maxn];

void println_opt(){
    for(int i=1;i<=n;i++){
        for(int j=1;j<=m;j++){
            for(int k=0;k<=t;k++){
                printf("opt[%d][%d][%d]=%d\n",i,j,k,opt[i][j][k]);
            }
        }
    }
}

int main(){
    read(n);read(m);read(t);
    for(int i=1;i<=n;i++){
        for(int j=1;j<=m;j++){
            scanf("%1d",&a[i][j]);
            s[i][j]=s[i][j-1]+a[i][j];
        }
    }
//  opt[1][0][0]=1;
    for(int i=1;i<=n;i++){
        for(int j=1;j<=m;j++){
//          int ed=min((i-1)*n+j,t);
            for(int k=1;k<=m;k++){
//              opt[i][j][k]=opt[i][0][k-1]+max(s[i][k],k-s[i][k]);
                for(int p=j-1;p<k;p++){
                    opt[i][j][k]=max(opt[i][j][k],opt[i][j-1][p]+max(s[i][k]-s[i][p],k-p-(s[i][k]-s[i][p])));
                }
            }
        }
        //for(int j=1;j<=t;j++) opt[i+1][0][j]=opt[i][m][j];
    }
    for(int i=1;i<=n;i++){
        for(int j=1;j<=t;j++){
            for(int k=0;k<=min(j,m);k++){
                f[i][j]=max(f[i][j],f[i-1][j-k]+opt[i][k][m]);
            }
        }
    }
    int ans=0;
    for(int i=1;i<=t;i++) ans=max(ans,f[n][i]);
    printf("%d\n",ans);
//  println_opt();
    return 0;
}

原文地址:https://www.cnblogs.com/liubainian/p/11617714.html

时间: 2024-10-31 01:43:19

LG4158 「SCOI2009」粉刷匠 线性DP的相关文章

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 次,他最多能正确粉刷多

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

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

bzoj1296【SCOI2009】粉刷匠

1296: [SCOI2009]粉刷匠 Time Limit: 10 Sec  Memory Limit: 162 MB Submit: 1479  Solved: 837 [id=1296" style="color:blue; text-decoration:none">Submit][Status][Discuss] Description windy有 N 条木板须要被粉刷. 每条木板被分为 M 个格子. 每一个格子要被刷成红色或蓝色. windy每次粉刷.仅仅

[SCOI2009][BZOJ1296] 粉刷匠

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

JZOJ 1035. 【SCOI2009】粉刷匠

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

LG2145 「JSOI2007」祖码 区间DP

问题描述 LG2145 题解 把颜色相同的一段看做一个点. 然后类似于合唱队区间DP即可. 但是这题好像出过一些情况,导致我包括题解区所有人需要特判最后一个点. \(\mathrm{Code}\) #include<bits/stdc++.h> using namespace std; template <typename Tp> void read(Tp &x){ x=0;char ch=1;int fh; while(ch!='-'&&(ch<'0

「SCOI2009」windy数

传送门 Luogu 解题思路 数位 \(\text{DP}\) 设状态 \(dp[now][las][0/1][0/1]\) 表示当前 \(\text{DP}\) 到第 \(i\) 位,前一个数是 \(las\),有没有顶到上界,有没有前导零的答案. 转移十分显然. 细节注意事项 咕咕咕 参考代码 #include <algorithm> #include <iostream> #include <cstring> #include <cstdlib> #i

LibreOJ #2002. 「SDOI2017」序列计数

二次联通门 : LibreOJ #2002. 「SDOI2017」序列计数 /* LibreOJ #2002. 「SDOI2017」序列计数 线性筛 + 矩阵优化dp 先构造出全部情况的矩阵 用矩阵快速幂计算答案 再构造出全不是质数的矩阵 计算出答案 前一个答案减后一个答案即可 */ #include <cstdio> #include <iostream> #include <cstring> const int BUF = 12312312; char Buf[BU

1296: [SCOI2009]粉刷匠

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