Description

题库链接

给你 \(k\) 个盒子,第 \(i\) 个盒子中有 \(n_i\) 个数,第 \(j\) 个数为 \(x_{i,j}\)。现在让你进行 \(k\) 次操作,第 \(i\) 次操作要求从第 \(i\) 个盒子中取出一个元素(这个元素最开始就在该盒子中),放入任意一个你指定的盒子中,要求经过 \(k\) 次操作后

  • 所有盒子元素个数和最开始相同;
  • 所有盒子元素总和相等

询问是否存在一种操作方式使之满足,若存在,输出任意一种方案即可。

\(1\leq k\leq 15,1\leq n_i\leq 5000,|x_{i,j}|\leq 10^9\)

Solution

由题,容易发现,对于任意一个盒子,会从其中拿出一个数,再从别处(或自己拿出的)添加一个数进来。

我们将数的拿出放入关系抽象成边,即从第 \(i\) 个盒子中拿出的数要放入 \(j\) 中,那么建边 \(i\rightarrow j\)。

因为这张图要求每个节点入度和出度均为 \(1\),显然这张图只能是若干个无相交的环构成的。

现在,我们考虑所有的拿出放入关系:

假设我要从第 \(i\) 个盒子中拿出元素 \(x\),那么要使得这个盒子满足最终条件,应该被放入的元素为 \(S-sum_i+x\),其中 \(S\) 为最终每个盒子的元素总和,\(sum_i\) 表示第 \(i\) 个盒子最初的元素总和。

那么我们建边 \(x\rightarrow S-sum_i+x\)(注意:此时图与之前建的图不同)。我们需要在这张图中找到所有满足下列条件的环:

  • 环上每个元素属于不同盒子;
  • 环上每种盒子只出现一次

用 \(dfs\) 找到这些环之后我们可以将盒子状压。具体地,令 \(f_i\) 表示状态 \(i\) 中所有的盒子构成的满足条件的图是否存在。转移枚举子集 \(dp\)。

若 \(f_{2^k-1}=1\) 即有解。注意另开数据记录转移关系,方便输出方案。

Code

#include <bits/stdc++.h>
#define ll long long
#define pb push_back
using namespace std;
const int N = 5000*15+5, B = (1<<15)+5; map<ll, int> mp;
int k, n[20], id[N], kp[N], tot;
int bin[20], x[16][5005], f[B], ok[B], p[B], vis[N], s[N], top;
ll sum[20], S;
vector<int> to[N], re[B];
int l[20], r[20]; void dfs(int u, int st) {
if (vis[u]) {
int now = 0;
for (int i = top; i; i--) {
now |= bin[id[s[i]]-1];
if (u == s[i]) break;
}
if (!ok[now]) {
ok[now] = 1;
for (int i = top; i; i--) {
re[now].pb(s[i]);
if (u == s[i]) break;
}
}
return;
}
if (st&bin[id[u]-1]) return;
st |= bin[id[u]-1], vis[u] = 1, s[++top] = u;
for (auto v : to[u]) dfs(v, st);
vis[u] = 0, --top;
}
int main() {
bin[0] = 1;
for (int i = 1; i <= 15; i++) bin[i] = bin[i-1]<<1;
scanf("%d", &k);
for (int i = 1; i <= k; i++) {
scanf("%d", &n[i]);
for (int j = 1; j <= n[i]; j++)
scanf("%d", &x[i][j]), mp[x[i][j]] = ++tot,
kp[tot] = x[i][j], id[tot] = i, sum[i] += x[i][j];
S += sum[i];
}
if (S%k) {puts("No"); return 0; }
S /= k;
for (int i = 1; i <= k; i++)
for (int j = 1; j <= n[i]; j++)
if (mp.count(S-sum[i]+x[i][j])) to[mp[x[i][j]]].pb(mp[S-sum[i]+x[i][j]]);
for (int i = 1; i <= tot; i++)
dfs(i, 0);
f[0] = 1;
for (int i = 0; i < bin[k]; i++)
if (f[i]) {
int C = i^(bin[k]-1);
for (int j = C; j; j = (j-1)&C)
if (ok[j])
f[i|j] = 1, p[i|j] = i;
}
if (!f[bin[k]-1]) {puts("No"); return 0; }
int x = bin[k]-1;
while (x) {
int U = x-p[x];
for (auto i : re[U]) {
l[id[mp[S-sum[id[i]]+kp[i]]]] = S-sum[id[i]]+kp[i],
r[id[mp[S-sum[id[i]]+kp[i]]]] = id[i];
}
x = p[x];
}
puts("Yes");
for (int i = 1; i <= k; i++)
printf("%d %d\n", l[i], r[i]);
return 0;
}

