花了两个晚上来搞这道题。

第一个晚上想思路和写代码,第二个晚上调试。

然而还是菜,一直调不对,我的队友是Debug小能手呀(真的是无敌,哈哈,两个人一会就改好了)

D. Timetable
 
time limit per test

2 seconds

memory limit per test

256 megabytes

input

standard input

output

standard output

Ivan is a student at Berland State University (BSU). There are n days in Berland week, and each of these days Ivan might have some classes at the university.

There are m working hours during each Berland day, and each lesson at the university lasts exactly one hour. If at some day Ivan's first lesson is during i-th hour, and last lesson is during j-th hour, then he spends j - i + 1 hours in the university during this day. If there are no lessons during some day, then Ivan stays at home and therefore spends 0 hours in the university.

Ivan doesn't like to spend a lot of time in the university, so he has decided to skip some lessons. He cannot skip more than k lessons during the week. After deciding which lessons he should skip and which he should attend, every day Ivan will enter the university right before the start of the first lesson he does not skip, and leave it after the end of the last lesson he decides to attend. If Ivan skips all lessons during some day, he doesn't go to the university that day at all.

Given nmk and Ivan's timetable, can you determine the minimum number of hours he has to spend in the university during one week, if he cannot skip more than k lessons?

Input

The first line contains three integers nm and k (1 ≤ n, m ≤ 500, 0 ≤ k ≤ 500) — the number of days in the Berland week, the number of working hours during each day, and the number of lessons Ivan can skip, respectively.

Then n lines follow, i-th line containing a binary string of m characters. If j-th character in i-th line is 1, then Ivan has a lesson on i-th day during j-th hour (if it is 0, there is no such lesson).

Output

Print the minimum number of hours Ivan has to spend in the university during the week if he skips not more than k lessons.

Examples
input

Copy
2 5 1
01001
10110
output
5
input

Copy
2 5 0
01001
10110
output
8
Note

In the first example Ivan can skip any of two lessons during the first day, so he spends 1 hour during the first day and 4 hours during the second day.

In the second example Ivan can't skip any lessons, so he spends 4 hours every day.

这个题的意思就是逃课。怎样逃课才能使得最后上完课的时间减去最开始的上课的时间最短。

这里是按0和1来代表有没有课,1代表有课,可以逃。但是最多只能逃k节课。否则就直接坐飞机了。。。

样例01001如果一节也不逃的话,就是最后一个1的位置减去第一个1的位置的距离。为4。

想思路真的是炸裂。

因为是找最优子结构,所以肯定是用动态规划写,有人说是背包,就发现是分组背包。

然后就想怎么处理数据才能用上分组背包。

因为是和1的 位置有关,所以先记录每一组的1的位置。然后num数组计数多少个1。

然后,就是,怎么处理记录位置的数组才能得到距离呢。如果直接贪心的话肯定是不对的,如果是01001010010010,贪心的话,就不对了。所以要考虑一下。

怎么处理呢,举例,一开始的位置是0,假设数组a存位置就是a[1]=0,a[2]=2,a[3]=3,a[4]=5,a[5]=9。

然后进行删0个1,删1个1,删2个1,删3个1,删。。。一直删num个1。怎么删呢。然后就模拟所有情况。这样做的意思就是假设只处理这一个串。后面会说为什么这样做。

(写的太乱了,凑活看。。。)

发现有重复的情况,才突然想起来(mdzz),肯定是从两边删才是最完美的呀。就不会有重复了。

但是怎么才能避免呢。贪心肯定不对,然后就想到一种遍历,怎么遍历呢。距离肯定是最后的记录位置的数减去最开始的记录位置的数。我想的就是一个for控制组数,一个for控制删几个数,一个for控制进行操作的两个数的位置,什么意思呢。因为知道有几个数(num数组存的),所以删j个数的话,就是从两边一共删j个数。还是用上面的例子。

a[1]=0,a[2]=2,a[3]=3,a[4]=5,a[5]=9。

这个是什么意思呢。就是删1个1的操作,就是控制位置。我写的是从最后往前推。一直推到头就可以了。然后for循环一边推,一边用数组保存最小值就可以。

但是写的时候智障,只考虑了一直都还有1的情况,没考虑所有1都删掉的情况。还是上面的例子,有5个1,如果5个1都删掉,距离就是0,不是1,这里判断一下。

然后就是写for的时候比较一下这组串中1的个数和要删掉的1的个数哪个小就用哪个作为删j个1的边界值。为什么呢。假设串里就2个1,题目要求删5个1,完全没必要一直删5个1,删2个1就没了,所以比较一下,如果1个个数本来就比要求的删的个数小的话,那肯定是把串里所有的1都删了,所以最后距离是0,这里判断一下就可以。

通过以上的炸裂操作,就可以得到每组串里删j个1的最优值。

