[cf1305G]Kuroni and Antihype
对整个过程构造一张有向图,其中$(x,y)\in E$当且仅当$x$把$y$加入,且边权为$a_{x}$
显然这是一棵外向树森林,并再做如下两个构造:
1.新建一个点$a_{0}=0$,将其向所有入度为0的点连边
2.将所有边变为无向边,且边权修改为$a_{x}+a_{y}$($x$和$y$为两端点)
显然最终得到的是一棵树,并且这棵树能被某个过程得到当且仅当$\forall (x,y)\in E,a_{x}\and a_{y}=0$
另一方面,初始答案即边权和,第一个构造中新增的边边权为0,第二个构造中即每一个点都额外产生了原来入度次贡献,而除了0以外其余点入度均为1(而$a_{0}=0$不需要考虑),那么最终答案即边权和-$\sum_{i=1}^{n}a_{i}$
由于后者固定,问题也即求$E=\{(x,y,a_{x}+a_{y})\mid 0\le x,y\le n$且$a_{x}\and a_{y}=0\}$的最大生成树
考虑Boruvka算法,即维护若干个集合(初始每一个点均作为一个集合),并不断加入所有集合(向集合外)最大的出边,注意到每一次集合数回减小一半,最终轮数即$o(\log n)$
在每一轮中,需要求出该最大出边,不妨对每一个点都求出该最大出边即可
而对于$a_{x}$,即需要求出权值是$U\oplus a_{x}$的子集(其中$U$为全集)中与$x$不在同一个连通块中且权值最大的点,对每一个集合预处理其子集权值最大的和不在同一个连通块中权值次大的即可
关于这个,做一个类似于高维前缀和的操作即可
时间复杂度为$o(n\alpha(n)+m2^{m}\log n)$(其中$m\approx 18$),可以通过