[Codeforces 1242C]Sum Balance的更多相关文章

  1. Codeforces Round #599 (Div. 1) C. Sum Balance 图论 dp

    C. Sum Balance Ujan has a lot of numbers in his boxes. He likes order and balance, so he decided to ...

  2. Codeforces 85D Sum of Medians(线段树)

    题目链接:Codeforces 85D - Sum of Medians 题目大意:N个操作,add x:向集合中加入x:del x:删除集合中的x:sum:将集合排序后,将集合中全部下标i % 5 ...

  3. Codeforces Round #599 (Div. 2) E. Sum Balance

    这题写起来真的有点麻烦,按照官方题解的写法 先建图,然后求强连通分量,然后判断掉不符合条件的换 最后做dp转移即可 虽然看起来复杂度很高,但是n只有15,所以问题不大 #include <ios ...

  4. Codeforces 1442D - Sum(找性质+分治+背包)

    Codeforces 题面传送门 & 洛谷题面传送门 智商掉线/ll 本来以为是个奇怪的反悔贪心,然后便一直往反悔贪心的方向想就没想出来,看了题解才发现是个 nb 结论题. Conclusio ...

  5. Codeforces 1303G - Sum of Prefix Sums(李超线段树+点分治)

    Codeforces 题面传送门 & 洛谷题面传送门 个人感觉这题称不上毒瘤. 首先看到选一条路径之类的字眼可以轻松想到点分治,也就是我们每次取原树的重心 \(r\) 并将路径分为经过重心和不 ...

  6. codeforces 616E Sum of Remainders (数论,找规律)

    E. Sum of Remainders time limit per test 2 seconds memory limit per test 256 megabytes input standar ...

  7. Codeforces 85D Sum of Medians

    传送门 D. Sum of Medians time limit per test 3 seconds memory limit per test 256 megabytes input standa ...

  8. Codeforces 616E - Sum of Remainders

    616E Sum of Remainders Calculate the value of the sum: n mod 1 + n mod 2 + n mod 3 + - + n mod m. As ...

  9. 数据结构(线段树):CodeForces 85D Sum of Medians

    D. Sum of Medians time limit per test 3 seconds memory limit per test 256 megabytes input standard i ...

随机推荐

  1. NFS挂载参数

    mount -t nfs -o rw,bg,hard,nointr,rsize=32768,wsize=32768,tcp,actimeo=0,vers=3,timeo=6 192.168.12.50 ...

  2. Java开发笔记(一百四十三)FXML布局的基本格式

    前面介绍了JavaFX的常见控件用法,虽然JavaFX控件比起AWT与Swing要好用些,但是一样通过代码编写控件界面,并没有提高什么开发效率.要想浏览界面的展示效果,都必须运行测试程序才能观看,即使 ...

  3. js确定取消—js确定取消判断

    国瑞前端: js确定取消,在html界面中,有css模拟的模态框,这样显示的就会更好看一些,那么javascript有没有自带的弹框呢,当然是有的,接下来我就来给大家介绍一下把: js确定取消-警告框 ...

  4. go开发环境

    1.go 下载地址 https://studygolang.com/dl 根据操作系统 下载相应的安装包 2.设置环境变量 goroot gopath path 增加%goroot%\bin 3.开发 ...

  5. 【题解】Luogu P5400 [CTS2019]随机立方体

    原题传送门 毒瘤计数题 我们设\(dp_i\)表示至少有\(i\)个极大数字的概率,\(ans_i\)表示恰好有\(i\)个极大数的概率,\(mi=Min(n,m,l)\) 易知: \[dp_i=\s ...

  6. POI2015 WYC

    也许更好的阅读体验 \(\mathcal{Description}\) 给定一张n个点m条边的带权有向图,每条边的边权只可能是1,2,3中的一种.将所有可能的路径按路径长度排序,请输出第k小的路径的长 ...

  7. Redis cluster的核心原理分析

    一.节点间的内部通信机制 1.基础通信原理 (1)redis cluster节点间采取gossip协议进行通信 跟集中式不同,不是将集群元数据(节点信息,故障,等等)集中存储在某个节点上,而是互相之间 ...

  8. AngularJS重型前端框架

    一.AngularJs AngularJs是一种前端的重型框架,而现在正在被Aue.js所取代.而Aue的和AngularJs大同小异. AngularJs核心功能有MVC.模块化.自动化双向数据绑定 ...

  9. 自学Python编程的第一天----------来自苦逼的转行人

    学习Python的第一天,也是我第一次写博客的一天,不怎么会写博客,也不怎么会Python,也不怎么会写总结.在学Python的第一天发现自己脑子不是很好用,在学习过程中出现很多错误,错误锦集如下,哈 ...

  10. global position

    观察, GestureDetector( child: CustomPaint(painter: StudyPaint(points: _points)), onPanEnd: (DragEndDet ...