Codeforces 1119H - Triple(FWT)
FWT 的 immortal tea %%%
首先我们可以写出一个朴素的 \(dp\),设 \(dp_{i,j}\) 表示考虑前 \(i\) 个三元组中取数异或和为 \(j\) 的方案数,那么显然有 \(dp_{i,j}=Xdp_{i-1,j\oplus a_i}+Ydp_{i-1,j\oplus b_i}+Zdp_{i-1,j\oplus c_i}\),这样暴力复杂度是 \(n2^k\) 的,无法通过。
考虑优化,不难发现这个 \(dp\) 是一个异或卷积的形式,下记 \(\times\) 为幂级数之间的异或卷积操作,考虑幂级数 \(f_i\) 为满足 \([x^{a_i}]f_i=X,[x^{b_i}]f_i=Y,[x^{c_i}]=Z\),其余各项系数均为 \(0\) 的幂级数,那么不难发现 \(dp_{i-1}\times f_i=dp_i\) 成立,但是如果硬要 FWT 优化,把所有幂级数都搞一遍 FWTxor 然后乘起来,复杂度会高达 \(nk2^k\),反而跑不过暴力。
那么我们这个想法有没有前途呢?不难发现我们这个幂级数很特殊,只有三项有值,其余项都是 \(0\),并且这三项的值还是对于所有幂级数都相同的 \(X,Y,Z\),考虑以此为突破口。记 \(\hat{A}\) 为幂级数 \(A\) 的 FWTxor,我们不妨来对于某个 \(i\in[1,n]\),\(\hat{f_i}\) 有什么性质,显然根据 FWTxor 的求法 \([x^j]\hat{f_i}=(-1)^{|j\cap a_i|}X+(-1)^{|j\cap b_i|}Y+(-1)^{|j\cap c_i|}Z\),也就是说 \(\hat{f_i}\) 的每一项总共有 \(2^3=8\) 种可能,这样分类讨论还是有点烦,这里有一个小小的减少分类讨论的技巧,考虑记 \(S=\operatorname{xor}\limits_{i=1}^na_i\),我们令 \(b_i\leftarrow b_i\oplus a_i,c_i\leftarrow c_i\oplus a_i,a_i\leftarrow 0\),不难发现经过这个转化之后,真正的 \(ans_i=ans_{i\oplus S}\),不过这样 \(X\) 前的系数 \((-1)^{|j\cap a_i|}\) 就恒为 \(1\) 了,这样就只有 \(4\) 种情况了,列一下分别是 \(X+Y+Z,X+Y-Z,X-Y+Z,X-Y-Z\),如果我们知道这四种情况分别出现的次数就可以快速幂求出 \(\widehat{f_1\times f_2\times\cdots\times f_n}\) 前的系数了,再 IFWTxor 一遍即可求出 \(ans\),具体来说我们记 \(p_j\) 为有多少个 \(i\) 满足 \([x^j]\hat{f_i}=X+Y+Z\),也类似地定义了 \(q_i\) 为 \(X+Y-Z\) 的个数,\(r_i\) 为 \(X-Y+Z\) 的个数,\(s_i\) 为 \(X-Y-Z\) 的个数,那么 \([x^j]\widehat{f_1\times f_2\times\cdots\times f_n}=(X+Y+Z)^{p_i}\times(X+Y-Z)^{q_i}\times(X-Y+Z)^{r_i}\times(X-Y-Z)^{s_i}\)。
于是现在问题就变为怎样求 \(p_j,q_j,r_j,s_j\),首先有一个显然的方程 \(p_j+q_j+r_j+s_j=n\),因为每个 \([x^j]\hat{f_i}\) 都是这四种情况之一,不过只有这一个方程是显然不够的,考虑记 \(t_j=\sum\limits_{i=1}^n(-1)^{|j\cap b_i|}\),那么显然 \(t_j=p_j+q_j-r_j-s_j\),因为每个 \(X+Y+Z,X+Y-Z\) 都会对其产生 \(+1\) 的贡献,而每个 \(X-Y+Z,X-Y-Z\) 都会对其产生 \(-1\) 的贡献,另一方面,如果我们记 \(c1_j\) 为 \(\sum\limits_{i=1}^n[j=b_i]\),那么 \(t_j=\sum\limits_{k}c1_k(-1)^{|j\cap k|}\),这应该很好理解,然后对 FWT 稍微敏感一些的同学就能发现,这不就 \(\hat{c1}\) 吗,故我们有了第二个方程 \(p_j+q_j-r_j-s_j=\hat{c1}_j\),类似地,如果我们记 \(c2_j\) 为 \(\sum\limits_{i=1}^n[j=c_i]\),\(c3_j\) 为 \(\sum\limits_{i=1}^n[j=b_i\oplus c_i]\),那么 \(p_j-q_j+r_j-s_j=\hat{c2}_j,p_j-q_j-r_j+s_j=\hat{c3}_j\),推导过程类似,这里就不再赘述了,因此我们有:
p_j+q_j+r_j+s_j=n\\
p_j+q_j-r_j-s_j=\hat{c1}_j\\
p_j-q_j+r_j-s_j=\hat{c2}_j\\
p_j-q_j-r_j+s_j=\hat{c3}_j
\end{cases}
\]
四个未知数四个方程,把它解出来即可,\(\hat{c1},\hat{c2},\hat{c3}\) 显然可以在 \(k2^k\) 的时间内求出,因此总复杂度 \(n+k2^k\),可以通过此题。
const int MAXN=1<<17;
const int MOD=998244353;
const int INV2=499122177;
int n,k,x,y,z,ans[MAXN+5],c1[MAXN+5],c2[MAXN+5],c3[MAXN+5];
int qpow(int x,int e){
int ret=1;
for(;e;e>>=1,x=1ll*x*x%MOD) if(e&1) ret=1ll*ret*x%MOD;
return ret;
}
void FWTxor(int *a,int len,int type){
for(int i=2;i<=len;i<<=1)
for(int j=0;j<len;j+=i)
for(int k=0;k<(i>>1);k++){
int X=a[j+k],Y=a[(i>>1)+j+k];
if(type==1) a[j+k]=X+Y,a[(i>>1)+j+k]=X-Y;
else a[j+k]=1ll*(X+Y)*INV2%MOD,a[(i>>1)+j+k]=1ll*(X-Y+MOD)*INV2%MOD;
}
}
int main(){
scanf("%d%d",&n,&k);int xsum=0;
scanf("%d%d%d",&x,&y,&z);
for(int i=1,a,b,c;i<=n;i++){
scanf("%d%d%d",&a,&b,&c);b^=a;c^=a;xsum^=a;
c1[b]++;c2[c]++;c3[b^c]++;
}
FWTxor(c1,1<<k,1);FWTxor(c2,1<<k,1);FWTxor(c3,1<<k,1);
for(int i=0;i<(1<<k);i++){
int w1=(n+c1[i]+c2[i]+c3[i])/4;
int w2=(n+c1[i]-c2[i]-c3[i])/4;
int w3=(n-c1[i]+c2[i]-c3[i])/4;
int w4=(n-c1[i]-c2[i]+c3[i])/4;
// printf("%d %d %d\n",c1[i],c2[i],c3[i]);
// printf("%d %d %d %d\n",w1,w2,w3,w4);
ans[i]=1ll*qpow((x+y)%MOD+z,w1)*qpow(x+y-z,w2)%MOD*
qpow(x-y+z,w3)%MOD*qpow(x-y-z,w4)%MOD;
if(ans[i]<0) ans[i]+=MOD;
} FWTxor(ans,1<<k,-1);
for(int i=0;i<(1<<k);i++) printf("%d ",ans[i^xsum]);
return 0;
}
Codeforces 1119H - Triple(FWT)的更多相关文章
- 【CF850E】Random Elections(FWT)
[CF850E]Random Elections(FWT) 题面 洛谷 CF 题解 看懂题就是一眼题了... 显然三个人是等价的,所以只需要考虑一个人赢了另外两个人就好了. 那么在赢另外两个人的过程中 ...
- 【CF662C】Binary Table(FWT)
[CF662C]Binary Table(FWT) 题面 洛谷 CF 翻译: 有一个\(n*m\)的表格(\(n<=20,m<=10^5\)), 每个表格里面有一个\(0/1\), 每次可 ...
- 「WC2018」州区划分(FWT)
「WC2018」州区划分(FWT) 我去弄了一个升级版的博客主题,比以前好看多了.感谢 @Wider 不过我有阅读模式的话不知为何 \(\text{LATEX}\) 不能用,所以我就把这个功能删掉了. ...
- 【HDU5909】Tree Cutting(FWT)
[HDU5909]Tree Cutting(FWT) 题面 vjudge 题目大意: 给你一棵\(n\)个节点的树,每个节点都有一个小于\(m\)的权值 定义一棵子树的权值为所有节点的异或和,问权值为 ...
- 【UOJ#310】【UNR#2】黎明前的巧克力(FWT)
[UOJ#310][UNR#2]黎明前的巧克力(FWT) 题面 UOJ 题解 把问题转化一下,变成有多少个异或和为\(0\)的集合,然后这个集合任意拆分就是答案,所以对于一个大小为\(s\)的集合,其 ...
- Codeforces663E Binary Table(FWT)
题目 Source http://codeforces.com/contest/663/problem/E Description You are given a table consisting o ...
- CodeForces - 615D Multipliers(数论)
http://codeforces.com/problemset/problem/615/D 题意 给出m个质因子,组成一个数n.问n的约数的乘积是多少,输出mod 1e+7的结果. 分析 从输入我们 ...
- 关于快速沃尔什变换(FWT)的一些个人理解
定义 FWT是一种快速完成集合卷积运算的算法. 它可以用于求解类似 $C[i]=\sum\limits_{j⊗k=i}A[j]*B[k]$ 的问题. 其中⊗代表位运算中的|,&,^的其中一种. ...
- Codeforces Round #328(Div2)
CodeForces 592A 题意:在8*8棋盘里,有黑白棋,F1选手(W棋往上-->最后至目标点:第1行)先走,F2选手(B棋往下-->最后至目标点:第8行)其次.棋子数不一定相等,F ...
随机推荐
- Java中的函数式编程(二)函数式接口Functional Interface
写在前面 前面说过,判断一门语言是否支持函数式编程,一个重要的判断标准就是:它是否将函数看做是"第一等公民(first-class citizens)".函数是"第一等公 ...
- Java项目中常用的的五大设计原则
今天我们一起来聊聊关于设计原则相关的知识点. SOLID五大原则是什么 SRP 单一责任原则 单一责任原则,从名字上我们就能比较好的去理解它.这项原则主张一个对象只专注于单个方面的逻辑,强调了职责的专 ...
- Java多线程中的死锁
Java多线程中的死锁 死锁产生的原因 线程死锁是指由两个以上的线程互相持有对方所需要的资源,导致线程处于等待状态,无法往前执行. 当线程进入对象的synchronized代码块时,便占有了资源,直到 ...
- CODING 助力江苏高速信息实现组织敏捷与研发敏捷,领跑智慧交通新基建
疫情之下的高速公路管控重任 江苏高速公路信息工程有限公司(以下简称:江苏高速信息)成立于 2002 年,是江苏交通控股旗下,专业从事高速公路领域机电系统集成.智能交通软硬件研发.大数据分析运营的高新技 ...
- Noip模拟81 2021.10.20
T1 语言 比较简单的题,然后就瞎写了,所以考场上就我一个写了线段树的,所以我的常数.... 所以就枚举动词的位置,找前面后面有没有出现$4$即可 1 #include<bits/stdc++. ...
- 认真讲说static关键字
static 关键字主要有以下四种使用场景 修饰成员变量和成员方法 静态代码块 修饰类(只能修饰内部类) 静态导包(用来导入类中的静态资源,1.5之后的新特性) 修饰成员变量和成员方法(常用) 被 s ...
- cf12D Ball(MAP,排序,贪心思想)
题意: N位女士一起聚在一个舞厅.每位女士有三个特征值B,I,R.分别代表美貌,智慧,富有. 对于一位女士而言,如果存在一个女士的B,I,R都分别大于她自己的B,I,R.则她自己会自杀. 统计总共有多 ...
- 第02课 OpenGL 多边形
你的第一个多边形: 在第一个教程的基础上,我们添加了一个三角形和一个四边形.也许你认为这很简单,但你已经迈出了一大步,要知道任何在OpenGL中绘制的模型都会被分解为这两种简单的图形.读完了这一课,你 ...
- clnt_create: RPC: Port mapper failure - Unable to receive: errno 113 (No route to host)
修改文件 /etc/sysconfig/nfs将#MOUNTD_PORT=892开启防火墙端口:firewalld-cmd --add-port=892/tcp
- 修改 openssh 版本号
1.查看 sshd 位置 #which sshd 2.查看 /usr/sbin/sshd(二进制文件) 内容 #strings /usr/sbin/sshd | grep nicai 3.修改版本号, ...