1 #include<bits/stdc++.h>
2 using namespace std;
3 #define N 200005
4 #define M 18
5 #define ll long long
6 #define pii pair<int,int>
7 #define fi first
8 #define se second
9 struct Data{
10 pii mx,cmx;
11 }g[1<<M];
12 int n,a[N],fa[N];
13 ll ans;
14 pii f[N];
15 int find(int k){
16 if (k==fa[k])return k;
17 return fa[k]=find(fa[k]);
18 }
19 void merge(int x,int y,int z){
20 x=find(x),y=find(y);
21 if (x!=y){
22 fa[x]=y;
23 ans+=z;
24 }
25 }
26 Data merge(Data x,Data y){
27 if (x.mx<y.mx)swap(x,y);
28 if (find(x.mx.se)==find(y.mx.se))swap(y.mx,y.cmx);
29 x.cmx=max(x.cmx,y.mx);
30 return x;
31 }
32 int main(){
33 scanf("%d",&n);
34 for(int i=1;i<=n;i++){
35 scanf("%d",&a[i]);
36 ans-=a[i];
37 }
38 for(int i=0;i<=n;i++)fa[i]=i;
39 while (1){
40 bool flag=0;
41 for(int i=1;i<=n;i++)
42 if (find(i)!=find(0)){
43 flag=1;
44 break;
45 }
46 if (!flag)break;
47 for(int i=0;i<(1<<M);i++)g[i].mx=g[i].cmx=make_pair(-1,-1);
48 for(int i=0;i<=n;i++)g[a[i]]=merge(g[a[i]],Data{make_pair(a[i],i),make_pair(-1,-1)});
49 for(int i=0;i<M;i++)
50 for(int j=0;j<(1<<M);j++)
51 if (j&(1<<i))g[j]=merge(g[j],g[j^(1<<i)]);
52 for(int i=0;i<=n;i++)f[i]=make_pair(-1,-1);
53 for(int i=0;i<=n;i++){
54 int pos=(a[i]^((1<<M)-1));
55 pii o=g[pos].mx;
56 if (o.fi>=0){
57 if (find(i)==find(o.se))o=g[pos].cmx;
58 if (o.fi>=0)f[find(i)]=max(f[find(i)],make_pair(o.fi+a[i],o.se));
59 }
60 }
61 for(int i=0;i<=n;i++)
62 if (f[i].fi>=0)merge(i,f[i].se,f[i].fi);
63 }
64 printf("%lld\n",ans);
65 return 0;
66 }
[cf1305G]Kuroni and Antihype的更多相关文章
- 2021record
2021-10-14 P2577 [ZJOI2004]午餐 2021-10-13 CF815C Karen and Supermarket(小小紫题,可笑可笑) P6748 『MdOI R3』Fall ...
- Codeforce-Ozon Tech Challenge 2020-D. Kuroni and the Celebration(交互题+DFS)
After getting AC after 13 Time Limit Exceeded verdicts on a geometry problem, Kuroni went to an Ital ...
- Codeforce-Ozon Tech Challenge 2020-C. Kuroni and Impossible Calculation(鸽笼原理)
To become the king of Codeforces, Kuroni has to solve the following problem. He is given n numbers a ...
- Codeforce-Ozon Tech Challenge 2020-B. Kuroni and Simple Strings(贪心)
B. Kuroni and Simple Strings time limit per test1 second memory limit per test256 megabytes inputsta ...
- Codeforce-Ozon Tech Challenge 2020-A. Kuroni and the Gifts
the i-th necklace has a brightness ai, where all the ai are pairwise distinct (i.e. all ai are diffe ...
- cf 1305 E. Kuroni and the Score Distribution
题目传送门:E. Kuroni and the Score Distribution 题目大意:给n和m,输出n个数,这些数里必须要有m对a[i]+a[j]==a[k] ( i < j < ...
- CF 1305E. Kuroni and the Score Distribution
题目大意:题目给定两个数n和m(1<=n<=5000,0<=m<=1e9)要求构造一个数列A,A中元素 大于等于1,小于等于1e9且满足严格递增 满足ai+aj=ak的(i,j ...
- Kuroni and the Punishment CodeForces - 1305F 随机函数mt19937 + 质因子分解
题意: 给你n个数,你每次操作可以对一个数加1或者减1,让你求你最少需要操作多少次可以使这n个数的公因子大于1 题解: 正常方法就是枚举质因子(假设质因子为x),然后对于这个数组中的数a[i],让a[ ...
- Codeforces 杂题集 2.0
记录一些没有写在其他随笔中的 Codeforces 杂题, 以 Problemset 题号排序 1326D2 - Prefix-Suffix Palindrome (Hard version) ...
随机推荐
- DCI架构是如何解决DDD战术建模缺点的?
摘要:将DCI架构总结成一句话就是:领域对象(Object)在不同的场景(Context)中扮演(Cast)不同的角色(Role),角色之间通过交互(Interactive)来完成具体的业务逻辑. 本 ...
- mapboxgl 纠偏百度地图
缘起 之前分享了mapboxgl 互联网地图纠偏插件,插件当时只集成了高德地图. 文章发布后,有小伙伴在后台留言,希望插件也能支持百度地图. 刚好国庆假期有时间就研究了一下. 插件加载瓦片原理 首先, ...
- xshell连接vmware系统完整版
设置静态ip需要修改文件一共有两个要修改的文件vi /etc/resolv.confvi /etc/sysconfig/network-scripts/ifcfg-eno16777736 第一个文件 ...
- 编程模仿MySql客服端
写在前面 通过自己编写的Java代码程序,去模仿实现MySql客服端的简单功能,最终以控制台操作,很像在Dos窗口通过命令操作MySql数据库. 关键问题 在编写过程中遇到的一些小问题和一些值得留心注 ...
- SudokuSolver 1.0:用C++实现的数独解题程序 【二】
本篇是 SudokuSolver 1.0:用C++实现的数独解题程序 [一] 的续篇. CQuizDealer::loadQuiz 接口实现 1 CQuizDealer* CQuizDealer::s ...
- Using C++ in VS Code
Using C++ in VS Code Get Started with C++ and Windows Subsystem for Linux in Visual Studio Code ...
- [JVM-6]类加载器
定义 前面说过加载阶段是一个可以让设计人员高度自控的模块,因为类文件的源头可以是多种多样的,代码生成.反射生成或从网络中生成等.因此类加载器作为对这些文件的处理就显得尤为重要. 但类加载器的功能不仅如 ...
- CSP踩被记
本来想起个清新脱俗的标题,但碍于语文功底不行,于是光明正大嫖了LiBoyi的高端创意,把这篇博客命名为踩被记. Day -6 用假暴力把真正解拍没了,伤心.Rp有点低 Day -4 信息学考,\(py ...
- 2021.7.15考试总结[NOIP模拟16]
ZJ模拟D2就是NB.. T1 Star Way To Heaven 谁能想到这竟是个最小生成树呢?(T1挂分100的高人JYF就在我身边 把上边界和下边界看成一个点和星星跑最小生成树,从上边界开始跑 ...
- 清除行列 牛客网 程序员面试金典 C++ Python
清除行列 牛客网 程序员面试金典 C++ Python 题目描述 请编写一个算法,若N阶方阵中某个元素为0,则将其所在的行与列清零. 给定一个N阶方阵int[]mat和矩阵的阶数n,请返回完成操作后的 ...