【做题】agc016d - XOR Replace——序列置换&环
原文链接 https://www.cnblogs.com/cly-none/p/9813163.html
题意:给出初始序列\(a\)和目标序列\(b\),都有\(n\)个元素。每次操作可以把\(a\)中的一个元素替换为\(a\)中所有元素的异或和,问最少操作多少次,把序列\(a\)变成序列\(b\),或判断无解。
\(n \leq 10^5\)
首先,这个“替换为所有元素的异或和”,其实就是一开始有\(v\)为\(a\)中所有元素异或和,每次操作把\(a\)中某个元素与\(v\)交换。再转化一下,也就是一开始\(a_0\)为\(a\)中所有元素的异或和,每次可以把\(a\)中的某个元素和\(a_0\)交换。
这样很容易能判断无解的情况。现在只考虑如何求出操作的最小次数。
先不考虑元素相等的情况。那么就变成了一个排列的置换。考虑其中的每一个环,设环大小为\(m \, (m > 1)\),假如其中不包含\(a_0\),那么这个环就需要\(m+1\)次操作来处理;否则只需要\(m-1\)次。
而当有元素相等时,我们也可以类似于排列处理。定义每一对\((a_i,b_i)\)产生一条\((a_i,b_i)\)的有向边。因为包含\(a_0\)的环的个数是固定的,所以要最小化答案,我们就要最小化环的个数。但是这道题的图存在特殊性质:每一个弱连通分量都是由若干个有向环组成的,这意味着一定存在欧拉闭迹。因此,每个连通分量都能用只一个环解决。这显然也是答案的下界。
然后还有\(a_0\)的一点细节。
- \(a_0 \neq b_0\)。那么,\(a_0\)一定在一个大于1的环上。最终答案减二。
- \(a_0 = b_0\),但\(a_0\)所在的弱连通分量大于1。这样,那个弱连通分量的操作个数能少一个。
- \(a_0 = b_0\),且\(a_0\)所在弱连通分量大小为1。那它对答案没有什么影响。
时间复杂度\(O(n \log n)\)。
#include <bits/stdc++.h>
using namespace std;
const int N = 100010;
int a[N],n,b[N],tmp,ans,cur,uni[N],sz[N],vis[N];
map<int,int> mp;
typedef pair<int,int> pii;
vector<int> vec;
template <typename tp>
void outputarr(tp *a,tp *b) {
while (a != b)
cerr << *a << ' ', ++ a;
cerr << endl;
}
int getfa(int pos) {
return pos == uni[pos] ? pos : uni[pos] = getfa(uni[pos]);
}
int main() {
scanf("%d",&n);
for (int i = 1 ; i <= n ; ++ i)
scanf("%d",&a[i]), tmp ^= a[i];
a[0] = tmp;
for (int i = 1 ; i <= n ; ++ i)
scanf("%d",&b[i]);
for (int i = 0 ; i <= n ; ++ i)
++ mp[a[i]];
for (int i = 1 ; i <= n ; ++ i) {
if (!mp.count(b[i]))
return puts("-1"), 0;
if (mp[b[i]] == 0) return puts("-1"), 0;
mp[b[i]]--;
}
for (int i = 0 ; i <= n ; ++ i)
if (mp[a[i]]) b[0] = a[i];
for (int i = 0 ; i <= n ; ++ i)
vec.push_back(a[i]);
sort(vec.begin(),vec.end());
vec.erase(unique(vec.begin(),vec.end()),vec.end());
for (int i = 0 ; i <= n ; ++ i)
a[i] = lower_bound(vec.begin(),vec.end(),a[i]) - vec.begin() + 1, \
b[i] = lower_bound(vec.begin(),vec.end(),b[i]) - vec.begin() + 1;
for (int i = 0 ; i <= n ; ++ i)
uni[a[i]] = a[i];
for (int i = 0 ; i <= n ; ++ i) {
int x = getfa(a[i]), y = getfa(b[i]);
if (x != y)
uni[x] = y;
}
for (int i = 0 ; i <= n ; ++ i) {
if (a[i] == b[i]) continue;
++ ans;
++ sz[getfa(a[i])];
}
for (int i = 0 ; i <= n ; ++ i) {
if (a[i] == b[i]) continue;
if (a[i] == getfa(a[i]) && !vis[a[i]]) {
++ ans;
vis[a[i]] = 1;
}
}
if (a[0] != b[0]) ans --;
if (sz[getfa(a[0])] > 1) ans --;
printf("%d\n",ans);
return 0;
}
小结:早上的训练赛搬了这道题。从我的FST中,也可以看出我对问题的考虑不够仔细。补题过程中思维混乱的问题也很明显,导致花费了很多时间。
【做题】agc016d - XOR Replace——序列置换&环的更多相关文章
- AGC016D - XOR Replace 置换/轮换
目录 题目链接 题解 代码 题目链接 AGC016D - XOR Replace 题解 可以发现一次操作相当于一次置换 对于每个a上的位置映射到b对应 可以找到置换群中的 所有轮换 一个k个元素的轮换 ...
- agc016D - XOR Replace(图论 智商)
题意 题目链接 给出两个长度为\(n\)的数组\(a, b\) 每次可以将\(a\)中的某个数替换为所有数\(xor\)之和. 若\(a\)数组可以转换为\(b\)数组,输出最少操作次数 否则输出\( ...
- [agc016d]xor replace
题意: 题解: 棒棒的神仙题...这题只是D题???(myh:看题五分钟,讨论两小时) 首先这个异或和是假的,比如我现在有$a=(a_1,a_2,a_3,a_4)$,操作一下$a_2$,就变成了$a= ...
- AtCoder Grand Contest 11~17 做题小记
原文链接https://www.cnblogs.com/zhouzhendong/p/AtCoder-Grand-Contest-from-11-to-20.html UPD(2018-11-16): ...
- CodeM美团点评编程大赛复赛 做题感悟&题解
[T1] [简要题意] 长度为N的括号序列,随机确定括号的方向:对于一个已确定的序列,每次消除相邻的左右括号(右左不行),消除后可以进一步合并和消除直到不能消为止.求剩下的括号的期望.\(N \l ...
- bzoj5108: [CodePlus2017]可做题
Description qmqmqm希望给sublinekelzrip出一道可做题.于是他想到了这么一道题目:给一个长度为n的非负整数序列ai,你需 要计算其异或前缀和bi,满足条件b1=a1,bi= ...
- SDOI2016 R1做题笔记
SDOI2016 R1做题笔记 经过很久很久的时间,shzr终于做完了SDOI2016一轮的题目. 其实没想到竟然是2016年的题目先做完,因为14年的六个题很早就做了四个了,但是后两个有点开不动.. ...
- AtCoder Grand Contest 1~10 做题小记
原文链接https://www.cnblogs.com/zhouzhendong/p/AtCoder-Grand-Contest-from-1-to-10.html 考虑到博客内容较多,编辑不方便的情 ...
- bzoj5108 [CodePlus2017]可做题 位运算dp+离散
[CodePlus2017]可做题 Time Limit: 10 Sec Memory Limit: 512 MBSubmit: 87 Solved: 63[Submit][Status][Dis ...
随机推荐
- java学习之路--String类方法的应用
消除字符串两端的空格 1.判断字符串第一个位置是否为空格,如果是继续向下判断,直到不是空格位置,末尾也是这样,往前判断,直到不是空格为止. 2.当开始和末尾都不是空格时,获取字符串. public s ...
- MySQL数据库(查询语句)
用户 角色 权限 select * from students1:查询这个表里的数据 select ssex from students1:在这个表里面查询ssex(性别) select dist ...
- Python学习之旅(十一)
Python基础知识(10):函数(Ⅱ) 一.全局变量和局部变量 局部变量:在函数内定义的变量,在函数内使用 全局变量:在函数外定义的变量,在程序任何地方都可以使用 1.全局变量与局部变量同名 这时函 ...
- phpredis中文开发文档
刚好要用看了网上翻译版本都是2011,2012年的,随手翻译一下新版 2017年10月28日23:48:08 使用方法 : Ctrl+F 官方英文版 https://github.com/phpred ...
- 将 ASP.NET Core 2.0 项目升级至 ASP.NET Core 2.1 RC 1
今天微软发布了 .NET Core 2.1 RC 1 ,虽然只是 Release Candidate 版,但已经可以在生产环境中使用. NET Core 2.1 RC is supported by ...
- Codeforces 677 - A/B/C/D/E - (Undone)
链接: A - Vanya and Fence - [水] AC代码: #include<bits/stdc++.h> using namespace std; ; int n,h; in ...
- JS常见的字符串操作
1.charAt() 获取字符串指定位置的字符 用法:strObj是字符串对象,index是指定的位置,(位置从0开始数) strObj.charAt(index) 2. indexOf() 方 ...
- 图->最短路径->多源最短路径(弗洛伊德算法Floyd)
文字描述 求每一对顶点间的最短路径,可以每次以一个顶点为源点,重复执行迪杰斯特拉算法n次.这样,便可求得每一对顶点之间的最短路径.总的执行时间为n^3.但是还有另外一种求每一对顶点间最短路径的方法,就 ...
- 苹果cms安装及配置详细教程
听说这个好!php+mysql的 下载 http://www.maccms.com/down.html 下载之后解压到你的网站跟目录中,就像这个样子的 后台目录 然后重要的一步来了,在ftp工具上 ...
- python基础(12)-包的导入&异常处理
包的导入 几种导入方式 import 包名 import time time.time() import 包名,包名 import time,sys time.time() sys.path from ...