我是靠谱客的博主 孝顺哈密瓜,最近开发中收集的这篇文章主要介绍[AtCoder Grand Contest 016] D: Xor Replace (agc016d),觉得挺不错的,现在分享给大家,希望可以做个参考。
概述
原题链接
https://agc016.contest.atcoder.jp/tasks/agc016_d
Description
给出一个n个数的序列a,每次操作可以将一个数变成整个序列的值的异或
求最少需要多少次才能将a变成目标序列b
无法完成输出-1
n<=100000
Solution
考虑操作的本质
只要按位稍微分析一下,就可以发现,题目相当于一开始手里抓着整个序列a的异或,一次操作可以将手上的数与序列中的某个数换过来
知道操作的本质就简单许多
如果能够完成,方便起见将a的异或和b的异或分别加到序列末,排序后两个序列显然完全相同
这样就把-1判掉了
只有a[i]!=b[i]的位置我们是需要调整的
那么将a,b离散化,a[i]与b[i]连边
将一个联通块内操作所需要的次数就是联通块大小
而联通块之间跳需要1的代价
此时由于我们一开始手上抓着的是a的异或和
如果它刚好在某一个联通块里面就不用考虑了,否则就必须将它自己看做一个大小为0的联通块
最后-1是在联通块之间跳的次数是联通块个数-1
联通块可以用并查集维护,离散化我实在懒惰了用了map
Code
#include <cstdio>
#include <cstdlib>
#include <cmath>
#include <cstring>
#include <algorithm>
#include <iostream>
#include <map>
#define fo(i,a,b) for(int i=a;i<=b;i++)
#define fod(i,a,b) for(int i=a;i>=b;i--)
#define N 100005
using namespace std;
int a[N],b[N],n,m,d[N],c[N],n1,f[N];
map <int,int> h;
bool bz[N];
int getf(int k)
{
if(f[k]==0||f[k]==k) return k;
return (f[k]=getf(f[k]));
}
int main()
{
cin>>n;
int v=0;
fo(i,1,n) scanf("%d",&a[i]),v^=a[i],c[i]=a[i];
a[n+1]=c[n+1]=v;
bool pd=0;
v=0;
fo(i,1,n) scanf("%d",&b[i]),d[i]=b[i],v^=b[i];
b[++n]=d[n]=v;
sort(c+1,c+n+1),sort(d+1,d+n+1);
n1=0;
fo(i,1,n)
{
if(c[i]!=d[i])
{
printf("-1n");
return 0;
}
if(h.count(c[i])==0) h[c[i]]=++n1;
if(h.count(d[i])==0) h[d[i]]=++n1;
}
int ans=0;
fo(i,1,n)
{
a[i]=h[a[i]],b[i]=h[b[i]];
if(a[i]!=b[i])
{
if(i<n) ans++;
int fx=getf(a[i]),fy=getf(b[i]);
f[fx]=fy;
bz[fx]=1,bz[fy]=1;
}
}
if(ans==0) printf("0n");
else
{
bz[a[n]]=bz[b[n]]=1;
fo(i,1,n1) if(bz[i]&&getf(i)==i) ans++;
printf("%dn",ans-1);
}
}
最后
以上就是孝顺哈密瓜为你收集整理的[AtCoder Grand Contest 016] D: Xor Replace (agc016d)的全部内容,希望文章能够帮你解决[AtCoder Grand Contest 016] D: Xor Replace (agc016d)所遇到的程序开发问题。
如果觉得靠谱客网站的内容还不错,欢迎将靠谱客网站推荐给程序员好友。
本图文内容来源于网友提供,作为学习参考使用,或来自网络收集整理,版权属于原作者所有。
发表评论 取消回复