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. 10 HashMap,Map.Entry,LinkedHashMap,TreeMap,Hashtable,Collections类

    Map集合的功能概述 添加功能 * V put(K key,V value):添加元素.            * 如果键是第一次存储,就直接存储元素,返回null            * 如果键不 ...

  2. mysql插入操作跳过(ignore)、覆盖(replace into)、更新(on duplicate key)

    原帖地址:http:.html .insert ignore into 当插入数据时,如出现错误时,如重复数据,将不返回错误,只以警告形式返回.所以使用ignore请确保语句本身没有问题,否则也会被忽 ...

  3. $trainClassLayer.find('input[name=data-item-checkbox]').eq(index).change();//激活第index+1那个checkbox

    ☆ $.each(data, function (index, org) { if (org.alreadySent) { $trainClassLayer.find('input[name=data ...

  4. 【BZOJ3144】【HNOI2013】切糕

    总算做了一道2011以后的省选题了……原题: 图片题面好评! P,Q,R≤40,0≤D≤R,给出的所有的不和谐值不超过1000. 文本样例好评! 恩这个是听妹主席讲过后会写的,首先把每个点拆成链,那么 ...

  5. sqler sql 转rest api 数据聚合操作

    sqler 2.0 提供了一个新的指令aggregate,注意这个和sql 的聚合函数不是一个概念,这个只是为了 方便api数据的拼接 参考格式   databases {    exec = &qu ...

  6. Unity5 图形系统介绍 学习

  7. adb command

  8. 艰苦的编译boost python (失败)

    1.下载 boost_1_67_0 2.在目录下执行 bootstrap 3.将python36添加到path环境变量 4.执行 b2 --with-python,将会声场如下dll 2018/04/ ...

  9. app崩溃后自动重启

    android 引用:http://blog.csdn.net/caiwenfeng_for_23/article/details/41184353 package com.tan.abnormalr ...

  10. Linux split命令参数及用法详解---linux分割文件命令

    转载自:http://blog.csdn.net/xiaoshunzi111/article/details/52173994 功能说明:分割文件. Split:按指定的行数截断文件 格式: spli ...