题目链接

题目

见链接。

题解

方法一

知识点:贪心,优先队列,二分。

显然,这道题可以用二分答案做。check 函数可以用小根堆,让维持时间最小的先充电。

但是不优化这道题会炸。有两个关键优化:一个是快读快写能省不少时间,还有一个是把维持天数当一个变量存起来以免重复运算浪费时间。其他一些小优化:用 pop 把元素弹出代替析构函数自己初始化能省一点时间,只让天数小于 \(k\) 的电脑入队,每次充完电检测维持天数小于 \(k\) 的才重新入队。优化前是超 \(3\) 秒限制的,优化后是 \(1.5\) 秒还算可以。

时间复杂度 \(O((n+k)\log n)\) ,常数应该在 \((100,1000)\)

空间复杂度 \(O(n)\)

方法二

知识点:贪心,二分。

方法一的检验并非正解,其实有一个更妙的方法去验证电脑是否会在 \(k\) 天之前关机。

我们用一个数组 \(cnt[i]\) 表示有多少电脑最晚第 \(i\) 天前要充一次电。比如一台电脑是初始电量是 \(20\) 每天耗电 \(15\) 要维持到 \(8\) 天,每次充电 \(40\) ,那么它最晚在第 \(2\) 、\(5\)、\(7\) 天要充一次电,于是 \(cnt[\{2,5,7\}]\) 都要加一。

我们不关心电脑在哪天充电,我们只关心电脑最晚要在什么时候前充电,所以 \(cnt[i]\) 在某些天超过 \(1\) 是可行的。因为既然我们知道在这天之后有三台电脑会关机,只要在这天之前什么时候充电都行,不过要满足之前有空闲的天数。

于是,现在我们把它从 \(1\) 到 \(i\) 累和,得到一个结果 \(sum\) ,表示到 \(i\) 天要至少要充几次电,显然每天只能充一次,那么如果 \(sum > i\) ,则存在电脑没在最晚时间前充上电,关机了,是这个答案是不可行的。如果 \(sum \leq i\) ,说明到第 \(i\) 天充电次数完全够用,可以继续。

时间复杂度 \(O(n+k)\) ,常数在 \((50,200)\)

空间复杂度 \(O(n+k)\)

代码

方法一

#include <bits/stdc++.h>
#define ll long long using namespace std; inline ll read() {
ll x = 0, f = 1;
char c = getchar();
while (c < '0' || c>'9') {
if (c == '-') f = -1;
c = getchar();
}///整数符号
while (c >= '0' && c <= '9') {
x = (x << 3) + (x << 1) + (c ^ 48);
c = getchar();
}///挪位加数
return x * f;
}///关键优化,快读 struct node {
ll a, b, v;///关键优化,存储天数
bool operator>(const node &x) const {///大根堆重载小于,小根堆重载大于,true代表优先级小,必须是常函数或者友元函数
return v > x.v;
}
}a[200007];
int n, k; // struct cmp {
// bool operator()(const node &a, const node &b) {
// return a.a / a.b > b.a / b.b;
// }
// } ///也可以写个比较类 priority_queue<node, vector<node>, greater<node>> pq;
//priority_queue<node, vector<node>, cmp> pq; bool check(ll mid) {
while (!pq.empty()) pq.pop();///优化
for (int i = 0;i < n;i++) if (a[i].v < k)pq.push(a[i]); ///优化
for (int i = 1;i <= k;i++) {
if (pq.empty()) return true;
node x = pq.top();
pq.pop();
if (x.v < i) return false;///加之前判断是否能撑到这个时候
x.a += mid;
x.v = x.a / x.b + 1;
if (x.v < k) pq.push(x);///优化
}
return true;
} int main() {
std::ios::sync_with_stdio(0), cin.tie(0), cout.tie(0);
n = read();
k = read();
for (int i = 0;i < n;i++) a[i].a = read();
for (int i = 0;i < n;i++) a[i].b = read(), a[i].v = a[i].a / a[i].b + 1;///因为一天结束才扣电,而到了这天就算,所以取下整能过几天,加一是往后一天也算到了。
ll l = 0, r = 2e12;
while (l <= r) {
ll mid = l + r >> 1;
if (check(mid)) r = mid - 1;
else l = mid + 1;
}
cout << (l > 2e12 ? -1 : l) << '\n';
return 0;
}

方法二

#include <bits/stdc++.h>
#define ll long long using namespace std; int n, k;
ll a[200007], b[200007]; bool check(ll mid) {
int r = k;
vector<int> sum(k);
for (int i = 0;i < n;i++) {
ll tmp = a[i];
while (tmp / b[i] + 1 < k && r >= 0) {
sum[tmp / b[i] + 1]++;
tmp += mid;
r--;
}
if (r < 0) return false;
}
for (int i = 1;i < k;i++) {
sum[i] += sum[i - 1];
if (sum[i] > i) return false;///充电次数超过天数,不可能实现
}
return true;
} int main() {
std::ios::sync_with_stdio(0), cin.tie(0), cout.tie(0);
cin >> n >> k;
for (int i = 0;i < n;i++) cin >> a[i];
for (int i = 0;i < n;i++) cin >> b[i];
ll l = 0, r = 2e12;
while (l <= r) {
ll mid = l + r >> 1;
if (check(mid)) r = mid - 1;
else l = mid + 1;
}
cout << (l > 2e12 ? -1 : l) << '\n';
return 0;
}

