题解Luogu P6102 谔运算

Posted cjtcalc

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了题解Luogu P6102 谔运算相关的知识,希望对你有一定的参考价值。

[ exttt{Description} ]

  • 给出一个长度为 (n) 的数列 (a),求 (sumlimits_{i=1}limits^{n}sumlimits_{j=1}limits^{n}sumlimits_{k=1}limits^{n}sumlimits_{l=1}limits^{n}(a_i ext{or} a_j) ext{xor} (a_k ext{and} a_l))

[ exttt{Solution} ]

  • 先考虑下普通的谔运算 ((a ext{or} b) ext{xor} (c ext{and} d)) 什么时候为真((a,b,c,d in { 0,1 })),我们发现 (a,b,c,d) 一共有 (16) 种取值,其中有 (10) 种取值使得式子为真:
  • (a = 0, b = 0, c = 1, d = 1)
    (a = 0, b = 1, c = 0, d = 0)
    (a = 0, b = 1, c = 0, d = 1)
    (a = 0, b = 1, c = 1, d = 0)
    (a = 1, b = 0, c = 0, d = 0)
    (a = 1, b = 0, c = 0, d = 1)
    (a = 1, b = 0, c = 1, d = 0)
    (a = 1, b = 1, c = 0, d = 0)
    (a = 1, b = 1, c = 0, d = 1)
    (a = 1, b = 1, c = 1, d = 0)

  • 然后考虑按位分组计算贡献。
  • 详细地说:((a ext{or} b) ext{xor} (c ext{and} d)) 得到的数,若第 (i) 位为真,则对答案有 (2^i) 的贡献。由于谔运算是按位处理的,也就是说我们可以计算出对答案有贡献的数中,有多少个数第 (i) 位为真,若将这个量记为 (c_i) ,最后答案即为 (sumlimits_{i=0}limits^{31}c_i imes 2^i)
  • 我们记 (cnt[i,j]) 表示 (a) 中有多少数第 (i) 位为 (j) ,可以 ( ext{O(32n)}) 预处理。
  • 然后按位分组计算贡献,根据乘法原理和加法原理,从 (a) 中选出 (4) 个数进行谔运算,第 (i) 位为真的四元组有 (cnt[i,0] imes cnt[i,0] imes cnt[i,1] imes cnt[i,1] + cnt[i,0] imes cnt[i,1] imes cnt[i,0] imes cnt[i,0] + ......) ,最后将其乘上 (2^i) 计入答案中。
  • 注意到此题的膜数为 (2^{32}) ,所以 技术图片 ,记得要开 ( ext{unsigned} ext{int})
  • 时间复杂度 ( ext{O(32n)})

[ exttt{Code} ]

#include<cstdio>
#include<iostream>

#define RI register int

using namespace std;

namespace IO
{
    static char buf[1<<20],*fs,*ft;
    inline char gc()
    {
        if(fs==ft)
        {
            ft=(fs=buf)+fread(buf,1,1<<20,stdin);
            if(fs==ft)return EOF;
        }
        return *fs++;
    }
    #define gc() getchar()
    inline int read()
    {
        unsigned int x=0,f=1;char s=gc();
        while(s<'0'||s>'9')s=gc();
        while(s>='0'&&s<='9')x=x*10+s-'0',s=gc();
        return x*f;
    }
}using IO::read;

const int N=500100;

int n;

int a[N];

unsigned int cnt[33][2];

unsigned int ans;

int main()
{
    n=read();

    for(RI i=1;i<=n;i++)
        a[i]=read();

    for(RI i=1;i<=n;i++)
        for(RI j=0;j<32;j++)
            cnt[j][(a[i]>>j)&1]++;

    for(RI i=0;i<32;i++)
    {
        unsigned int c=0;

        c+=cnt[i][0]*cnt[i][0]*cnt[i][1]*cnt[i][1];
        c+=cnt[i][0]*cnt[i][1]*cnt[i][0]*cnt[i][0];
        c+=cnt[i][0]*cnt[i][1]*cnt[i][0]*cnt[i][1];
        c+=cnt[i][0]*cnt[i][1]*cnt[i][1]*cnt[i][0];
        c+=cnt[i][1]*cnt[i][0]*cnt[i][0]*cnt[i][0];
        c+=cnt[i][1]*cnt[i][0]*cnt[i][0]*cnt[i][1];
        c+=cnt[i][1]*cnt[i][0]*cnt[i][1]*cnt[i][0];
        c+=cnt[i][1]*cnt[i][1]*cnt[i][0]*cnt[i][0];
        c+=cnt[i][1]*cnt[i][1]*cnt[i][0]*cnt[i][1];
        c+=cnt[i][1]*cnt[i][1]*cnt[i][1]*cnt[i][0];

        ans+=c*(1<<i);
    }

    printf("%u
",ans);

    return 0;
}

[ exttt{Thanks} exttt{for} exttt{watching} ]

以上是关于题解Luogu P6102 谔运算的主要内容,如果未能解决你的问题,请参考以下文章

(luogu题解搬运系列)luogu p2651 添加括号Ⅲ

洛谷 P1226 取余运算||快速幂 题解

洛谷 P1449 后缀表达式 题解

题解:琪露诺的冰雪小屋luogu3693

题解 外星千足虫(线性基+高斯消元)

Luogu P1351 联合权值 题解