题目链接OvO

题目大意

  给你\(n\)串数字,\(1\)代表该位置是亮的,\(0\)代表是灭的。你必须修改\(k\)个数字,使某些\(0\)变为\(1\)。注意,只能把原来的\(0\)改成\(1\)。

分析

  由于每串数字上的\(1\)是不能修改的,所以每串数字并不一定能完整的表示\(0-9\)之内的所有数,所有需要先对每串数字做一下预处理,计算出能改为哪些数字和修改的代价。然后第一步,我们需要判断是否可以构成\(n\)个完整数字。这个直接来判断似乎不太好做。但是可以转换一下,变成能否只消耗\(k\)构成\(n\)个完整数字。这个类似于背包问题,问能否用\(n\)个数构成\(k\)。

  状态转移方式为:若第\(i\)步可以到达容量\(j\),那么第\(i+1\)步就可以到达\(j+cost[i+1]\)。

  我们将上面稍微变形,不难得出:\(dp[i][j] |= dp[i-1][j-cost[i][t]],0\leq t\leq 9\)。

  所以只要判断\(dp[n][k]\)是否为\(true\)就能知道能否只消耗\(k\)显示\(n\)个完整数字了。但是这还不够,答案还需要输出最大的可能结果。我们通过上面的\(dp\)已经获得了一个状态转移的图,红色终点代表错误的终点,黑色终点代表正确的终点画出来(只是部分)大概是这样子的(这里只有\(end1\)是正确的,其他的要么是容量不为\(k\)(\(end2\)),要么是选的个数不够(\(end3\))):



  我们的\(dp\)数组里存储了从起始点到终点的所有路径,可以看出,只要从终点向起点方向走,一定能走到起点,每次消耗为\(cost[i][t]\)。但是题目要求我们输出答案最大,答案最大的话就要要求大的数字靠前,所以倒着从最后一个数开始贪心肯定是不行的,我们需要正着贪心。那么我们就只能倒着\(dp\),这样我们就得到一个从最后一个数到第一个数的状态转移图。这时候我们只需要从第一个数开始贪心的每步选择能够到达的,代表数字最大的路径就行了。这样的话第一步的状态转移方程就变成了\(dp[i][j] |= dp[i+1][j-cost[i][k]],0\leq t\leq 9\)。

具体实现

第一步

  先预处理,求每串数变成每个数字的消耗,显示不了的数字消耗就是\(-1\),存入\(cost\)数组中。

    char nums[][10] = {"1110111","0010010","1011101","1011011","0111010","1101011","1101111","1010010","1111111","1111011"};
char str[maxn][10];
void solve(int pos) {
for (int i = 9; i>=0; --i) {
int cnt = 0;
for (int j = 0; j<7; ++j) {
if (str[pos][j]=='1'&&nums[i][j]=='0') {
cnt = -1;
break;
}
if (str[pos][j]=='0'&&nums[i][j]=='1') ++cnt;
}
cost[pos][i] = cnt;
}
}

第二步

  根据\(dp[i][j] |= dp[i+1][j-cost[i][k]],0\leq t\leq 9\)倒着得到\(dp\)数组的值。

        dp[n+1][0] = true;
for (int i = n; i>=1; --i)
for (int k = 9; k>=0; --k)
if (~cost[i][k])
for (int j = m; j>=cost[i][k]; --j)
dp[i][j] |= dp[i+1][j-cost[i][k]];
if (!dp[1][m]) {
cout << -1 << endl;
return 0;
}

第三步

  正着贪心得到最大可能的数字。

        for (int i = 1; i<=n; ++i)
for (int j = 9; j>=0; --j)
if (~cost[i][j] && m>=cost[i][j] && dp[i+1][m-cost[i][j]]) {
cout << j;
m -= cost[i][j];
break;
}

完整代码

    const int maxn = 2e3+10;
char nums[][10] = {"1110111","0010010","1011101","1011011","0111010","1101011","1101111","1010010","1111111","1111011"};
char str[maxn][10];
int n, m, dp[maxn][maxn], cost[maxn][10];
void solve(int pos) {
for (int i = 9; i>=0; --i) {
int cnt = 0;
for (int j = 0; j<7; ++j) {
if (str[pos][j]=='1'&&nums[i][j]=='0') {
cnt = -1;
break;
}
if (str[pos][j]=='0'&&nums[i][j]=='1') ++cnt;
}
cost[pos][i] = cnt;
}
}
int main(void) {
cin >> n >> m;
for (int i = 1; i<=n; ++i) {
cin >> str[i];
solve(i);
}
dp[n+1][0] = true;
for (int i = n; i>=1; --i)
for (int k = 9; k>=0; --k)
if (~cost[i][k])
for (int j = m; j>=cost[i][k]; --j)
dp[i][j] |= dp[i+1][j-cost[i][k]];
if (!dp[1][m]) {
cout << -1 << endl;
return 0;
}
for (int i = 1; i<=n; ++i)
for (int j = 9; j>=0; --j)
if (~cost[i][j] && m>=cost[i][j] && dp[i+1][m-cost[i][j]]) {
cout << j;
m -= cost[i][j];
break;
}
cout << endl;
return 0;
}