CF1132D Stressful Training的更多相关文章

  1. CF 3-6 2级组 D题 STRESSFUL TRAINING 紧张的比赛

    题目大概是这样的: 给出一个数列a[n] ,对于每一个数 a [i] 来说 都会在 T - - 时 -= b[i] 每个数都在任何时刻不能小于0 你可以在每次T - - 之前时给 一 个 a[i] + ...

  2. 【Codeforces 1132D】Stressful Training

    Codeforces 1132 D 题意:给\(n\)个电脑的电量和耗电速度,你可以买一个充电器,它的充电速度是每秒\(v\)单位,\(v\)你自己定.问最小的\(v\)能使得在\(k\)秒内每秒给某 ...

  3. Codeforces 1132D - Stressful Training - [二分+贪心+优先队列]

    题目链接:https://codeforces.com/contest/1132/problem/D 题意: 有 $n$ 个学生,他们的电脑有初始电量 $a[1 \sim n]$,他们的电脑每分钟会耗 ...

  4. CF集萃1

    因为cf上一堆水题,每个单独开一篇博客感觉不太好,就直接放一起好了. CF1096D Easy Problem 给定字符串,每个位置删除要代价.求最小代价使之不含子序列"hard" ...

  5. Codeforces 1132 - A/B/C/D/E/F - (Undone)

    链接:http://codeforces.com/contest/1132 A - Regular Bracket Sequence - [水] 题解:首先 "()" 这个的数量多 ...

  6. CF1132.Educational Codeforces Round 61(简单题解)

    A .Regular Bracket Sequence 题意:给定“((” , “()” ,  “)(”,  “))”四种,问是否可以组成合法括号匹配 思路:设四种是ABCD,B可以不用管,而C在A或 ...

  7. Educational Codeforces Round 61 (Rated for Div. 2) D,F题解

    D. Stressful Training 题目链接:https://codeforces.com/contest/1132/problem/D 题意: 有n台电脑,每台电脑都有初始电量ai,也有一个 ...

  8. CF 1132A,1132B,1132C,1132D,1132E,1132F(Round 61 A,B,C,D,E,F)题解

    A.Regular bracket sequence A string is called bracket sequence if it does not contain any characters ...

  9. hdu 4946 2014 Multi-University Training Contest 8

    Area of Mushroom Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others) ...

随机推荐

  1. 超越iTerm! 号称下一代终端神器,功能贼强大!

    程序员的一生,用的最多的两个工具,一个是代码编辑器(Code Editor),另外一个就是命令行终端工具(Terminal).这两个工具对于提高开发效率至关重要. 代码编辑器在过去的 40 年里不断进 ...

  2. C#/VB.NET 获取Excel中图片所在的行、列坐标位置

    本文以C#和vb.net代码示例展示如何来获取Excel工作表中图片的坐标位置.这里的坐标位置是指图片左上角顶点所在的单元格行和列位置,横坐标即顶点所在的第几列.纵坐标即顶点所在的第几行.下面是获取图 ...

  3. css的过渡transition和动画animation

    过渡 过渡(transition)是CSS3中具有颠覆性的特性之一,我们可以在不使用Flash动画或JavaScript的情况下,当元素从一种样式变换为另一种样式时元素添加效果.过渡动画:是从一个状态 ...

  4. 聊聊如何在华为云IoT平台进行产品开发

    摘要:华为云物联网平台承载着南北向数据互通的功能职责. 本文分享自华为云社区<如何基于华为云IoT物联网平台进行产品开发>,作者: Super.雯 . 华为云物联网平台承载着南北向数据互通 ...

  5. 命令工具 -(1)Vim 文本编辑器学习

    关注「开源Linux」,选择"设为星标" 回复「学习」,有我为您特别筛选的学习资料~ 前言 提起 Linux,大家都听说过这句话:Linux 一切皆文件. 配置一个服务就是在修改它 ...

  6. 聊聊FLINK-25631贡献

    从入行做数据库开发,到2018年过渡到大数据开发,可以说我已经与sql朝夕相处了七八年了,经常惊讶于简单的语法就能产生复杂的操作,而且还能根据索引等统计信息自动优化,不禁很想实现自己的sql语法,却不 ...

  7. 吊炸天,Spring Security还有这种用法!

    在用Spring Security项目开发中,有时候需要放通某一个接口时,我们需要在配置中把接口地址配置上,这样做有时候显得麻烦,而且不够优雅.我们能不能通过一个注解的方式,在需要放通的接口上加上该注 ...

  8. 文件操作(Java)

    学习内容:文件操作        1.输入流:InputStream类是字节输入流的抽象类,常用的一些方法有: raed()方法:从输入流中读取数据的下一个字节 reset()方法:将输入指针返回到当 ...

  9. 个人冲刺(一)——体温上报app(一阶段)

    任务:完成了体温上报app的整体页面布局 activity_main.xml <?xml version="1.0" encoding="utf-8"?& ...

  10. MongoDB 常用启动参数

    每日一句 Once you choose your way of life, be brave to stick it out and never return. 生活的道路一旦选定,就要勇敢地走到底 ...