/*
HDU 6057 - Kanade's convolution [ FWT ] | 2017 Multi-University Training Contest 3
题意:
给定两个序列 A[0...2^m-1], B[0...2^m-1]
求 C[0...2^m-1] ,满足:
C[k] = ∑[i&j==k] A[i^j] * B[i|j]
m <= 19
分析:
看C[k]的形式与集合卷积的形式接近,故转化式子时主要向普通的集合卷积式方向靠
与三种位运算都相关的结论是 : i^j + i&j = i|j
设 x = i^j, y = i|j,则显然 k = y-x,且 k 与 x 互成关于 y 的补集,即 k = x^y 再来关心给定(x,y),符合 x = i^j, y = i|j的(i,j)对的数目
注意到相同的位 i&j 是确定的,x = i^j 是i和j不同的位的数目,这部分谁是 0 谁是 1 不固定
故(i,j)对的数目为 2^bits(x) 此时重写原式: C[k] = ∑ [k == x^y] [k == y-x] A[x]*2^bits(x) * B[y] 设 A'[x] = A[x]*2^bits(x)
由于 [k == x^y],第二个条件 [k == y-x] 等价于 bits(k) == bits(y) - bits(x)
C[k] = ∑ [k == x^y] [bits(k) == bits(x) - bits(y)] A'[x] * B[y] 将 A,B,C三个数组按 bits 划分:
C[bits(k)][k] = ∑ [k == x^y] A[bits(x)][x]*2^bits(x) * B[bits(y)][y] 最后按不同的维度(bits)做 FWT即可
*/
#include <bits/stdc++.h>
using namespace std;
const int MOD = 998244353;
const int N = 1<<20;
int rev2;
long long inv( long long a , long long m)
{
if (a == 1) return 1;
return inv(m%a, m) * (m - m/a) % m;
}
void FWT(int a[], int n) {
for (int d = 1; d < n; d <<= 1)
for (int m = d<<1, i = 0; i < n; i += m)
for (int j = 0; j < d; j++)
{
int x = a[i+j], y = a[i+j+d];
a[i+j] = (x+y) % MOD;
a[i+j+d] = (x-y+MOD) % MOD;
}
}
void UFWT(int a[], int n) {
for (int d = 1; d < n; d <<= 1)
for (int m = d<<1, i = 0; i < n; i += m)
for (int j = 0; j < d; j++)
{
int x = a[i+j], y = a[i+j+d];
a[i+j] = 1LL*(x+y) * rev2 % MOD;
a[i+j+d] = (1LL*(x-y)*rev2 % MOD + MOD) % MOD;
}
}
int a[20][N], b[20][N], c[20][N];
int bits[N];
int m, n;
void init()
{
rev2 = inv(2, MOD);
bits[0] = 0;
for (int i = 1; i < N; i++) bits[i] = bits[i>>1] + (i&1);
}
int main()
{
init();
scanf("%d", &m);
n = 1<<m;
for (int i = 0; i < n; i++)
{
int x; scanf("%d", &x);
a[bits[i]][i] = 1LL*x * (1<<bits[i]) % MOD;
}
for (int i = 0; i < n; i++)
{
int x; scanf("%d", &x);
b[bits[i]][i] = x;
}
for (int i = 0; i <= m; i++) FWT(a[i], n);
for (int i = 0; i <= m; i++) FWT(b[i], n);
for (int i = 0; i <= m; i++)
for (int j = i; j <= m; j++)
for (int k = 0; k < n; k++)
{
c[j-i][k] = (c[j-i][k] + 1LL*a[i][k] * b[j][k] % MOD) % MOD;
}
for (int i = 0; i <= m; i++) UFWT(c[i], n);
long long ans = 0, base = 1;
for (int i = 0; i < n; i++)
{
ans = ( ans + c[bits[i]][i] * base % MOD ) % MOD;
base = base * 1526 % MOD;
}
printf("%lld\n", ans);
}

  

