题解 CF461D 【Appleman and Complicated Task】

传送门

挺难的思维题。

考虑将"x"视作1,将"o"视作0,则一个点周围四个点的异或和为0。

这样,我们得到 a i , j = a i − 1 , j − 1 ⊕ a i − 1 , j + 1 ⊕ a i − 2 , j a_{i,j}=a_{i-1,j-1}\oplus a_{i-1,j+1}\oplus a_{i-2,j} ai,j=ai1,j1ai1,j+1ai2,j

假如我们第一行已知,我们会发现这样一个东西(网上嫖的)

其中例如246的数字序列,表示这个位置的值等于第一行2,4,6位置异或起来的值

题解 CF461D 【Appleman and Complicated Task】_第1张图片

容易发现,一个点的值总是第一行某些位置的值异或起来,且这些位置同奇同偶。再分析一下,对于一个点 ( i , j ) (i,j) (i,j),发现它是第一行 l , l + 2 , . . . , r l,l+2,...,r l,l+2,...,r异或起来的值,其中 l = a b s ( i + j ) , r = m i n ( i + j , ( n − 1 ) × 2 − ( i + j ) ) l=abs(i+j),r=min(i+j,(n-1)\times 2-(i+j)) l=abs(i+j),r=min(i+j,(n1)×2(i+j))。(下标从0开始)

所以对第一行奇偶位置分开做异或前缀和得到 s i s_i si,那么对于一个给出的信息 ( i , j , t ) (i,j,t) (i,j,t),可以转化成第一行 s [ l ] ⊕ s [ r + 2 ] = t s[l]\oplus s[r+2]=t s[l]s[r+2]=t l − 2 l-2 l2 r r r也是可以的,不过 l − 2 l-2 l2会变成负数不好处理),即这两个数是否相等。然后就很套路了,拆点,用并查集判断即可。

C o d e   b e l o w : Code\ below: Code below:

#include
#define ts cout<<"ok"<
#define int long long
#define hh puts("")
#define pc putchar
#define mo 1000000007
//#define getchar() (p1==p2&&(p2=(p1=buf)+fread(buf,1,1<<21,stdin),p1==p2)?EOF:*p1++)
//char buf[1<<21],*p1=buf,*p2=buf;
using namespace std;
const int N=200005;
int n,k,f[N];
char s[5];
inline int read(){
    int ret=0,ff=1;char ch=getchar();
    while(!isdigit(ch)){if(ch=='-') ff=-1;ch=getchar();}
    while(isdigit(ch)){ret=ret*10+(ch^48);ch=getchar();}
    return ret*ff;
}
void write(int x){if(x<0){x=-x,pc('-');}if(x>9) write(x/10);pc(x%10+48);}
void writeln(int x){write(x),hh;}
void writesp(int x){write(x),pc(' ');}
int ksm(int x,int y){
    int res=1;
    while(y){
        if(y&1) res=res*x%mo;
        y>>=1;
        x=x*x%mo;
    }
    return res;
}
int find(int x){
    return f[x]==-1?x:f[x]=find(f[x]);
}
bool merge(int x,int y){
    int tx=find(x),ty=find(y);
    if(find(x^1)==ty) return 0;
    if(find(y^1)==tx) return 0;
    if(tx!=ty) f[tx]=ty;
    return 1;
}
bool check(int l,int r,int t){
    if(t==0){//s[l]^s[r]=0
        return merge(l*2,r*2)&&merge(l*2+1,r*2+1);
    }
    else{//s[l]^s[r]=1
        return merge(l*2,r*2+1)&&merge(l*2+1,r*2);
    }
}
signed main(){
    n=read(),k=read();
    memset(f,-1,sizeof(f));
    for(int i=1;i<=k;i++){
        int x=read()-1,y=read()-1;
        int l=x-y,r=x+y;
        if(l<0) l=-l;
        if(r>n-1) r=(n-1)*2-r;
        scanf("%s",s);
        if(!check(l,r+2,s[0]=='x'?0:1)){
            write(0);
            return 0;
        }
    }
    int ans=0;
    for(int i=0;i<=(n+2)*2;i++)
        if(find(i)==i)
            ans++;
    write(ksm(2,ans/2-2));
    return 0;
}

你可能感兴趣的:(题解)