题解【Luogu P6102 谔运算】
\]
- 给出一个长度为 \(n\) 的数列 \(a\),求 \(\sum\limits_{i=1}\limits^{n}\sum\limits_{j=1}\limits^{n}\sum\limits_{k=1}\limits^{n}\sum\limits_{l=1}\limits^{n}(a_i \ \text{or} \ a_j) \ \text{xor} \ (a_k \ \text{and} \ a_l)\) 。
\]
先考虑下普通的谔运算 \((a \ \text{or} \ b) \ \text{xor} \ (c \ \text{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 \ \text{or} \ b) \ \text{xor} \ (c \ \text{and} \ d)\) 得到的数,若第 \(i\) 位为真,则对答案有 \(2^i\) 的贡献。由于谔运算是按位处理的,也就是说我们可以计算出对答案有贡献的数中,有多少个数第 \(i\) 位为真,若将这个量记为 \(c_i\) ,最后答案即为 \(\sum\limits_{i=0}\limits^{31}c_i \times 2^i\) 。
我们记 \(cnt[i,j]\) 表示 \(a\) 中有多少数第 \(i\) 位为 \(j\) ,可以 \(\text{O(32n)}\) 预处理。
然后按位分组计算贡献,根据乘法原理和加法原理,从 \(a\) 中选出 \(4\) 个数进行谔运算,第 \(i\) 位为真的四元组有 \(cnt[i,0] \times cnt[i,0] \times cnt[i,1] \times cnt[i,1] + cnt[i,0] \times cnt[i,1] \times cnt[i,0] \times cnt[i,0] + ......\) ,最后将其乘上 \(2^i\) 计入答案中。
注意到此题的膜数为 \(2^{32}\) ,所以 ,记得要开 \(\text{unsigned} \ \text{int}\)。
时间复杂度 \(\text{O(32n)}\) 。
\]
#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\n",ans);
return 0;
}
\]
最新文章
- Rest webservice 和SOAP webservice
- 关于DOM的一些笔记(一)
- 配置 DHCP 服务 - 每天5分钟玩转 OpenStack(89)
- Bash 是如何从环境变量中导入函数的
- SQL分页查询,纯Top方式和row_number()解析函数的使用及区别
- iptables常用命令
- Amoeba For MySQL入门:实现数据库水平切分
- Android之AlarmManager(全局定时器/闹钟)指定时长或以周期形式执行某项操作
- [AngularJS] angular-formly: Extending Types
- const变量与define定义常量的区别
- YII 登陆时 session持久化
- android 获取屏幕尺寸
- 2017广东工业大学程序设计竞赛决赛 题解&;源码(A,数学解方程,B,贪心博弈,C,递归,D,水,E,贪心,面试题,F,贪心,枚举,LCA,G,dp,记忆化搜索,H,思维题)
- 【MyBatis源码分析】Configuration加载(上篇)
- MyDAL - .Where() &; .And() &; .Or() 使用
- 开发同学的福利--mysql监控工具sqlprofiler,类似sqlserver的profiler工具
- js中关于Blob对象的介绍与使用
- iOS 设计模式-NSNotificationCenter 通知中心
- WM_COMMAND和WM_NOTIFY区别[转]
- 专访|HPE软件部中国区总经理李时:HPE引领IT战略新形态
热门文章
- ZooKeeper Java Example
- EntityFramework Core表名原理解析,让我来,揭开你神秘的面纱
- Android 平台JS调试技术
- 一步一步教你PowerBI利用爬虫获取天气数据分析
- js-xlsx 一个实用的js 导出列表插件
- dubbo配置文件解读(1)
- Falco 进入 CNCF Incubator 项目 | 云原生生态周报 Vol. 35
- Git将一个项目同时从本地推送到GitHub和Gitee
- (转) fuzzing XSS filter
- Dynamics CRM 快速获取custom entity