Codeforces Round #835 (Div. 4) A-G
A
题意
给出三个不同的数,求中位数。
题解
知识点:模拟。
显然。
时间复杂度 \(O(1)\)
空间复杂度 \(O(1)\)
代码
#include <bits/stdc++.h>
#define ll long long
using namespace std;
bool solve() {
vector<int> v(3 + 1);
for (int i = 1;i <= 3;i++) cin >> v[i];
sort(v.begin() + 1, v.end());
cout << v[2] << '\n';
return true;
}
int main() {
std::ios::sync_with_stdio(0), cin.tie(0), cout.tie(0);
int t = 1;
cin >> t;
while (t--) {
if (!solve()) cout << -1 << '\n';
}
return 0;
}
B
题意
给出一个小写字母字符串,求最少需要字母表前缀多少个字母才能覆盖所有出现的字母。
题解
知识点:模拟。
找到字符串里最大的字母即可。
时间复杂度 \(O(n)\)
空间复杂度 \(O(n)\)
代码
#include <bits/stdc++.h>
#define ll long long
using namespace std;
bool solve() {
int n;
cin >> n;
string s;
cin >> s;
int mx = 0;
for (char ch : s) {
mx = max(mx, ch - 'a' + 1);
}
cout << mx << '\n';
return true;
}
int main() {
std::ios::sync_with_stdio(0), cin.tie(0), cout.tie(0);
int t = 1;
cin >> t;
while (t--) {
if (!solve()) cout << -1 << '\n';
}
return 0;
}
C
题意
给出一组数,求出每个数减去这组数中一个数后能得到的最小值。
题解
知识点:模拟。
显然,每个数减去最大的数即可得到最小值。
但注意最大的数可能只有一个,但自己不能减自己,因此我们还需要求出次大值,让所有等于最大值的数减去次大值即可。
时间复杂度 \(O(n)\)
空间复杂度 \(O(n)\)
代码
#include <bits/stdc++.h>
#define ll long long
using namespace std;
int a[200007];
bool solve() {
int n;
cin >> n;
for (int i = 1;i <= n;i++) cin >> a[i];
int mx1 = 0, mx2 = 0;
for (int i = 1;i <= n;i++) {
if (a[i] > mx1) mx2 = mx1, mx1 = a[i];
else if (a[i] > mx2) mx2 = a[i];
}
for (int i = 1;i <= n;i++) {
if (a[i] < mx1) cout << a[i] - mx1 << ' ';
else cout << a[i] - mx2 << ' ';
}
cout << '\n';
return true;
}
int main() {
std::ios::sync_with_stdio(0), cin.tie(0), cout.tie(0);
int t = 1;
cin >> t;
while (t--) {
if (!solve()) cout << -1 << '\n';
}
return 0;
}
D
题意
给出一组数,问其中是否有且只有一个山谷。
山谷的定义是,数组中的一个连续子串 \(a_{l\cdots r}\) ,满足以下条件:
- \(0≤l≤r≤n−1\)
- \(a_l=a_{l+1}=a_{l+2}=\cdots =a_r\)
- \(l=0\) 或 \(a_{l−1} > al\)
- \(r=n−1\) 或 \(a_r<a_{r+1}\)
题解
知识点:模拟。
显然第一个山谷一定会出现在第一次严格递增前,如果没有则出现在右端点。
因此,这组数一旦有一次严格递增,之后就不能严格递减(可以相等),因为再次严格递减后必然会再出现一个山谷。
时间复杂度 \(O(n)\)
空间复杂度 \(O(n)\)
代码
#include <bits/stdc++.h>
#define ll long long
using namespace std;
int a[200007];
bool solve() {
int n;
cin >> n;
for (int i = 1;i <= n;i++) cin >> a[i];
bool dz = 0;
for (int i = 2;i <= n;i++) {
if (a[i] < a[i - 1] && dz) return false;
dz |= a[i] > a[i - 1];
}
cout << "YES" << '\n';
return true;
}
int main() {
std::ios::sync_with_stdio(0), cin.tie(0), cout.tie(0);
int t = 1;
cin >> t;
while (t--) {
if (!solve()) cout << "NO" << '\n';
}
return 0;
}
E
题意
给定一个 01
串,可以选择一位翻转一次,求最大逆序数。
题解
知识点:枚举,前缀和,贪心。
我们只需要知道每位修改后变化量,取最大值即可。
变化量可以通过维护前缀后缀 1
的个数和 \(l1,r1\) 获得。在第 \(i\) 位, 1->0
,则变化量为 \(l1_{i-1} - (n-i -r1_{i+1})\) ;0->1
,则变化量为 \((n-i -r1_{i+1}) - l1_{i-1}\) 。
时间复杂度 \(O(n)\)
空间复杂度 \(O(n)\)
代码
#include <bits/stdc++.h>
#define ll long long
using namespace std;
bool a[200007];
int l1[200007], r1[200007];
bool solve() {
int n;
cin >> n;
for (int i = 1;i <= n;i++) cin >> a[i];
l1[0] = r1[n + 1] = 0;
for (int i = 1;i <= n;i++) l1[i] = l1[i - 1] + a[i];
for (int i = n;i >= 1;i--) r1[i] = r1[i + 1] + a[i];
int delta = 0;
ll sum = 0;
for (int i = 1;i <= n;i++) {
if (a[i]) delta = max(delta, l1[i - 1] - (n - i - r1[i + 1])), sum += (n - i - r1[i + 1]);
else delta = max(delta, (n - i - r1[i + 1]) - l1[i - 1]);
}
cout << sum + delta << '\n';
return true;
}
int main() {
std::ios::sync_with_stdio(0), cin.tie(0), cout.tie(0);
int t = 1;
cin >> t;
while (t--) {
if (!solve()) cout << -1 << '\n';
}
return 0;
}
F
题意
给定 \(n\) 个任务,任务 \(i\) 完成后能得到金币 \(a_i\) 个。
每天能选择一个任务完成并得到其金币,任务可以重复完成,但每个任务完成后这个任务会冷却 \(k\) 天,之后才能再次选择。
给定需求金币数 \(c\) ,以及总天数 \(d\) ,问在 \(d\) 天内获得 \(c\) 个金币的前提下, \(k\) 最大是多少。
题解
知识点:二分,贪心。
显然用二分检验求解。
令 \(k \in [0,d]\) ,\(k<0\) 则无解,\(k\geq d\) 则每个任务只需要完成一次就能满足,则 \(k\) 可以无穷大。
对于一个需要检验的 \(k\) ,显然我们优先选金币数大的任务。又因为一个任务完成后 \(k\) 天又可以再完成,因此我们只需要选前 \(\min(k+1,n)\) 个任务重复完成即可,能完整完成 \(\lfloor \frac{d}{k+1} \rfloor\) 轮,并再多完成 \(\min(n,d - (k+1) \cdot \lfloor \frac{d}{k+1} \rfloor)\) 个任务。总额大于等于 \(c\) ,这个答案就合法。
时间复杂度 \(O(n \log n + \log d)\)
空间复杂度 \(O(n)\)
代码
#include <bits/stdc++.h>
#define ll long long
using namespace std;
int n;
ll c;
int d;
ll a[200007];
bool check(int mid) {
int cyc = d / (mid + 1);
int rest = d % (mid + 1);
return a[min(mid + 1, n)] * cyc + a[min(rest, n)] >= c;
}
bool solve() {
cin >> n >> c >> d;
for (int i = 1;i <= n;i++) cin >> a[i];
sort(a + 1, a + n + 1, [&](int a, int b) {return a > b;});
for (int i = 1;i <= n;i++) a[i] += a[i - 1];
int l = 0, r = d;
while (l <= r) {
int mid = l + r >> 1;
if (check(mid)) l = mid + 1;
else r = mid - 1;
}
if (r < 0) cout << "Impossible" << '\n';
else if (r >= d) cout << "Infinity" << '\n';
else cout << r << '\n';
return true;
}
int main() {
std::ios::sync_with_stdio(0), cin.tie(0), cout.tie(0);
int t = 1;
cin >> t;
while (t--) {
if (!solve()) cout << -1 << '\n';
}
return 0;
}
G
题意
给一个有 \(n\) 个节点的带权树。
从节点 \(a\) 出发到节点 \(b\) ,开始时 \(x = 0\) ,经过一条边 \(i\) 时,则异或边权 \(w_i\) ,即 \(x = x \oplus w_i\) 。当且仅当进入 \(b\) 时恰好 \(x = 0\) 才能进入 \(b\) ,否则不能进入 \(b\) 。有一次传送机会,可以从任意点出发,传送到除了 \(b\) 的任意点。
问是否存在方案从 \(a\) 走到 \(b\) 。
题解
知识点:dfs,枚举,位运算。
异或有个性质,\(a \oplus a = 0\) 。
结合我们可以任意传送考虑,我们发现只需要两个点 \(i,j\) (除了 \(b\) )满足 \(a\) 到 \(i\) 的异或和等于 \(j\) 到 \(b\) 的异或和,那我们就可以从 \(i\) 传送到 \(j\) 即可存在 \(a\) 到 \(b\) 的方案。
因此,先从 \(a\) 遍历每个点的异或和。注意,\(a\) 出发的路径不能跨越 \(b\) ,因为此时 \(b\) 是走不通的。当然有可能到 \(b\) 直接就能进去了,我们可以把这个归类为走到 \(b\) 前的一个点原地传送以后,再从这个点出发进入 \(b\) 。
随后,我们把 \(a\) 能到达的点的异或和放进 set
里,再从 \(b\) 出发遍历每个点的异或和,因为异或具有交换律,倒着走和正着走结果一样。
最后,在 set
中查找 \(b\) 出发每个点得到的异或和是否已经存在即可。
时间复杂度 \(O(n \log n)\)
空间复杂度 \(O(n)\)
代码
#include <bits/stdc++.h>
#define ll long long
using namespace std;
template<class T>
struct Graph {
struct edge {
int v, nxt;
T w;
};
int idx;
vector<int> h;
vector<edge> e;
Graph(int n, int m) :idx(0), h(n + 1), e(m + 1) {}
void init(int n) {
idx = 0;
h.assign(n + 1, 0);
}
void add(int u, int v, T w) {
e[++idx] = edge{ v,h[u],w };
h[u] = idx;
}
};
const int N = 100007, M = 100007 << 1;
int n, a, b;
Graph<int> g(N, M);
int dis[N];
void dfs(int u, int fa) {
for (int i = g.h[u];i;i = g.e[i].nxt) {
int v = g.e[i].v, w = g.e[i].w;
if (v == fa || v == b) continue;
dis[v] = dis[u] ^ w;
dfs(v, u);
}
}
bool solve() {
cin >> n >> a >> b;
g.init(n);
for (int i = 1;i <= n - 1;i++) {
int u, v, w;
cin >> u >> v >> w;
g.add(u, v, w);
g.add(v, u, w);
}
for (int i = 1;i <= n;i++) dis[i] = -1;
dis[a] = 0;
dfs(a, 0);
set<int> st;
for (int i = 1;i <= n;i++) if (dis[i] != -1)st.insert(dis[i]);
dis[b] = 0;
dfs(b, 0);
for (int i = 1;i <= n;i++) {
if (i != b && st.count(dis[i])) {
cout << "YES" << '\n';
return true;
}
}
cout << "NO" << '\n';
return true;
}
int main() {
std::ios::sync_with_stdio(0), cin.tie(0), cout.tie(0);
int t = 1;
cin >> t;
while (t--) {
if (!solve()) cout << -1 << '\n';
}
return 0;
}
Codeforces Round #835 (Div. 4) A-G的更多相关文章
- Educational Codeforces Round 47 (Div 2) (A~G)
目录 Codeforces 1009 A.Game Shopping B.Minimum Ternary String C.Annoying Present D.Relatively Prime Gr ...
- Educational Codeforces Round 46 (Div 2) (A~G)
目录 Codeforces 1000 A.Codehorses T-shirts B.Light It Up C.Covered Points Count(差分) D.Yet Another Prob ...
- Educational Codeforces Round 45 (Div 2) (A~G)
目录 Codeforces 990 A.Commentary Boxes B.Micro-World C.Bracket Sequences Concatenation Problem D.Graph ...
- Codeforces Round #582 (Div. 3)-G. Path Queries-并查集
Codeforces Round #582 (Div. 3)-G. Path Queries-并查集 [Problem Description] 给你一棵树,求有多少条简单路径\((u,v)\),满足 ...
- Codeforces Round #368 (Div. 2)
直达–>Codeforces Round #368 (Div. 2) A Brain’s Photos 给你一个NxM的矩阵,一个字母代表一种颜色,如果有”C”,”M”,”Y”三种中任意一种就输 ...
- Codeforces Round #279 (Div. 2) ABCDE
Codeforces Round #279 (Div. 2) 做得我都变绿了! Problems # Name A Team Olympiad standard input/outpu ...
- 贪心+模拟 Codeforces Round #288 (Div. 2) C. Anya and Ghosts
题目传送门 /* 贪心 + 模拟:首先,如果蜡烛的燃烧时间小于最少需要点燃的蜡烛数一定是-1(蜡烛是1秒点一支), num[g[i]]记录每个鬼访问时已点燃的蜡烛数,若不够,tmp为还需要的蜡烛数, ...
- Codeforces Round #383 (Div. 2) 题解【ABCDE】
Codeforces Round #383 (Div. 2) A. Arpa's hard exam and Mehrdad's naive cheat 题意 求1378^n mod 10 题解 直接 ...
- 模拟 Codeforces Round #249 (Div. 2) C. Cardiogram
题目地址:http://codeforces.com/contest/435/problem/C /* 题意:给一组公式,一组数据,计算得到一系列的坐标点,画出折线图:) 模拟题:蛮恶心的,不过也简单 ...
- Codeforces Round #368 (Div. 2) B. Bakery (模拟)
Bakery 题目链接: http://codeforces.com/contest/707/problem/B Description Masha wants to open her own bak ...
随机推荐
- 使用J2EE 登录实例开发
我们先了解下Servlet的生命周期 Servlet部署在容器里,其生命周期由容器管理. 概括为以下几个阶段: 1)容器加载Servlet类. 当第一次有Web客户请求Servlet服务或当Web服务 ...
- 我眼中的大数据(二)——HDFS
Hadoop的第一个产品是HDFS,可以说分布式文件存储是分布式计算的基础,也可见分布式文件存储的重要性.如果我们将大数据计算比作烹饪,那么数据就是食材,而Hadoop分布式文件系统HDFS就是烧菜的 ...
- 使用Docker方式部署Gitlab,Gitlab-Runner并使用Gitlab提供的CI/CD功能自动化构建SpringBoot项目
1.Docker安装Gitlab,地址:https://www.cnblogs.com/sanduzxcvbnm/p/13814730.html 2.Docker安装Gitlab-runner,地址: ...
- MinIO多租户(Multi-tenant)部署指南
官方文档地址:http://docs.minio.org.cn/docs/master/multi-tenant-minio-deployment-guide 单机部署 在单台机器上托管多个租户,为每 ...
- [题解] Codeforces 1268 D Invertation in Tournament 结论,兰道定理
题目 本题需要用到的结论: 一.兰道定理 二.如果\(n\geq4\),那么\(n\)个点的强连通竞赛图存在\(n-1\)个点的强连通子图. 证明: 现在有一个n-1个点的竞赛图(不一定强连通,称其为 ...
- Vue-amap的使用
(1)Npm安装:npm install vue-amap –save (2)在main.js中配置 首先需要在项目初始化时,通过 initAMapApiLoader 引入所需要的插件: (3)vue ...
- 追求性能极致:Redis6.0的多线程模型
Redis系列1:深刻理解高性能Redis的本质 Redis系列2:数据持久化提高可用性 Redis系列3:高可用之主从架构 Redis系列4:高可用之Sentinel(哨兵模式) Redis系列5: ...
- OnionArch - 如何实现更新指定字段的通用Handler
博主最近失业在家,找工作之余,自己动手写了个洋葱架构(整洁架构)解决方案,以总结和整理以前的项目经验,起名叫OnionArch,其目的是为了更好的实现采用DDD(领域驱动分析)和命令查询职责分离(CQ ...
- 细聊.Net Core中IServiceScope的工作方式
前言 自从.Net Core引入IOC相关的体系之后,关于它的讨论就从来没有停止过,因为它是.Net Core体系的底层框架,你只要使用了.Net Core的时候就必然会用到它.当然关于使用它的过程中 ...
- Dubbo 02: 直连式
直连式 需要用到两个相互独立的maven的web项目 项目1:o1-link-userservice-provider 作为服务的提供者 项目2:o2-link-consumer 作为使用服务的消费者 ...