自我吐槽:脑子不好,控制位置那里想了好久,最后发现,因为我存位置是从第一个1出现位置开始存的,所以存位置的数组最开始是从1开始的,不是从0开始的。

 for(int i=;i<n;i++){  //控制组数
int x=min(k,num[i]); //该组中1的个数和要处理的k个1比较一下,取最小值
for(int j=;j<=x;j++){ //控制删几个1
for(int k=;k<=j;k++){ //控制位置
if(j==num[i]) //1全删完的判断
p[i][j]=;
else
p[i][j]=min(p[i][j],a[i][num[i]-k]-a[i][+j-k]+); //存最短的距离
}
}
}

就是上面这个傻子操作。。。

然后就是怎么用分组背包来得到结果呢。

一篇简短介绍分组背包的博客,传送门:我不想当咸鱼

那么背包的最大能装的质量就是题目要求删的1的个数,就是k。最大质量为k。

然后每组的每个物品的质量就是删几个1,如果是删0个1的,那么质量为0,删1个1的,质量就是2,直接就是p[i][j]的j值。

物品的价值就是距离。因为我写的分组背包是用max求最大值的,所以就要找最大节省的时间(就是存位置的一开始的值减去最短距离就是最大节省时间)

然后就可以进行分组背包的操作了。求出来最大节省的时间,然后总的不逃课的时间-最大节省的时间,就是最少的上课时间。游戏结束。

代码:

 //D-数据处理+分组背包(处理炸裂)
#include<iostream>
#include<cstring>
#include<cstdio>
#include<cmath>
#include<algorithm>
#include<cstdlib>
using namespace std;
typedef long long ll;
const int maxn=+;
const int inf=+;
#define ios ios::sync_with_stdio(false);cin.tie(0);cout.tie(0);
int h[maxn][maxn],a[maxn][maxn];
int p[maxn][maxn],num[maxn],dp[maxn];
char s[maxn][maxn];
int main(){
int n,m,k;
ios;
cin>>n>>m>>k;
for(int i=;i<n;i++)
cin>>s[i];
for(int i=;i<n;i++){
for(int j=;j<m;j++)
h[i][j]=s[i][j]-''; //h数组存串
}
memset(num,,sizeof(num));
for(int i=;i<n;i++){ //记录位置
for(int j=;j<m;j++){
if(h[i][j]==){
num[i]++; //记录1的个数
a[i][num[i]]=j; //记录第几个1的位置
}
}
}
for(int i=;i<n;i++){ //初始化
for(int j=;j<=min(k,num[i]);j++)
p[i][j]=inf;
}
for(int i=;i<n;i++){ //i控制组数 处理距离
int x=min(k,num[i]); //找最小的删1的个数
for(int j=;j<=x;j++){ //控制删1的操作
for(int k=;k<=j;k++){ //控制位置
if(j==num[i]) //特判一下
p[i][j]=;
else
p[i][j]=min(p[i][j],a[i][num[i]-k]-a[i][+j-k]+); //还有1存在的正常情况
}
}
}
ll sum=;
for(int i=;i<n;i++)
sum+=p[i][]; //将一开始一个1都不删的距离求和
memset(dp,,sizeof(dp));
for(int i=;i<n;i++) //i控制组数 分组背包
for(int j=k;j>=;j--) //j控制质量
for(int h=;h<=min(k,num[i]);h++) //h控制每组的物品数,h也是物品的质量,想一下就懂了
if(j>=h) //一开始没写,数组可能会越界,写上就对了。。。
dp[j]=max(dp[j-h]+(p[i][]-p[i][h]),dp[j]);
// for(int i=0;i<n;i++)
// {
// for(int j=0;j<=k;j++)
// {
// cout<<p[i][j]<<" ";
// }
// cout<<endl;
// }
// for(int i=0;i<=k;i++)
// {
// cout<<dp[i]<<endl;
// }
// cout<<sum<<endl;
cout<<sum-dp[k]<<endl; //游戏结束
}
//2 7 2
//0100101
//

就这样,这题就是处理数据不好想。。。

就这样吧。溜了。。。

