Description

你办了一场比赛,有n给人参加,只有一道题,有m个数据点,标号为1~m,每个测试点都有一个分数a[i]。现在所

有选手已经提交了程序并且测评完了,你知道每个人都能通过哪些测试点。你现在要安排捆绑测试的方式,把数据
点划分为若干个连续的区间,每个区间至少有一个测试点。每个区间只要有一个测试点错误就不会得分,如果所有
点都正确得分为所有测试点的分数的和。你的目的是最小化所有人的得分和。你需要对1<=i<=S,输出当把所有测
试点划分为i组时,最小的所有人分数和。

Input

第一行三个整数n,m,S
接下来一行m个整数,代表a[i]
接下来n行每行一个长度为m的01串,代表第i个人是否通过了第j个测试点
n<=50
m<=20000
S<=min(50,m)
a[i]<=10000,sigma a[i]*n<=2000000000

Output

S行,每行一个整数,代表当划分为i个捆绑测试点时所有人分数和的最小值

每次dp多分一段,dp时状态转移方程可分为O(n)段,且段之间的分界单调右移,用单调队列维护每个段的最值即可转移,时间复杂度O(nmS)

#include<bits/stdc++.h>
const int inf=0x7f7f7f7f;
int _(){
int x=,c=getchar();
while(c<)c=getchar();
while(c>)x=x*+c-,c=getchar();
return x;
}
void mins(int&a,int b){if(a>b)a=b;}
int min(int a,int b){return a<b?a:b;}
int n,m,q,a[],p0[];
char s[][];
int f[],t[],g[];
struct Q{
int q[],ql,qr,id;
void clr(){ql=,qr=;}
void ins(int x){
if(f[x]==inf)return;
int v=f[x]-a[x]*id;
while(ql<=qr){
int y=q[qr];
if(f[y]-a[y]*id>=v)--qr;
else break;
}
q[++qr]=x;
}
void del(int x){
if(ql<=qr&&q[ql]==x)++ql;
}
int gmn(int w){
if(ql>qr)return inf;
int x=q[ql];
return f[x]+(a[w]-a[x])*id;
}
}qs[];
int main(){
n=_(),m=_(),q=_();
for(int i=;i<=m;++i)a[i]=_()+a[i-];
for(int i=;i<=n;++i){
scanf("%s",s[i]+);
qs[i].id=i;
for(int j=;j<=m&&s[i][j]=='';++j)f[j]+=a[j];
}
printf("%d\n",f[m]);
for(int i=;i<=q;++i){
for(int j=;j<=n;++j){
p0[j]=;
qs[j].clr();
}
f[]=inf;
for(int j=;j<=m;++j){
qs[t[j]=n].ins(j-);
for(int k=;k<=n;++k)if(s[k][j]==''){
for(int z=p0[k]+;z<=j;++z){
qs[ t[z]].del(z-);
qs[--t[z]].ins(z-);
}
p0[k]=j;
}
g[j]=inf;
for(int k=;k<=n;++k)mins(g[j],qs[k].gmn(j));
}
memcpy(f,g,sizeof(int)*(m+));
printf("%d\n",f[m]);
}
return ;
}

bzoj4937: [Ceoi2016]popeala的更多相关文章

  1. [Ceoi2016|BZOJ4936] Match

    哈希+分治+stack 题目: 给你一个由小写字母组成的字符串s,要你构造一个字典序最小的(认为左括号的字典序比右括号小)合法的括号 序列与这个字符串匹配,字符串和括号序列匹配定义为:首先长度必须相等 ...

  2. 正睿OI DAY3 杂题选讲

    正睿OI DAY3 杂题选讲 CodeChef MSTONES n个点,可以构造7条直线使得每个点都在直线上,找到一条直线使得上面的点最多 随机化算法,check到答案的概率为\(1/49\) \(n ...

随机推荐

  1. xdoj-1022-A simple math problem 2 // 太强了

    //其实题目中f[n]的值可理解为存在多少个整数对使a*b<=n #include<cstdio> #define N 1007 #define maxn 1000005 using ...

  2. hdu4135 Co-prime 容斥原理

    Given a number N, you are asked to count the number of integers between A and B inclusive which are ...

  3. day28Spark

    PS:因为Spark是用内存运行 的,非常快 PS: 1.下面就是将conf的spark-env.template改变成spark-env.sh,并添加红色部分 2.修改slaves文件添加从设备 启 ...

  4. 关闭MongoDB服务的几种方法

    MongoDB 提供几种关闭服务的命令,具体为以下: 一 使用 Crtl+C 关闭 [mongo@redhatB data]$ mongod --dbpath=/database/mongodb/da ...

  5. Android USB gadget configfs学习笔记总结

    1.一个config_item 是通过显式用户空间mkdir操作创建的,通过rmdir销毁.属性(文件)在mkdir之后出现,可以通过read和write读取或修改属性文件.与sysfs一样,read ...

  6. python结合redis模拟队列

    实在无聊就写了个很小的python程序用来实现模拟redis队列的代码如下: redis_lpush.py   #!/usr/bin/python3 import time import redis ...

  7. java自学总结

    经过了一段时间的java学习,感觉自己在编程方面还只是一个初学者,感觉学会了c,在学c++的时候就是以c为基础,java应该也是以c或者c++为基础,但是并非如此,java和c++虽然有一些相似之处, ...

  8. python之路---11 第一类对象 函数名 闭包 迭代器

    二十九. 1.函数名的运用    ①函数名是⼀个变量, 但它是⼀个特殊的变量, 与括号配合可以执⾏函数的变量 ②函数名是一个内存地址    ③ 函数名可以赋值给其他变量         ④函数名可以当 ...

  9. 混合pyqt和qtcreator (2): Impl a image viewer (can show FIji ROI manager data)

    # -*- coding: utf-8 -*- """ Simple demonstration of TreeWidget, which is an extension ...

  10. Reactor/Proactor的比较 (ZZ)

    一般情况下,I/O 复用机制需要事件分享器(event demultiplexor [1.3]). 事件分享器的作用,即将那些读写事件源分发给各读写事件的处理者,就像送快递的在楼下喊: 谁的什么东西送 ...