热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

AtCoderGrandContest012

题目传送门:AtCoderGrandContest012。目录A-AtCoderGroupContestB-SplatterPaintingC-TautonymPuzzleD-Co

题目传送门:AtCoder Grand Contest 012。

目录

  • A - AtCoder Group Contest
  • B - Splatter Painting
  • C - Tautonym Puzzle
  • D - Colorful Balls
  • E - Camel and Oases
  • F - Prefix Median


A - AtCoder Group Contest

从大到小排序后选择 \(a_2, a_4, \ldots, a_{2 N}\)。

#include
#include
typedef long long LL;
const int MN = 300005;
int N, A[MN];
LL Ans;
int main() {
scanf("%d", &N), N *= 3;
for (int i = 1; i <= N; ++i) scanf("%d", &A[i]);
std::sort(A + 1, A + N + 1);
for (int i = N / 3 + 1; i <= N; i += 2) Ans += A[i];
printf("%lld\n", Ans);
return 0;
}

B - Splatter Painting

注意到 \(d\) 的范围只有 \(10\),考虑到一个 \(d = d_0\) 的操作等价于它以及它周围所有点的 \(d = d_0 - 1\) 的操作。

从大到小处理 \(d\),每次把 \(d_0\) 通过一个 \(\mathcal O (N + M)\) 的操作推向 \(d_0 - 1\) 然后再加入 \(d = d_0 - 1\) 的操作。

时间复杂度为 \(\mathcal O ((N + M + Q) \max d_i)\)。

#include
#include
#include
const int MN = 100005, MQ = 100005;
int N, M;
std::vector G[MN];
int Q, qv[MQ], qd[MQ], qc[MQ];
int tag[MN], tmp[MN];
int main() {
scanf("%d%d", &N, &M);
for (int i = 1, x, y; i <= M; ++i) {
scanf("%d%d", &x, &y);
G[x].push_back(y);
G[y].push_back(x);
}
scanf("%d", &Q);
for (int i = 1; i <= Q; ++i) scanf("%d%d%d", &qv[i], &qd[i], &qc[i]);
for (int j = 10; j >= 0; --j) {
for (int u = 1; u <= N; ++u) tmp[u] = 0;
for (int u = 1; u <= N; ++u) if (tag[u])
for (int v : G[u]) tmp[v] = std::max(tmp[v], tag[u]);
for (int u = 1; u <= N; ++u) tag[u] = std::max(tag[u], tmp[u]);
for (int i = 1; i <= Q; ++i) if (qd[i] == j) tag[qv[i]] = std::max(tag[qv[i]], i);
}
for (int u = 1; u <= N; ++u) printf("%d\n", qc[tag[u]]);
return 0;
}

C - Tautonym Puzzle

考虑 \(1 \sim k\) 的排列 \(p_1, p_2, \ldots , p_k\),若 \(s\) 序列为 \([p_1, p_2, \ldots , p_k, 1, 2, \ldots , k]\),则好子序列数量恰好为 \(p\) 中的非空递增子序列数量。

考虑一个 \(1 \sim (k - 1)\) 的排列 \(q\),令其递增子序列(可以为空)的数量为 \(c\),则:



  • 令 \(p = [k] + q\),则 \(p\) 的递增子序列(可以为空)的数量应为 \(c + 1\);

  • 令 \(p = q + [k]\),则 \(p\) 的递增子序列(可以为空)的数量应为 \(c \times 2\)。

特别地,空序列的递增子序列(可以为空)的数量为 \(1\)。

令 \(N\) 加上 \(1\),我们要求即是一个排列 \(p\) 满足递增子序列(可以为空)的数量恰好为 \(N\)。

根据上面的结论,不难想到一个按照 \(N\) 的二进制位从高到低进行递归构造的做法,\(s\) 的长度不超过 \(4 \log_2 n \approx 160\)。