Codeforces 946 D.Timetable-数据处理+动态规划(分组背包) 处理炸裂的更多相关文章

  1. Codeforces 946D Timetable(预处理+分组背包)

    题目链接:http://codeforces.com/problemset/problem/946/D 题目大意:有n个字符串,代表n天的课表,1表示这个时间要上课,0表示不要上课,一天在学校时间为第 ...

  2. Codeforces Round #383 (Div. 2) D 分组背包

    给出一群女孩的重量和颜值 和她们的朋友关系 现在有一个舞台 ab是朋友 bc是朋友 ac就是朋友 给出最大承重 可以邀请这些女孩来玩 对于每一个朋友团体 全邀请or邀请一个or不邀请 问能邀请的女孩的 ...

  3. BZOJ1296 [SCOI2009]粉刷匠 动态规划 分组背包

    欢迎访问~原文出处——博客园-zhouzhendong 去博客园看该题解 题目传送门 - BZOJ1296 题意概括 有 N 条木板需要被粉刷. 每条木板被分为 M 个格子. 每个格子要被刷成红色或蓝 ...

  4. Codeforces 37D Lesson Timetable - 组合数学 - 动态规划

    题目传送门 神奇的门I 神奇的门II 题目大意 有$n$组学生要上课2次课,有$m$个教室,编号为$1$到$m$.要确定有多少种不同的安排上课的教室的方案(每组学生都是本质不同的),使得它们满足: 每 ...

  5. [HDU 3033] I love sneakers! (动态规划分组背包)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3033 题意:给你K种品牌,每种品牌有不同种鞋,现在每种品牌至少挑一款鞋,问获得的最大价值,如果不能每种 ...

  6. #分组背包 Educational Codeforces Round 39 (Rated for Div. 2) D. Timetable

    2018-03-11 http://codeforces.com/contest/946/problem/D D. Timetable time limit per test 2 seconds me ...

  7. Codeforces 946D - Timetable (预处理+分组背包)

    题目链接:Timetable 题意:Ivan是一个学生,在一个Berland周内要上n天课,每天最多会有m节,他能逃课的最大数量是k.求他在学校的时间最小是多少? 题解:先把每天逃课x节在学校呆的最小 ...

  8. CodeForces - 946D Timetable (分组背包+思维)

    题意 n天的课程,每天有m个时间单位.若时间i和j都有课,那么要在学校待\(j-i+1\)个时间.现在最多能翘k节课,问最少能在学校待多少时间. 分析 将一天的内容视作一个背包的组,可以预处理出该天内 ...

  9. CJOJ 2040 【一本通】分组背包(动态规划)

    CJOJ 2040 [一本通]分组背包(动态规划) Description 一个旅行者有一个最多能用V公斤的背包,现在有n件物品,它们的重量分别是W1,W2,...,Wn,它们的价值分别为C1,C2, ...

随机推荐

  1. 定义一个Rectangle类,该类提供getLength和getWidth方法。

    import java.util.Comparator; /** * 定义一个Rectangle类,该类提供getLength和getWidth方法.利用图1-18中的findMax例程编写 * 一种 ...

  2. Unity shader学习之屏幕后期处理效果之高斯模糊

    高斯模糊,见 百度百科. 也使用卷积来实现,每个卷积元素的公式为: 其中б是标准方差,一般取值为1. x和y分别对应当前位置到卷积中心的整数距离. 由于需要对高斯核中的权重进行归一化,即使所有权重相加 ...

  3. idea创建java的web项目

    2. 3. 4. 5. 6. 步骤八: 点击那个倒立的三角形,然后点击Edit Configurations; 步骤八: 步骤九:配置tocat服务器 步骤十:哎,发现,我怎么就只有一个选项呀,art ...

  4. .NET Core Tools for Visual Studio 2015 安装失败

    You may be blocked from installing the .NET Core Tooling Preview 2 for Visual Studio 2015 installer ...

  5. codeforces 984B Minesweeper

    题意: 给出一个矩阵,如果一个格子是数字,那么与这个格子相邻的格子中有炸弹的数量必须等于这个格子中的数字: 如果一个格子是空地,那么这个格子的所有相邻的格子中就不能有炸弹. 判断这个矩阵是否合法. 思 ...

  6. arm cortex-m0plus源码学习(二)AMBA3.0_ AHBLite

    1. AMBA总线概述 AMBA2.0 以上版本都是基于单沿时钟.单向信号线的协议[1]. 现在市场上大部分的基于 AMBA 架构的 SoC 产品, 系统总线采用 AHB, 外部总线采用 APB.系统 ...

  7. 转:C#串口编程

    本文用来简单介绍一下C#串口编程的知识,主要以实例为内容. 凡是串口设备和计算机交互的时候都用到串口,在C#中我们如何来操作串口呢? 大话串口工作原理 实际串口是用来和外部设备进行交换数据的,我抽象出 ...

  8. hashCode 一致性hash 算法

    1 如果两个对象相同,那么它们的hashCode值一定要相同.也告诉我们重写equals方法,一定要重写 hashCode方法,同一个对象那么hashcode就是同一个(同一个对象什么都是相同的).2 ...

  9. .NET 常用ORM之Gentle.Net

    .Net常用的就是微软的EF框架和Nhibernate,这两个框架用的都比较多就不做详细介绍了,今天我们来看看Gentle.Net,Gentle.Net是一个开源的优秀O/R Mapping的对象持久 ...

  10. P1012 拼数

    P1012 拼数 输入输出样例 输入样例 3 13 312 343 输出样例 34331213 注意 当你输入: 6321 32 407 135 13 217 应该输出: 40732321217135 ...