题目:http://codeforces.com/contest/914/problem/G

其实就是把各种都用子集卷积和FWT卷起来算即可;

注意乘 Fibonacci 数组的位置;

子集卷积时不能一边做一边更新卷积的数组!

代码如下:

#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
typedef long long ll;
int rd()
{
int ret=,f=; char ch=getchar();
while(ch<''||ch>''){if(ch=='-')f=; ch=getchar();}
while(ch>=''&&ch<='')ret=ret*+ch-'',ch=getchar();
return f?ret:-ret;
}
int Max(int x,int y){return x>y?x:y;}
int const xm=(<<)+,mod=1e9+;
int n,p[xm],F[xm],f[][xm],g[xm],h[xm],t[xm],bin[],cnt[xm],inv2,lm;
ll pw(ll a,int b){ll ret=; for(;b;b>>=,a=a*a%mod)if(b&)ret=ret*a%mod; return ret;}
int upt(int x){while(x>=mod)x-=mod; while(x<)x+=mod; return x;}
int cal(int s){int ret=; while(s)ret+=(s&),s>>=; return ret;}
int get(int x){int ret=; while(x)ret++,x>>=; return ret;}
void init()
{
F[]=; F[]=;
for(int i=;i<bin[lm];i++)F[i]=upt(F[i-]+F[i-]);
}
void fmt(int *a,int tp)
{
for(int d=;d<bin[lm];d<<=)
for(int s=;s<bin[lm];s++)
if(s&d)a[s]=upt(a[s]+a[s^d]*tp);
}
void fwt1(int *a,int tp)//&
{
for(int mid=;mid<bin[lm];mid<<=)
for(int j=,len=(mid<<);j<bin[lm];j+=len)
for(int k=;k<mid;k++)
a[j+k]=upt(a[j+k]+a[j+mid+k]*tp);
}
void fwt2(int *a,int tp)//^
{
for(int mid=;mid<bin[lm];mid<<=)
for(int j=,len=(mid<<);j<bin[lm];j+=len)
for(int k=;k<mid;k++)
{
int x=a[j+k],y=a[j+mid+k];
a[j+k]=upt(x+y); a[j+mid+k]=upt(x-y);
if(tp==-)a[j+k]=(ll)a[j+k]*inv2%mod,a[j+mid+k]=(ll)a[j+mid+k]*inv2%mod;
}
}
int main()
{
n=rd(); lm=;
bin[]=; for(int i=;i<=lm;i++)bin[i]=bin[i-]*;
for(int s=;s<bin[lm];s++)cnt[s]=cal(s);
int mx=;
for(int i=,x;i<=n;i++)x=rd(),p[x]++,mx=Max(mx,x);
lm=get(mx)+; init();
for(int s=;s<bin[lm];s++)f[cnt[s]][s]=p[s];
for(int i=;i<lm;i++)fmt(f[i],);
/*
for(int i=1;i<lm;i++)
{
for(int j=0;j<=i;j++)
for(int s=0;s<bin[lm];s++)
f[i][s]=(f[i][s]+(ll)f[j][s]*f[i-j][s])%mod;
for(int s=0;s<bin[lm];s++)if(cnt[s]!=i)f[i][s]=0;
}
for(int i=0;i<lm;i++)fmt(f[i],-1);
for(int s=0;s<bin[lm];s++)g[s]=f[cnt[s]][s];
*/
for(int i=;i<lm;i++)
{
memset(t,,sizeof t);
for(int j=;j<=i;j++)
for(int s=;s<bin[lm];s++)
t[s]=(t[s]+(ll)f[j][s]*f[i-j][s])%mod;
fmt(t,-);
for(int s=;s<bin[lm];s++)
if(cnt[s]==i)g[s]=upt(g[s]+t[s]);
} for(int s=;s<bin[lm];s++)h[s]=p[s];
inv2=pw(,mod-); fwt2(h,);
for(int s=;s<bin[lm];s++)h[s]=(ll)h[s]*h[s]%mod;
fwt2(h,-); for(int s=;s<bin[lm];s++)g[s]=(ll)g[s]*F[s]%mod;
for(int s=;s<bin[lm];s++)h[s]=(ll)h[s]*F[s]%mod;
for(int s=;s<bin[lm];s++)p[s]=(ll)p[s]*F[s]%mod;
fwt1(g,); fwt1(h,); fwt1(p,);
for(int s=;s<bin[lm];s++)g[s]=(ll)g[s]*h[s]%mod*p[s]%mod;
fwt1(g,-); int ans=;
for(int i=;i<lm;i++)ans=upt(ans+g[bin[i]]);
printf("%d\n",ans);
return ;
}