#include
#include
typedef long long LL;
std::vector Solve(LL N) {
if (N == 1) return std::vector();
if (N & 1) {
auto v = Solve(N - 1);
v.insert(v.begin(), (int)v.size() + 1);
return v;
}
auto v = Solve(N / 2);
v.push_back((int)v.size() + 1);
return v;
}
int main() {
LL N;
scanf("%lld", &N);
auto Ans = Solve(N + 1);
printf("%d\n", 2 * (int)Ans.size());
for (int x : Ans) printf("%d ", x);
for (int i = 1; i <= (int)Ans.size(); ++i) printf("%d ", i);
puts("");
return 0;
}

D - Colorful Balls

我们考虑把每个球初始在的位置的下标写在这个球上,那么交换两个球的位置就可以看成交换两个球上的数。

如果两个球可以交换,那么我们在它们之间连边。显然我们只需对每个连通分量分别考虑即可。

对于一个连通分量,考虑它的任意一棵生成树,不难发现仅通过树边上的交换操作就可以到达任意排列所对应的状态。

也就是说对于一个连通分量对应回原序列的下标位置,其任意一个颜色排列均可以被达到。

所以方案数是一个每个颜色的出现次数构成的多重组合数。

朴素的实现的边数是 \(\mathcal O (N^2)\) 的,接下来的问题在于减少边数,使得连通分量的性质仍然保留。

对于每种颜色,按照重量从小到大对这种颜色的所有球排序,记颜色为 \(c\) 的所有球中重量最小的为 \(\min[c]\)。

那么对于同种颜色球之间的连边 \(u \leftrightarrow v\),如果没有经过 \(\min[c]\),则转化为 \(u \leftrightarrow \min[c] \leftrightarrow v\) 绝不会更劣。

也就是对于每种颜色 \(c\),每个非 \(\min[c]\) 的球尝试向 \(\min[c]\) 连边即可。

对于非同种颜色的球之间的连边,令 \(k\) 为 \(\min[c]\) 的重量最小的颜色 \(c\):

则如果两非颜色 \(k\) 的球 \(u, v\) 之间连边,转化为 \(u \leftrightarrow \min[k] \leftrightarrow v\) 绝不会更劣。

而如果 \(u, v\) 之一的颜色为 \(k\),令 \(k'\) 为 \(\min[c]\) 的重量次小的颜色 \(c\):



  • 如果 \(u, v\) 的颜色均不为 \(k'\),转化为 \(u \leftrightarrow \min[k'] \leftrightarrow v\) 绝不会更劣。

  • 如果 \(u, v\) 中其一颜色为 \(k\),另一颜色为 \(k'\),不妨设 \(u\) 的颜色为 \(k\),转化为 \(u \leftrightarrow \min[k'] \leftrightarrow \min[k] \leftrightarrow v\) 绝不会更劣。

