zoukankan      html  css  js  c++  java
  • [agc016d]xor replace

    题意:

    题解:

    棒棒的神仙题。。。这题只是D题???(myh:看题五分钟,讨论两小时)

    首先这个异或和是假的,比如我现在有$a=(a_1,a_2,a_3,a_4)$,操作一下$a_2$,就变成了$a=(a_1,a_1oplus a_2oplus a_3oplus a_4,a_3,a_4)$;

    再操作一下$a_3$,因为$a_ioplus a_i=0$,它就变回了$a=(a_1,a_1oplus a_2oplus a_3oplus a_4,a_2,a_4)$。

    所以这个操作只是第一次把一个位置变成全体的异或和,后面就是在交换两个数的位置。。。

    可以考虑把多出来的这个全体异或和放到$N+1$的位置,然后就变成了一次操作是交换$1$到$N$中的一个位置的数和第$N+1$个位置的数,求要多少次操作把数组a变成数组b。

    无解的情况显然:如果两个数组排序后有位置不同则必定无解,先判掉;

    把数组离散化,然后对于位置$i$,若$a_i eq b_i$则从$a_i$到$b_i$连一条边;

    这样子对于每个联通块,设其大小为$S$,则必定可以用$S-1$次操作使其中的位置全部满足条件(感性理解一下?);

    在每个联通块之间,需要额外的一次操作使$N+1$的位置在两个块之间转换;

    最后要特殊考虑$N+1$这个位置,如果它已经在某个联通块内则没有影响,否则要单独作为一个联通块考虑,答案++;

    所以最后的答案就是边数+联通块数-1,用并查集xjb维护一下即可。

    代码:

     1 #include<algorithm>
     2 #include<iostream>
     3 #include<cstring>
     4 #include<cstdio>
     5 #include<cmath>
     6 #include<queue>
     7 #include<map>
     8 #define inf 2147483647
     9 #define eps 1e-9
    10 using namespace std;
    11 typedef long long ll;
    12 int n,ans=0,ta=0,tb=0,cnt=0,fa[100001],a[100001],b[100001],aa[100001],bb[100001],lsh[200001];
    13 map<int,bool>mp;
    14 int ff(int u){
    15     return fa[u]==u?u:fa[u]=ff(fa[u]);
    16 }
    17 int main(){
    18     scanf("%d",&n);
    19     for(int i=1;i<=n;i++){
    20         scanf("%d",&a[i]);
    21         ta^=a[i];
    22         aa[i]=a[i];
    23     }
    24     for(int i=1;i<=n;i++){
    25         scanf("%d",&b[i]);
    26         tb^=b[i];
    27         bb[i]=b[i];
    28     }
    29     n++;
    30     a[n]=aa[n]=ta,b[n]=bb[n]=tb;
    31     sort(aa+1,aa+n+1);
    32     sort(bb+1,bb+n+1);
    33     for(int i=1;i<n;i++){
    34         if(aa[i]!=bb[i])return puts("-1"),0;
    35     }
    36     for(int i=1;i<n;i++){
    37         if(a[i]!=b[i]){
    38             lsh[++cnt]=a[i];
    39             lsh[++cnt]=b[i];
    40             ans++;
    41         }
    42     }
    43     lsh[++cnt]=ta;
    44     lsh[++cnt]=tb;
    45     if(!ans)return puts("0"),0;
    46     sort(lsh+1,lsh+cnt+1);
    47     cnt=unique(lsh+1,lsh+cnt+1)-lsh-1;
    48     for(int i=1;i<=cnt;i++)fa[i]=i;
    49     for(int i=1;i<=n;i++){
    50         if(a[i]!=b[i]){
    51             a[i]=lower_bound(lsh+1,lsh+cnt+1,a[i])-lsh;
    52             b[i]=lower_bound(lsh+1,lsh+cnt+1,b[i])-lsh;
    53             if(!mp[a[i]])mp[a[i]]=true;
    54             if(!mp[b[i]])mp[b[i]]=true;
    55             int f1=ff(a[i]),f2=ff(b[i]);
    56             fa[f1]=f2;
    57         }
    58     }
    59     for(int i=1;i<=cnt;i++){
    60         if(fa[i]==i)ans++;
    61     }
    62     printf("%d",ans-1);
    63     return 0;
    64 }
  • 相关阅读:
    Python 从入门到实践
    Python 斐波那契数列
    Python 纸牌游戏
    Python hangman小游戏
    BC #49 1001 Untitled
    BC#50 1003 The mook jong
    BC #50 1001 Distribution money
    vector
    stack
    queue
  • 原文地址:https://www.cnblogs.com/dcdcbigbig/p/9826557.html
Copyright © 2011-2022 走看看