Description
LF是毒瘤出题人中AK IOI2019,不屑于参加NOI的唯一的人。他对人说话,总是满口垃圾题目者也,教人半懂不懂的。因为他姓李,别人便从QQ群上的“毒瘤李Fee”这半懂不懂的话里,替他取下一个绰号,叫做李Fee。
李Fee一到机房,所有做题的人便都看着他笑,有的叫道,“李Fee,你又来出毒瘤题了!”他不回答,对验题人说,“我又出了两道题,给我验验。”便排出一排毒瘤题。大家又故意的高声嚷道,“你又暴露奸商本性拿毒瘤题骗钱剥削验题人了!”李Fee睁大眼睛说,“你怎么这样凭空污人清白……”“什么清白?我前天亲眼见你出了道毒瘤骗钱题,被PTY把std吊着打。” 李Fee便涨红了脸,额上的青筋条条绽出,争辩道,“出题人的题不能算骗……毒瘤!……出题人的题,能算毒瘤骗钱题么?”接连便是难懂的话,什么“多叉splay随机点分治”,什么“树链剖分套分治FFT”之类,引得众人都哄笑起来:机房内外充满了快活的空气。
虽然他的题十分毒瘤,但他的题还总是有买家。李Fee现在有N道毒瘤题,想将这些题出成一组题来骗大钱。然而显而易见的是,一组题的毒瘤程度不仅和每道题的毒瘤程度有关,也跟它们的排列顺序有关,李Fee需要将它们排列成最毒瘤但又最能骗钱的那个顺序。
具体来说,这N道题每题都有一个毒瘤值,它们构成了一个序列。李Fee心目中有一个理想的毒瘤值序列,这个序列并不一定每一题的毒瘤值都是原本N道题中出现的,所以李Fee准备进行一些改动。这些改动体现在毒瘤值上就是将某道题的毒瘤值改为所有题的毒瘤值的二进制异或值。但是,改动题目是很麻烦的,他想算出最少需要多少次改动才能将原本的毒瘤值序列改成理想的毒瘤值序列,李Fee忙于出毒瘤题,他想请发明O(1/n)算法用暴力搜过所有毒瘤题的你帮他算出答案。但是他是个奸商,所以他并不打算给你报酬。
李Fee一到机房,所有做题的人便都看着他笑,有的叫道,“李Fee,你又来出毒瘤题了!”他不回答,对验题人说,“我又出了两道题,给我验验。”便排出一排毒瘤题。大家又故意的高声嚷道,“你又暴露奸商本性拿毒瘤题骗钱剥削验题人了!”李Fee睁大眼睛说,“你怎么这样凭空污人清白……”“什么清白?我前天亲眼见你出了道毒瘤骗钱题,被PTY把std吊着打。” 李Fee便涨红了脸,额上的青筋条条绽出,争辩道,“出题人的题不能算骗……毒瘤!……出题人的题,能算毒瘤骗钱题么?”接连便是难懂的话,什么“多叉splay随机点分治”,什么“树链剖分套分治FFT”之类,引得众人都哄笑起来:机房内外充满了快活的空气。
虽然他的题十分毒瘤,但他的题还总是有买家。李Fee现在有N道毒瘤题,想将这些题出成一组题来骗大钱。然而显而易见的是,一组题的毒瘤程度不仅和每道题的毒瘤程度有关,也跟它们的排列顺序有关,李Fee需要将它们排列成最毒瘤但又最能骗钱的那个顺序。
具体来说,这N道题每题都有一个毒瘤值,它们构成了一个序列。李Fee心目中有一个理想的毒瘤值序列,这个序列并不一定每一题的毒瘤值都是原本N道题中出现的,所以李Fee准备进行一些改动。这些改动体现在毒瘤值上就是将某道题的毒瘤值改为所有题的毒瘤值的二进制异或值。但是,改动题目是很麻烦的,他想算出最少需要多少次改动才能将原本的毒瘤值序列改成理想的毒瘤值序列,李Fee忙于出毒瘤题,他想请发明O(1/n)算法用暴力搜过所有毒瘤题的你帮他算出答案。但是他是个奸商,所以他并不打算给你报酬。
Input
第一行1个正整数N,如题目所示。
第二行N个非负整数,表示初始的题目毒瘤值序列
第三行N个非负整数,表示理想的题目毒瘤值序列
第二行N个非负整数,表示初始的题目毒瘤值序列
第三行N个非负整数,表示理想的题目毒瘤值序列
Output
单独一行,一个整数,表示最少需要多少次改动
如果怎么改动都无法改成理想的毒瘤值序列,说明这组题出的相当失败,请输出-1
如果怎么改动都无法改成理想的毒瘤值序列,说明这组题出的相当失败,请输出-1
Sample Input
3 0 1 2 3 1 0
Sample Output
2 样例解释: 第一次,整个序列异或为3,把第一个数0换成3,序列变成3,1,2 第二次,整个序列异或为0,把第三个数2换成0,序列变成3,1,0
Data Constraint
对于10%的数据,1<=N<=5
对于30%的数据,1<=N<=10
另有20%的数据,毒瘤值为0或1
对于100%的数据,1<=N<=100000 毒瘤值<2^30
对于30%的数据,1<=N<=10
另有20%的数据,毒瘤值为0或1
对于100%的数据,1<=N<=100000 毒瘤值<2^30
Hint
不要被事物的表面现象所迷惑
做法(摘自jzoj):考虑操作的本质 只要按位稍微分析一下,就可以发现,题目相当于一开始手里抓着整个序列 a 的异 或,一次操作可以将手上的数与序列中的某个数换过来 知道操作的本质就简单许多 如果能够完成,方便起见将 a 的异或和 b 的异或分别加到序列末,排序后两个序 列显然完全相同,这样就把-1 判掉了 只有 a[i]!=b[i]的位置我们是需要调整的 那么将 a,b 离散化,a[i]与 b[i]连边 考虑统计答案 将一个联通块内换完所需要的次数明显就是联通块大小。 而联通块之间跳需要 1 的代价 此时由于我们一开始手上抓着的是 a 的异或和 如果它刚好在某一个联通块里面就不用考虑了,否则就必须将它自己看做一个大小 为 0 的联通块 联通块可以用并查集维护
1 #include <cstdio> 2 #include <cstring> 3 #include <iostream> 4 #include <algorithm> 5 #define N 100007 6 using namespace std; 7 int n,t1,t2,f[N],cnt,ans,sum[N]; 8 struct arr{ 9 int num,id; 10 }a[N],b[N]; 11 12 int cmp(arr x,arr y){ 13 return x.num<y.num; 14 } 15 16 int Cmp(arr x,arr y){ 17 return x.id<y.id; 18 } 19 20 int find(int x){ 21 if (f[x]==x) return x; 22 return f[x]=find(f[x]); 23 } 24 25 int main(){ 26 freopen("duliu.in","r",stdin); 27 freopen("duliu.out","w",stdout); 28 scanf("%d",&n); 29 for (int i=1;i<=n;i++) scanf("%d",&a[i].num),a[i].id=i,t1^=a[i].num; 30 for (int i=1;i<=n;i++) scanf("%d",&b[i].num),b[i].id=i,t2^=b[i].num; 31 a[n+1].num=t1,b[n+1].num=t2; 32 a[n+1].id=n+1,b[n+1].id=n+1; 33 int tmp=-0x3f3f3f3f; 34 sort(a+1,a+n+2,cmp); 35 sort(b+1,b+n+2,cmp); 36 for (int i=1;i<=n+1;i++){ 37 if (a[i].num!=b[i].num){ 38 printf("-1"); 39 return 0; 40 } 41 cnt++; 42 if (a[i].num==tmp) cnt--; 43 tmp=a[i].num; 44 a[i].num=cnt,b[i].num=cnt; 45 } 46 for (int i=1;i<=cnt;i++) f[i]=i; 47 sort(a+1,a+n+2,Cmp); 48 sort(b+1,b+n+2,Cmp); 49 for (int i=1;i<=n;i++){ 50 int u=a[i].num,v=b[i].num; 51 if (u==v) continue; 52 int q=find(u),p=find(v); 53 ans++; 54 if (q!=p) f[q]=p; 55 } 56 for (int i=1;i<=cnt;i++) sum[find(i)]++; 57 for (int i=1;i<=cnt;i++) 58 if (sum[i]>1) ans++; 59 if (f[a[n+1].num]==a[n+1].num&&sum[a[n+1].num]==1) ans++; 60 printf("%d",ans-1); 61 }