于是,我们仅需处理出 \(k\) 和 \(k'\),然后将每个与 \(k\) 或 \(k'\) 不同颜色的球向 \(\min[k]\) 或 \(\min[k']\) 连边即可。

总边数控制在 \(\mathcal O (N)\),然后运行 DFS 找出这张图的每个连通分量,计算多重组合数贡献给答案即可。

#include
#include
#include
typedef long long LL;
const int Mod = 1000000007;
const int MN = 200005;
inline int qPow(int b, int e) {
int a = 1;
for (; e; e >>= 1, b = (LL)b * b % Mod)
if (e & 1) a = (LL)a * b % Mod;
return a;
}
int Fac[MN], iFac[MN];
inline void Init(int N) {
Fac[0] = 1;
for (int i = 1; i <= N; ++i) Fac[i] = (LL)Fac[i - 1] * i % Mod;
iFac[N] = qPow(Fac[N], Mod - 2);
for (int i = N; i >= 1; --i) iFac[i - 1] = (LL)iFac[i] * i % Mod;
}
int N, X, Y, p1, p2, C;
std::vector V[MN], G[MN];
int index[MN], col[MN];
inline void Ins(int x, int y) {
G[x].push_back(y);
G[y].push_back(x);
}
int vis[MN], buk[MN], stk[MN], tp;
void DFS(int u) {
vis[u] = 1, ++buk[col[u]], stk[++tp] = u;
for (int v : G[u]) if (!vis[v]) DFS(v);
}
int main() {
scanf("%d%d%d", &N, &X, &Y);
Init(N);
for (int i = 1, x, y; i <= N; ++i) {
scanf("%d%d", &x, &y);
V[x].push_back(y);
}
for (int i = 1; i <= N; ++i) if (!V[i].empty()) {
std::sort(V[i].begin(), V[i].end());
if (!p1) p1 = i;
else {
if (!p2 || V[p2][0] > V[i][0]) p2 = i;
if (V[p1][0] > V[p2][0]) std::swap(p1, p2);
}
int s = V[i].size();
index[i] = C += s;
for (int j = 0; j }
for (int i = 1; i <= N; ++i) if (!V[i].empty()) {
int s = V[i].size(), c = index[i];
for (int j = 1; j if (p1 && i != p1)
for (int j = 0; j if (p2 && i == p1)
for (int j = 1; j }
int Ans = 1;
for (int i = 1; i <= N; ++i) if (!vis[i]) {
DFS(i);
int s = 0;
for (int x = 0; tp; --tp) if (buk[x = col[stk[tp]]])
s += buk[x], Ans = (LL)Ans * iFac[buk[x]] % Mod, buk[x] = 0;
Ans = (LL)Ans * Fac[s] % Mod;
}
printf("%d\n", Ans);
return 0;
}

E - Camel and Oases

令 \(k = \lfloor \log_2 V \rfloor + 1\),则骆驼可以跳跃 \(k\) 次。而骆驼的储水量,包括初始的那次,依次为 \(v_0 \sim v_k\),其中 \(v_j = \left\lfloor V / 2^j \right\rfloor\)。

我们对每个 \(j\),预处理在每个位置向左向右走,跨过长度不超过 \(v_j\) 的段,能走到的区间的左右端点。这部分复杂度为 \(\mathcal O (N \log V)\)。

考虑骆驼能到达到每个位置的条件,应该是能将 \(n\) 个位置划分成 \(k + 1\) 个连续段,使得每一段对应一个互不相同的 \(v_j\)(\(0 \le j \le k\)),必须满足每一段内的位置都能在其对应的 \(v_j\) 的限制下通过走路能互相到达。

而骆驼从一个起点出发能到达每个位置,就是这个起点必须在 \(v_0\) 所在的段内。

我们考虑枚举起点,求出它只通过走路能到达的区间的左右端点 \(\ell, r\),那么该区间的左右两边就可以分别考虑。

枚举在左侧需要的 \(j\) 的集合 \(S \subseteq \{1, 2, \ldots , k\}\),则右侧能够使用的集合就是 \(\{1, 2, \ldots , k\} \setminus S\)。

我们通过状压 DP 可以处理出 \(\mathrm{f}_\ell[S]\) 表示仅通过集合 \(S\) 中的 \(j\),最长可以覆盖到多长的前缀位置。同理定义后缀的 DP 数组 \(\mathrm{f}_r\)。

那么如果 \(\mathrm{f}_\ell[S]\) 的位置在 \(\ell\) 右侧,并且 \(\mathrm{f}_r[\{1, 2, \ldots , k\} \setminus S]\) 的位置在 \(r\) 左侧,这个起点就是能到达每个位置的。

DP 的时间复杂度为 \(\mathcal O (k 2^k) = \mathcal O (V \log V)\)。

最后实现时不能在每个位置上枚举 \(S\),那样复杂度是 \(\mathcal O (N V)\) 的。

而应该先枚举 \(S\) 然后处理出 \(\mathrm{f}_\ell[S]\) 在每个位置右侧时 \(\mathrm{f}_r[\{1, 2, \ldots , k\} \setminus S]\) 最左能到的位置,然后利用这个信息再进行枚举。

总时间复杂度为 \(\mathcal O ((N + V) \log V)\)。

#include
#include
const int MN = 200005, MM = 18;
int N, V, A[MN], v[MM], M;
int L[MM + 1][MN], R[MM + 1][MN];
int fl[1 <int val[MN];
int main() {
scanf("%d%d", &N, &V);
for (int i = 1; i <= N; ++i) scanf("%d", &A[i]);
for (int x = V; x; ) v[M++] = x /= 2;
for (int j = 0; j <= M; ++j) {
int nv = j ? v[j - 1] : V;
L[j][0] = L[j][1] = 1, R[j][N + 1] = R[j][N] = N;
for (int i = 2; i <= N; ++i) L[j][i] = A[i] - A[i - 1] > nv ? i : L[j][i - 1];
for (int i = N - 1; i >= 1; --i) R[j][i] = A[i + 1] - A[i] > nv ? i : R[j][i + 1];
}
fl[0] = 0, fr[0] = N + 1;
for (int S = 1; S <1 < fl[S] = 0, fr[S] = N + 1;
for (int j = 0; j > j & 1)
fl[S] = std::max(fl[S], R[j + 1][fl[S ^ 1 < fr[S] = std::min(fr[S], L[j + 1][fr[S ^ 1 < }
for (int i = 0; i <= N; ++i) val[i] = N + 2;
for (int S = 0; S <1 < val[fl[S]] = std::min(val[fl[S]], fr[((1 < for (int i = N - 1; i >= 0; --i) val[i] = std::min(val[i], val[i + 1]);
for (int i = 1; i <= N; ++i) puts(val[L[0][i] - 1] <= R[0][i] + 1 ? "Possible" : "Impossible");
return 0;
}

F - Prefix Median

将 \(a\) 从小到大排序。我们先考虑 \(a_i\) 互不相同的情况,分析此时 \(b_i\) 需要满足的条件。

首先有 \(a_i \le b_i \le a_{2 N - i}\),这是显然的,极值在全选最小或最大的 \(2 i - 1\) 个元素时取到。

还有一个很关键的性质,因为每次只添加两个新元素,所以 \(b_i\) 相比 \(b_{i - 1}\),在选 \(2 i - 1\) 个数时的集合内,相对位置只会变化最多 \(1\)。

形式化地说就是,对于 \(j b_j > b_{i + 1}\)。

我们可以证明上述两个条件就是充要条件。具体的证明过程是倒序考虑:

每次删除 \(a_i\) 中的两个数,使得 \(b_i\)(\(1 \le i

也就是剩下的 \(b_1 \sim b_{N - 1}\) 还必须满足第一个条件(第二个条件显然会满足)。

具体过程在此不详细展开,主要思路是根据 \(b_{N - 1}\) 和 \(a_N\) 的大小关系分成三类来讨论。

对于 \(a_i\) 可能相同的情况,也是类似的,仍然是 \(a_i \le b_i \le a_{2 N - i}\),且不允许 \(b_i b_j > b_{i + 1}\),注意不取等号。

那么我们考虑按照 \(b_N, b_{N - 1}, \ldots , b_1\) 的顺序倒序依次确定 \(b\) 中的元素。

考虑到第二个条件,如果此时选取的最小和最大元素分别为 \(x, y\),则下一个元素就必须 \(\le x\) 或者 \(\ge y\),绝不能落在 \((x, y)\) 内。

令 \(\mathrm{f}[i][a][b]\) 表示考虑了 \(b_N \sim b_i\) 了,且此时小于 \(b_i\) 的不同元素个数为 \(a\),大于 \(b_i\) 的不同元素个数为 \(b\) 时,\(b_1 \sim b_{i - 1}\) 的方案数。

转移时加入 \(a_{i - 1}\) 和 \(a_{2 N - i + 1}\),并据此更新 \(a, b\) 的值,枚举选取 \((1 + a + b)\) 个数中的哪一个,总时间复杂度 \(\mathcal O (N^4)\)。

答案即为 \(\mathrm{f}[N][0][0]\)。

#include
#include
#include
typedef long long LL;
const int Mod = 1000000007;
const int MN = 55;
int N, A[MN * 2];
int f[MN][MN * 2][MN * 2];
int DP(int i, int a, int b) {
if (i == 1) return 1;
if (~f[i][a][b]) return f[i][a][b];
int na = a + (A[i - 1] != A[i]);
int nb = b + (A[2 * N - i + 1] != A[2 * N - i]);
LL v = DP(i - 1, na, nb);
for (int j = 1; j <= na; ++j) v = (v + DP(i - 1, na - j, nb + 1)) % Mod;
for (int j = 1; j <= nb; ++j) v = (v + DP(i - 1, na + 1, nb - j)) % Mod;
return f[i][a][b] = v % Mod;
}
int main() {
scanf("%d", &N);
for (int i = 1; i <= 2 * N - 1; ++i) scanf("%d", &A[i]);
std::sort(A + 1, A + 2 * N);
memset(f, -1, sizeof(f));
printf("%d\n", DP(N, 0, 0));
return 0;
}


推荐阅读
  • 本文探讨了如何在模运算下高效计算组合数C(n, m),并详细介绍了乘法逆元的应用。通过扩展欧几里得算法求解乘法逆元,从而实现除法取余的计算。 ... [详细]
  • 火星商店问题:线段树分治与持久化Trie树的应用
    本题涉及编号为1至n的火星商店,每个商店有一个永久商品价值v。操作包括每天在指定商店增加一个新商品,以及查询某段时间内某些商店中所有商品(含永久商品)与给定密码值的最大异或结果。通过线段树分治和持久化Trie树来高效解决此问题。 ... [详细]
  • 题目Link题目学习link1题目学习link2题目学习link3%%%受益匪浅!-----&# ... [详细]
  • 题目描述:给定n个半开区间[a, b),要求使用两个互不重叠的记录器,求最多可以记录多少个区间。解决方案采用贪心算法,通过排序和遍历实现最优解。 ... [详细]
  • C++: 实现基于类的四面体体积计算
    本文介绍如何使用C++编程语言,通过定义类和方法来计算由四个三维坐标点构成的四面体体积。文中详细解释了四面体体积的数学公式,并提供了两种不同的实现方式。 ... [详细]
  • UNP 第9章:主机名与地址转换
    本章探讨了用于在主机名和数值地址之间进行转换的函数,如gethostbyname和gethostbyaddr。此外,还介绍了getservbyname和getservbyport函数,用于在服务器名和端口号之间进行转换。 ... [详细]
  • 本文详细介绍了如何构建一个高效的UI管理系统,集中处理UI页面的打开、关闭、层级管理和页面跳转等问题。通过UIManager统一管理外部切换逻辑,实现功能逻辑分散化和代码复用,支持多人协作开发。 ... [详细]
  • 扫描线三巨头 hdu1928hdu 1255  hdu 1542 [POJ 1151]
    学习链接:http:blog.csdn.netlwt36articledetails48908031学习扫描线主要学习的是一种扫描的思想,后期可以求解很 ... [详细]
  • Splay Tree 区间操作优化
    本文详细介绍了使用Splay Tree进行区间操作的实现方法,包括插入、删除、修改、翻转和求和等操作。通过这些操作,可以高效地处理动态序列问题,并且代码实现具有一定的挑战性,有助于编程能力的提升。 ... [详细]
  • 本文探讨了如何通过最小生成树(MST)来计算严格次小生成树。在处理过程中,需特别注意所有边权重相等的情况,以避免错误。我们首先构建最小生成树,然后枚举每条非树边,检查其是否能形成更优的次小生成树。 ... [详细]
  • 本文介绍如何使用Objective-C结合dispatch库进行并发编程,以提高素数计数任务的效率。通过对比纯C代码与引入并发机制后的代码,展示dispatch库的强大功能。 ... [详细]
  • C++实现经典排序算法
    本文详细介绍了七种经典的排序算法及其性能分析。每种算法的平均、最坏和最好情况的时间复杂度、辅助空间需求以及稳定性都被列出,帮助读者全面了解这些排序方法的特点。 ... [详细]
  • 本文探讨了如何在给定整数N的情况下,找到两个不同的整数a和b,使得它们的和最大,并且满足特定的数学条件。 ... [详细]
  • 本文探讨了 C++ 中普通数组和标准库类型 vector 的初始化方法。普通数组具有固定长度,而 vector 是一种可扩展的容器,允许动态调整大小。文章详细介绍了不同初始化方式及其应用场景,并提供了代码示例以加深理解。 ... [详细]
  • 本实验主要探讨了二叉排序树(BST)的基本操作,包括创建、查找和删除节点。通过具体实例和代码实现,详细介绍了如何使用递归和非递归方法进行关键字查找,并展示了删除特定节点后的树结构变化。 ... [详细]
author-avatar
手机用户2602913361
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有