Codeforces 1340B Nastya and Scoreboard(dp,贪心)的更多相关文章

  1. CF#637 D. Nastya and Scoreboard DP

    D. Nastya and Scoreboard 题意 一块电子屏幕上有n个数字. 每个数字是通过这样7个线段显示的,现在你不小心打坏了k个线段,给出打坏之后的n个数字的显示方式,问之前的屏幕表示的最 ...

  2. Codeforces 1136D - Nastya Is Buying Lunch - [贪心+链表+map]

    题目链接:https://codeforces.com/problemset/problem/1136/D 题意: 给出 $1 \sim n$ 的某个排列 $p$,再给出若干 $(x,y)$ 表示当序 ...

  3. Codeforces 1136D Nastya Is Buying Lunch (贪心)

    题意: 给一个序列和一组交换序列(a,b),当且仅当a在b的前面(不允许有间隔),这两个数才能交换,问最后一个数最多能移动多少个位置. 分析: 这题是思路是十分的巧妙呀 , 用一个数组num[x]  ...

  4. Codeforces 437C The Child and Toy(贪心)

    题目连接:Codeforces 437C  The Child and Toy 贪心,每条绳子都是须要割断的,那就先割断最大值相应的那部分周围的绳子. #include <iostream> ...

  5. Codeforces Round #546 (Div. 2) D 贪心 + 思维

    https://codeforces.com/contest/1136/problem/D 贪心 + 思维 题意 你面前有一个队列,加上你有n个人(n<=3e5),有m(m<=个交换法则, ...

  6. 【bzoj4027】[HEOI2015]兔子与樱花 树形dp+贪心

    题目描述 很久很久之前,森林里住着一群兔子.有一天,兔子们突然决定要去看樱花.兔子们所在森林里的樱花树很特殊.樱花树由n个树枝分叉点组成,编号从0到n-1,这n个分叉点由n-1个树枝连接,我们可以把它 ...

  7. BZOJ 2021 [Usaco2010 Jan]Cheese Towers:dp + 贪心

    题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=2021 题意: John要建一个奶酪塔,高度最大为m. 他有n种奶酪.第i种高度为h[i]( ...

  8. [BZOJ 3625] [Codeforces 438E] 小朋友的二叉树 (DP+生成函数+多项式开根+多项式求逆)

    [BZOJ 3625] [Codeforces 438E] 小朋友的二叉树 (DP+生成函数+多项式开根+多项式求逆) 题面 一棵二叉树的所有点的点权都是给定的集合中的一个数. 让你求出1到m中所有权 ...

  9. 洛谷P2507 [SCOI2008]配对 题解(dp+贪心)

    洛谷P2507 [SCOI2008]配对 题解(dp+贪心) 标签:题解 阅读体验:https://zybuluo.com/Junlier/note/1299251 链接题目地址:洛谷P2507 [S ...

随机推荐

  1. Java基础语法(9)-面向对象之类的成员

    title: Java基础语法(9)-面向对象之类的成员 blog: CSDN data: Java学习路线及视频 1.面向对象特征--封装 为什么需要封装?封装的作用和含义? 我要用洗衣机,只需要按 ...

  2. SublimeのJedi (自动补全)

    关于 Sublime 3 - Jedi Package 的设置和使用方法 我是一枚小白,安装后 Sublime 后,想在码字时,达到如下效果: 打字时,自动提示相关内容 按Tab键,相关内容自动填充 ...

  3. XCode8 安装模拟器

    1.下载模拟器 参考:http://blog.csdn.net/piratest/article/details/52538978 参考:http://blog.csdn.net/zhangao008 ...

  4. 树莓派 Raspberry PI基础

    树莓派 Raspberry PI基础 官网网址:https://www.raspberrypi.org 下载地址:https://www.raspberrypi.org/downloads/ 官方系统 ...

  5. 磅日波浪分析4H 20190927

    磅日的短线调整预计在132结束 目前已显现出ABC调整末端. 未来预计开启第五浪升势 破前期反弹高点.

  6. python opencv 图片缺陷检测(讲解直方图以及相关系数对比法)

    一.利用直方图的方式进行批量的图片缺陷检测(方法简单) 二.步骤(完整代码见最后) 2.1灰度转换(将原图和要检测对比的图分开灰度化) 灰度化的作用是因为后面的直方图比较需要以像素256为基准进行相关 ...

  7. vue配置环境踩坑

    Vue 环境配置踩坑 目录 Vue 环境配置踩坑 windows下cnpm -v :无法将"cnpm"项识别为 cmdlet.函数.脚本文件或可运行程序的名称. windows下c ...

  8. [ASP.NET Core MVC] 如何实现运行时动态定义Controller类型?

    昨天有个朋友在微信上问我一个问题:他希望通过动态脚本的形式实现对ASP.NET Core MVC应用的扩展,比如在程序运行过程中上传一段C#脚本将其中定义的Controller类型注册到应用中,问我是 ...

  9. Python数据库MySQL之数据备份、pymysql模块

    一 IDE工具介绍 生产环境还是推荐使用mysql命令行,但为了方便我们测试,可以使用IDE工具 下载链接:https://pan.baidu.com/s/1bpo5mqj 掌握: #1. 测试+链接 ...

  10. 认识STM32芯片

    STM32中的ST指的是意法半导体,M是Microelectronics的缩写,32表示32位,即意法半导体公司开发的32位微控制器 ST官网:https://www.st.com/content/s ...