CF 914 G Sum the Fibonacci —— 子集卷积,FWT的更多相关文章

  1. CF 914G Sum the Fibonacci——子集卷积

    题目:http://codeforces.com/contest/914/problem/G 第一个括号可以子集卷积:第三个括号可以用 FWT 异或卷积:这样算出选两个数组成 x 的方案数:三个部分的 ...

  2. Codecraft-18 and Codeforces Round #458 (Div. 1 + Div. 2, combined)G. Sum the Fibonacci

    题意:给一个数组s,求\(f(s_a | s_b) * f(s_c) * f(s_d \oplus s_e)\),f是斐波那契数列,而且要满足\(s_a\&s_b==0\),\((s_a | ...

  3. @总结 - 2@ 位运算卷积/子集卷积 —— FWT/FMT

    目录 @0 - 参考资料@ @1 - 异或卷积概念及性质@ @2 - 快速沃尔什正变换(异或)@ @3 - 快速沃尔什逆变换(异或)@ @4 - 与卷积.或卷积@ @5 - 参考代码实现@ @6 - ...

  4. CF914G Sum the Fibonacci (快速沃尔什变换FWT + 子集卷积)

    题面 题解 这是一道FWT和子集卷积的应用题. 我们先设 cnt[x] 表示 Si = x 的 i 的数量,那么 这里的Nab[x]指满足条件的 Sa|Sb=x.Sa&Sb=0 的(a,b)二 ...

  5. CF914G Sum the Fibonacci FWT、子集卷积

    传送门 一道良心的练习FWT和子集卷积的板子-- 具体来说就是先把所有满足\(s_a \& s_b = 0\)的\(s_a \mid s_b\)的值用子集卷积算出来,将所有\(s_a \opl ...

  6. 【codeforces914G】Sum the Fibonacci FWT+FST(快速子集变换)

    题目描述 给出一个长度为 $n$ 的序列 $\{s\}$ ,对于所有满足以下条件的五元组 $(a,b,c,d,e)$ : $1\le a,b,c,d,e\le n$ : $(s_a|s_b)\& ...

  7. 【CF914G】Sum the Fibonacci 快速??变换模板

    [CF914G]Sum the Fibonacci 题解:给你一个长度为n的数组s.定义五元组(a,b,c,d,e)是合法的当且仅当: 1. $1\le a,b,c,d,e\le n$2. $(s_a ...

  8. UOJ #348 州区划分 —— 状压DP+子集卷积

    题目:http://uoj.ac/problem/348 一开始可以 3^n 子集DP,枚举一种状态的最后一个集合是什么来转移: 设 \( f[s] \) 表示 \( s \) 集合内的点都划分好了, ...

  9. Future Failure CodeForces - 838C (博弈论,子集卷积)

    大意: 两人轮流操作一个长$n$, 只含前$k$种小写字母的串, 每次操作删除一个字符或者将整个串重排, 每次操作后得到的串不能和之前出现过的串相同, 求多少种串能使先手必胜. 找下规律发现$n$为奇 ...

随机推荐

  1. 多语言中的“默认语言”设置

    最近在搞一个多语言的东西,打算如果用户是中文环境就显示中文,其他任何非中文环境就显示英文.换句话说,把默认语言设置成英文. 不过因为VS是中文的,发现即使默认资源文件是英文(AppResource.r ...

  2. Residual (numerical analysis)

    In many cases, the smallness of the residual means that the approximation is close to the solution, ...

  3. spring AOP理解和相关术语

    一.AOP理解 AOP:横向抽取机制,底层使用代理方式实现. 示例: 现有LogDAO接口以及实现Log接口的Log类.类有add的方法,现在要打印add方法的开始时间和结束时间.(即增强Log的ad ...

  4. 如何从统计中批量获取BD搜索关键词及对应的入口页面?

    前面我们介绍了通过cnzz的访问明细获取到搜索关键词及对应的入口页面,但是从BD搜索进来的关键词无法完整显示,只能呈现一些bd图片搜索的关键词,这是因为百度宣布从去年5月开始逐渐取消了referer关 ...

  5. linux下查找最耗iowait的进程

    抓哪个进程干坏事前要先停掉syslogservice syslog stop 打开block dump:echo 1 > /proc/sys/vm/block_dump 统计:dmesg | e ...

  6. Redis持久化——RDB(一)

    核心知识点: 1.RDB:将当前数据生成快照保存到硬盘 2.手动触发 save:会阻塞Redis服务器直到RDB完成. bgsave:执行fork创建子进程,由子进程负责RDB操作,阻塞只发生在for ...

  7. 《C+编程规范 101条规则、准则与最佳实践》笔记

    <C+编程规范 101条规则.准则与最佳实践> 0.不要拘泥于小节(了解哪些东西不应该标准化) * 与组织内现有编码规范一致即可 * 包括但不限于: - 缩进 - 行长度 - 命名规范 - ...

  8. 事件监听机制——列出指定目录内容、添加Dialog对话框

    事件监听机制理解与Dialog练习 利用Java语言,仿照我的电脑目录进行打开目录,输入文件路径,查看该路径下所有的文件,设置两个文本框,一个转到按钮,当点击转到按钮时,查看路径是否正确,若正确在第二 ...

  9. Java多线程系列 JUC线程池07 线程池原理解析(六)

     关闭“线程池” shutdown()的源码如下: public void shutdown() { final ReentrantLock mainLock = this.mainLock; // ...

  10. 小学生都能看懂的数位dp

    前言 数位dp其实很久前就知道了,也做过几道和其他算法混在一起的题目,其实通过手玩是能做的 但毕竟是种算法,还是系统学下比较好(节省手玩时间) 模板题 P2602 [ZJOI2010]数字计数 化简题 ...