HDU 6057 - Kanade's convolution | 2017 Multi-University Training Contest 3的更多相关文章

  1. HDU 6057 Kanade's convolution(FWT)

    [题目链接] http://acm.hdu.edu.cn/showproblem.php?pid=6057 [题目大意] 有 C[k]=∑_(i&j=k)A[i^j]*B[i|j] 求 Ans ...

  2. HDU 6057 Kanade's convolution

    题目链接:HDU-6057 题意: 思路:先按照官方题解推导出下面的式子: 现在唯一的问题就是怎么解决[bit(x)-bit(y)=bit(k)]的问题. 我们定义\( F(A,k)_{i}=\lef ...

  3. hdu 6057 Kanade's convolution(子集卷积)

    题解: 然后就是接下来如何fwt 也就是如何处理bit(x) - bit(y) = bit(k)这个条件. 其实就是子集卷积. 把bit(x)和bit(y)划分成两个集合,然后就是子集卷积的形式. 这 ...

  4. HDU 6059 - Kanade's trio | 2017 Multi-University Training Contest 3

    思路来自题解(看着题解和标程瞎吉尔比划了半天) /* HDU 6059 - Kanade's trio [ 字典树 ]  |  2017 Multi-University Training Conte ...

  5. HDU 6058 - Kanade's sum | 2017 Multi-University Training Contest 3

    /* HDU 6058 - Kanade's sum [ 思维,链表 ] | 2017 Multi-University Training Contest 3 题意: 给出排列 a[N],求所有区间的 ...

  6. HDU 6162 - Ch’s gift | 2017 ZJUT Multi-University Training 9

    /* HDU 6162 - Ch’s gift [ LCA,线段树 ] | 2017 ZJUT Multi-University Training 9 题意: N节点的树,Q组询问 每次询问s,t两节 ...

  7. HDU 6058 Kanade's sum —— 2017 Multi-University Training 3

    Kanade's sum Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)Tota ...

  8. 2017 Wuhan University Programming Contest (Online Round) Lost in WHU 矩阵快速幂 一个无向图,求从1出发到达n最多经过T条边的方法数,边可以重复经过,到达n之后不可以再离开。

    /** 题目:Lost in WHU 链接:https://oj.ejq.me/problem/26 题意:一个无向图,求从1出发到达n最多经过T条边的方法数,边可以重复经过,到达n之后不可以再离开. ...

  9. 2017 Wuhan University Programming Contest (Online Round) C. Divide by Six 分析+模拟

    /** 题目:C. Divide by Six 链接:https://oj.ejq.me/problem/24 题意:给定一个数,这个数位数达到1e5,可能存在前导0.问为了使这个数是6的倍数,且没有 ...

随机推荐

  1. HDU-6170 Two strings

    http://acm.hdu.edu.cn/showproblem.php?pid=6170 . 匹配任意字符,x* 匹配任意长度的 x (x 为任意字符,长度可以为 0 ) 正则表达式 #inclu ...

  2. Markdown Note

    Markdown-Note 1.基本编辑 粗体和斜体 源代码: *斜体*或_斜体_ **粗体** ***加粗斜体*** ~~删除线~~ 显示效果: 斜体或_斜体_ 粗体 加粗斜体 删除线 分级标题 第 ...

  3. jemeter鬓发压力测试包

    使用: 为子线程添加响应时间:https://www.cnblogs.com/duanxz/p/5464993.html 结果查看分析:聚合报告在监听器里面: https://wenku.baidu. ...

  4. c++11 原生字符串字面值

    c++11 原生字符串字面值 #define _CRT_SECURE_NO_WARNINGS #include <iostream> #include <string> #in ...

  5. Django rest-framework框架-用户权限实例

    简单实例: class MyPermission(object): ''' 权限控制类 ''' def has_permission(self,request,view): if request.us ...

  6. C#动态生成Word文档并填充数据

    C#也能动态生成Word文档并填充数据 http://www.cnblogs.com/qyfan82/archive/2007/09/14/893293.html 引用http://blog.csdn ...

  7. 【C++】如何提高Cache的命中率,示例

    参考链接     https://stackoverflow.com/questions/16699247/what-is-a-cache-friendly-code 只是堆积:缓存不友好与缓存友好代 ...

  8. Oracle使用基础

    1.Oracle的基本概念: 数据库:存储数据的数据库,Oracle一般只有一个全局数据库 XE,ORCL. EX:Express Edition 速成版 ORCL:企业版 SID:SID是Syste ...

  9. 12个提高Java程序员工作效率的工具

    Java开发者常常都会想办法如何更快地编写Java代码,让开发过程变得更加轻松,更加高效.目前,市面上涌现出越来越多的高效编程工具.团长总结了几个常用的工具,其中包含了大多数开发人员已经使用.正在使用 ...

  10. JavaMaven【八、pom.xml】

    简介: 重点学习: 1.dependency-scope 依赖范围 compile 编译 默认,对编译.测试.运行都有效 provided 编译和测试时有效 runtime 测试和运行时有效 test ...