• uoj#310. 【UNR #2】黎明前的巧克力


    题目描述

    题解

    考虑到选出的两个集合的异或值为 $0$ ,所以我们可以看做找出集合,其异或值为 $0$ ,然后如果这个集合大小是 $x$ ,对答案的贡献就是 $2^x$

    所以我们考虑每个 $i$ 对应一个多项式 $(1+2x^{a_i})$ ,只要我们把多项式乘起来即可

    我们考虑 $fwt$ 过程中 $i$ 位置上的数对 $j$ 位置的贡献是数值乘上 $(-1)^{popcount(iAnd j)}$ ,不难发现每个多项式 $fwt$ 后数值要么是 $-1$ ,要么是 $3$

    所以我们可以把这些多项式相加后做 $fwt$ ,然后设这个位置有 $x$ 个 $-1$,那就有 $n-x$ 个 $3$ ,解方程即可,然后这一位将变成 $(-1)^x3^{n-x}$,之后再做 $ifwt$ 即可

    最后答案因为不能选出空集所以要减一

    代码

    #include <bits/stdc++.h>
    using namespace std;
    const int N=1<<20,P=998244353;
    const int U=(P+1)>>1,V=748683265;
    int n,a[N],w[N];
    void fwt(int *a,int o){
        for (int i=1;i<N;i<<=1)
            for (int j=0;j<N;j+=(i<<1))
                for (int x,y,k=0;k<i;k++){
                    x=a[j+k];y=a[i+j+k];
                    a[j+k]=(x+y)%P;a[i+j+k]=(x-y+P)%P;
                    if (o) a[j+k]=1ll*a[j+k]*U%P,
                        a[i+j+k]=1ll*a[i+j+k]*U%P;
                }
    }
    int main(){
        scanf("%d",&n);w[0]=1;
        for (int i=1;i<N;i++) w[i]=3ll*w[i-1]%P;
        for (int x,i=1;i<=n;i++)
            scanf("%d",&x),a[0]++,a[x]+=2;
        fwt(a,0);for (int x,i=0;i<N;i++)
            x=1ll*(3*n-a[i]+P)*V%P,
            a[i]=1ll*((x&1)?P-1:1)*w[n-x]%P;
        fwt(a,1);cout<<(a[0]+P-1)%P<<endl;return 0;
    }
  • 相关阅读:
    mycat实例(1)
    Java连接Oracle数据库的示例代码
    文本处理grep命令
    回调函数
    算法基础--快排序,堆排序,归并排序
    c++ 中double与string之间的转换,char *
    c++ 类型转换
    allocator class
    csapp读书笔记-并发编程
    树的遍历-递归方法,非递归方法
  • 原文地址:https://www.cnblogs.com/xjqxjq/p/12266789.html
Copyright © 2020-2023  润新知