zoukankan      html  css  js  c++  java
  • CF1119H Triple

    你的生日礼物是(n)个整数三元组。第(i)个三元组是({a_i,b_i,c_i}),所有数字(j)都满足(0le j<2^k),(k)是一个固定的数字,将由题目输入。

    一天,你玩三元组玩腻了,所以你有了3个新的整数(x,y,z),然后填充了(n)个数组。第(i)个数组有(x)(a_i),(y)(b_i),(z)(c_i).这样,每个数组的大小都是(x+y+z).

    你希望从每个数组里选择1个整数,使得它们的xor(按位异或)值恰好为(t). 对于区间([0,2^k-1])内的每个数字(t),输出满足上面的条件的方案数 模 (998244353)

    输入:

    第一行读入(n,k),第二行读入(x,y,z),接下来读入(n)行。第(i)行读入(a_i),(b_i),(c_i)

    输出:

    一行(2^k)个整数,用空格分隔。第(i)个数字是从每个数组里选一个整数使它们的xor值等于(t=i-1)的方案数模(998244353)

    (1 leq n leq 10^{5},1 leq k leq 17,0 leq x,y,z leq 10^{9},0 leq a_{i} , b_{i} , c_{i} leq 2^{k} - 1)


    考虑暴力做法,整出来 (n) 个多项式 ,(F_i[a_i]=x,F_i[b_i]=y,F_i[c_i]=z) ,然后 FWT 过去再 IFWT 回来就能算出来答案,这样子复杂度是 (O(kn2^k))

    然后优化,写出来第 (t) 项的式子:

    [F_t'[i]=sum_{j=1}^nc(i,j)F_t[j]=sum_{j=1}^nc(i,a_t)x+c(i,b_t)y+c(i,c_t)z ]

    因为异或卷积的 (c(i,j)=(-1)^{num(i&j)})(num) 是二进制下 (1) 的个数.

    于是就有:

    [F_t'[i]=sum_{j=1}^n(-1)^{num(i&a_t)}x+(-1)^{num(i&b_t)}y+(-1)^{num(i&c_t)}z ]

    这样子 (x+y+z) 的情况有 (8) 种,考虑减少情况,对每一个三元组都异或上 (a_i) ,那么三元组就变成了 ({0,b_ioplus a_i,c_ioplus a_i})

    这样子 ((-1)^{num(i&0)}=1) ,那么情况就变成了四种:

    [x+y+z,x+y-z,x-y+z,x-y-z ]

    考虑分别对这个系数贡献了 (s_1,s_2,s_3,s_4) 次,那么我们一定会有这个方程:

    [s_1+s_2+s_3+s_4=n ]

    然后考虑赋 (x=0,y=1,z=0) ,那么 (F'_t[i]=sum_{j=1}^n(-1)^{num(i&b_t)}) ,只需要看 (y) 的正负号,相当于数 (s_1+s_2-s_3-s_4) ,设 (=B)

    同理赋 (x=0,y=0,z=1) ,那么就有 (s_1-s_2+s_3-s_4=C)

    然后把这两个多项式卷起来,就相当于对 (F[b_ioplus c_i]) FWT ,这样就看 (y)(z) 的正负号,得到 (s_1-s_2-s_3+s_4=BC)

    于是就可以解方程了,最后可以解得:

    [egin{cases}s_1=frac{n+B+C+BC}{4}\s_2=frac{n+B-C-BC}{4}\s_3=frac{n-B+C-BC}{4}\s_4=frac{n-B-C+BC}{4}end{cases} ]

    然后算出来 (F') IFWT 回去就可以了。

    Code

    #include <iostream>
    #include <cstdio>
    #include <algorithm>
    #include <cstring>
    const int N = 1e5;
    const int M = 17;
    const int p = 998244353;
    using namespace std;
    int n,k,x,y,z,a[N + 5],b[N + 5],c[N + 5],A[1 << M],B[1 << M],C[1 << M],D[1 << M],sa;
    int mypow(int a,int x){int s = 1;for (;x;x & 1 ? s = 1ll * s * a % p : 0,a = 1ll * a * a % p,x >>= 1);return s;}
    void FWT(int *a)
    {
        for (int i = 1;i < (1 << k);i <<= 1)
            for (int j = 0;j < (1 << k);j += i << 1)
                for (int k = 0;k < i;k++)
                {
                    int a0 = a[j + k],a1 = a[j + k + i];
                    a[j + k] = (a0 - a1) % p;
                    a[j + k + i] = (a0 + a1) % p;
                }
    }
    void IFWT(int *a)
    {
        int inv = mypow(2,p - 2);
        for (int i = 1;i < (1 << k);i <<= 1)
            for (int j = 0;j < (1 << k);j += i << 1)
                for (int k = 0;k < i;k++)
                {
                    int a0 = 1ll * a[j + k] * inv % p,a1 = 1ll * a[j + k + i] * inv % p;
                    a[j + k] = (a0 + a1) % p;
                    a[j + k + i] = (-a0 + a1) % p;
                }
    }
    int main()
    {
        scanf("%d%d",&n,&k);
        scanf("%d%d%d",&x,&y,&z);
        for (int i = 1;i <= n;i++)
        {
            scanf("%d%d%d",&a[i],&b[i],&c[i]);
            b[i] ^= a[i];
            c[i] ^= a[i];
            sa ^= a[i];
        }
        for (int i = 1;i <= n;i++)
            B[b[i]]++,C[c[i]]++,D[b[i] ^ c[i]]++;
        FWT(B);FWT(C);FWT(D);
        int inv = mypow(4,p - 2);
        int x1 = (0ll + x + y + z) % p,x2 = (0ll + x + y - z) % p,x3 = (0ll + x - y + z) % p,x4 = (0ll + x - y - z) % p;
        for (int i = 0;i < (1 << k);i++)
        {
            int S1 = ((0ll + n + B[i] + C[i] + D[i]) % p * inv % p + p) % p;
            int S2 = ((0ll + n + B[i] - C[i] - D[i]) % p * inv % p + p) % p;
            int S3 = ((0ll + n - B[i] + C[i] - D[i]) % p * inv % p + p) % p;
            int S4 = ((0ll + n - B[i] - C[i] + D[i]) % p * inv % p + p) % p;
            A[i] = 1ll * mypow(x1,S1) * mypow(x2,S2) % p * mypow(x3,S3) % p * mypow(x4,S4) % p; 
        }
        IFWT(A);
        for (int i = 0;i < (1 << k);i++)
            printf("%d ",(A[i ^ sa] + p) % p);
        return 0;
    }
    
  • 相关阅读:
    js ++i和i++的区别
    js斐波那契数列
    js二分查找算法
    js查找、自组织数据
    查找数组最小值、最大值
    CSS布局(圣杯、双飞翼、flex)
    碰撞检测实现
    ECharts注释
    购物查看放大
    动手封装一个滚轮事件吧!
  • 原文地址:https://www.cnblogs.com/sdlang/p/14295138.html
Copyright © 2011-2022 走看看