【学习笔记】[ABC274Ex] XOR Sum of Arrays

有点难😅

真的是 A B C ABC ABC的难度吗😅

非常精妙的哈希题目。

定义矩阵乘法: c i , j = ⊕ ( a i , k & b k , j ) c_{i,j}=\oplus (a_{i,k}\& b_{k,j}) ci,j=(ai,k&bk,j)

之所以可以矩阵乘法是因为满足 ( a ⊕ b ) & c = ( a & c ) ⊕ ( b & c ) (a\oplus b)\& c=(a\& c)\oplus (b\& c) (ab)&c=(a&c)(b&c)

其实非常好验证,就把两个运算符按顺序写下来,然后把括号拆开,看等式两边是否恒等即可。

定义对于序列 { a i } \{a_i\} {ai}的哈希规则为,将每个 a k a_k ak与上 k − 1 k-1 k1 p p p后的结果全部异或起来

因为 ( a ⊕ b ) & p = ( a & p ) ⊕ ( b & p ) (a\oplus b)\&p=(a\& p)\oplus (b\& p) (ab)&p=(a&p)(b&p),所以如果 c i = a i ⊕ b i c_i=a_i\oplus b_i ci=aibi,那么 { c i } \{c_i\} {ci}的哈希值就是把 { a i } \{a_i\} {ai} { b i } \{b_i\} {bi}对应的哈希值异或起来

但是发现与上的 p p p都是相同的,所以这个方案冲突的概率很大

仔细观察正解的代码,发现他是把 p p p设计成了一个 M × M M\times M M×M 01 01 01矩阵 (其中 M M M表示二进制位数,也就是 64 64 64), a i a_i ai看成一个长度为 M M M 01 01 01向量 ,这个向量的第 i i i位就是 a i a_i ai在二进制下的第 i i i

不妨形式化的写一下,首先我们随机一个矩阵 p p p,哈希值就是 ⊕ i = 1 n v i p i − 1 \oplus_{i=1}^nv_ip^{i-1} i=1nvipi1。发现 c i = a i ⊕ b i c_{i}=a_{i}\oplus b_{i} ci=aibi那不就是把两个向量做按位异或吗。

直接倍增的复杂度是 O ( n M 2 log ⁡ n ) O(nM^2\log n) O(nM2logn),考虑优化。

注意到是 01 01 01矩阵,所以可以把同一行压成一个 64 64 64位整数,这样转移优化到了 O ( M ) O(M) O(M)

最终复杂度 O ( n M log ⁡ n ) O(nM\log n) O(nMlogn)。瓶颈在于倍增预处理,不知道可不可以做的更好。

#include<bits/stdc++.h>
#define ll long long
#define fi first
#define ull unsigned long long
using namespace std;
const int N=5e5+5;
const int M=64;
mt19937_64 t(time(0));
struct Matrix{
    ull c[M];
    Matrix(){memset(c,0,sizeof c);}
    Matrix operator *(const Matrix &a)const{
        Matrix r;
        for(int i=0;i<M;i++){
            for(int j=0;j<M;j++){
                if(c[i]>>j&1){
                    r.c[i]^=a.c[j];
                }
                
            }
        }return r;
    }
}pw[20];
int n,m;
ll a[N];
ull st[N][20];
ull get(ull x,Matrix y){
    ull res(0);
    for(int i=0;i<M;i++){
        if(x>>i&1)res^=y.c[i];
    }
    return res;
}
void write(ull x){
    if(x<10){
        cout<<(char)('0'+x);
        return;
    }write(x/10),cout<<(char)('0'+x%10);
}
int main(){
    ios::sync_with_stdio(false);
    cin.tie(0),cout.tie(0);
    cin>>n>>m;srand(time(0));
    for(int i=1;i<=n;i++)cin>>a[i],st[i][0]=a[i];
    for(int i=0;i<M;i++){
        pw[0].c[i]=t();
    }
    for(int i=1;i<20;i++)pw[i]=pw[i-1]*pw[i-1];
    for(int j=1;j<20;j++){
        for(int i=1;i<=n-(1<<j)+1;i++){
            st[i][j]=st[i][j-1]^get(st[i+(1<<j-1)][j-1],pw[j-1]);
        }
    }
    for(int i=1;i<=m;i++){
        int b,c,d,e,f,g;
        cin>>b>>c>>d>>e>>f>>g;
        for(int j=19;j>=0;j--){
            if(b+(1<<j)-1<=c&&f+(1<<j)-1<=g&&(st[b][j]^st[d][j])==st[f][j]){
                b+=(1<<j),d+=(1<<j),f+=(1<<j);
            }
        }
        if(b>c){
            if(f>g)cout<<"No"<<"\n";
            else cout<<"Yes"<<"\n";
        }
        else{
            if(f>g)cout<<"No"<<"\n";
            else if((a[b]^a[d])<a[f])cout<<"Yes"<<"\n";
            else cout<<"No"<<"\n";
        }
    }
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值