【bzoj3771】【xsy1728】Triple
【bzoj3771】【xsy1728】
题意
求\(\sum_{i}[a_i=S]+\sum_{i<j}[a_i+a_j=S]+\sum_{i<j<k}[a_i+a_j+a_k=S]\)
\(n\leq 30000\),\(a_i\leq 40000\)
分析
容斥+FFT。
主要展现一些通过翻阅众多题解,得到的一些启示。
①把IDFT的除法放在内部。
void FFT(CP a[S],int len,int kd) {
//...
if (kd==-1) {
rep(i,0,len-1)
a[i].x/=len;
}
}
②变量声明的合理放置
int lenA,lenB,lenR;
int bit,len; int rev[S];
struct CP {
double x,y;
CP (double _x=0,double _y=0) {
x=_x,y=_y;
}
friend CP operator + (CP a,CP b) {
return CP(a.x+b.x,a.y+b.y);
}
friend CP operator - (CP a,CP b) {
return CP(a.x-b.x,a.y-b.y);
}
friend CP operator * (CP a,CP b) {
return CP(a.x*b.x-a.y*b.y,a.x*b.y+a.y*b.x);
}
}a[S],b[S],r[S],tmp[S];
③良好的封闭性Mut(a,lenA,b,lenB,r,lenR)
void Mut(CP a[S],int lenA,CP b[S],int lenB,CP r[S],int &lenR) {
fill(r,r+S,0);
lenR=lenA+lenB;
len=1,bit=0;
while (len<=lenA||len<=lenB)
len<<=1,bit++;
len<<=1,bit++;
memset(rev,0,sizeof rev);
rep(i,0,len-1) {
int x=0;
rep(j,0,bit-1)
if (i>>j&1)
x+=(1<<(bit-1-j));
rev[i]=x;
}
FFT(a,len,1);
FFT(b,len,1);
rep(i,0,len-1) r[i]=a[i]*b[i];
FFT(r,len,-1);
}
LL Trans(CP a) {
return (LL)(a.x+0.5);
}
注意要使用Fill而不能使用memset。
④生成函数对数组的初始化
由于大量使用了相同的生成函数,所以写作一个过程。
int cmp(double x) {
if (fabs(x)<EPS) return 0;
return x<0?-1:1;
}
void IA(CP *a,int &lenA) {
fill(a,a+S,0);
rep(i,1,n) a[w[i]].x+=1.0;
lenA=C1;
while (lenA>=0&&!cmp(a[lenA].x))
lenA--;
}
void IA2(CP a[S],int &lenA) {
fill(a,a+S,0);
rep(i,1,n) a[w[i]+w[i]].x+=1.0;
lenA=C2;
while (lenA>=0&&!cmp(a[lenA].x))
lenA--;
}
void IA3(CP a[S],int &lenA) {
fill(a,a+S,0);
rep(i,1,n) a[w[i]+w[i]+w[i]].x+=1.0;
lenA=C3;
while (lenA>=0&&!cmp(a[lenA].x))
lenA--;
}
⑤生成函数支持加法的合并
写了199行。
然后看了一下别人的代码。
woc77行......
发现生成函数支持加法的合并。
所以先用生成函数算出所有情况,然后再一次转换回来就好了。
代码
初次的AC代码:
#include <cstdio>
#include <cstring>
#include <cctype>
#include <cmath>
#include <algorithm>
using namespace std;
#define rep(i,a,b) for (int i=(a);i<=(b);i++)
typedef long long LL;
const int N=65536;
const int S=262144;
const double PI=M_PI;
const int C1=40000;
const int C2=80000;
const int C3=120000;
const double EPS=1e-8;
int n;
int w[N];
LL cnt[S];
LL c[S];
int lenA,lenB,lenR;
int bit,len; int rev[S];
struct CP {
double x,y;
CP (double _x=0,double _y=0) {
x=_x,y=_y;
}
friend CP operator + (CP a,CP b) {
return CP(a.x+b.x,a.y+b.y);
}
friend CP operator - (CP a,CP b) {
return CP(a.x-b.x,a.y-b.y);
}
friend CP operator * (CP a,CP b) {
return CP(a.x*b.x-a.y*b.y,a.x*b.y+a.y*b.x);
}
}a[S],b[S],r[S],tmp[S];
int rd(void) {
int x=0,f=1; char c=getchar();
for (;!isdigit(c);c=getchar()) if (c=='-') f=-1;
for (;isdigit(c);c=getchar()) x=x*10+c-'0';
return x*f;
}
void FFT(CP a[S],int len,int kd) {
rep(i,0,len-1) tmp[i]=a[rev[i]];
rep(i,0,len-1) a[i]=tmp[i];
for (int i=2;i<=len;i<<=1) {
CP wn(cos(2*PI/i),sin(kd*2*PI/i));
for (int k=0;k<len;k+=i) {
CP w(1,0);
rep(j,0,i/2-1) {
CP x=a[k+j],y=w*a[k+j+i/2];
a[k+j]=x+y,a[k+j+i/2]=x-y;
w=w*wn;
}
}
}
if (kd==-1) {
rep(i,0,len-1)
a[i].x/=len;
}
}
void Mut(CP a[S],int lenA,CP b[S],int lenB,CP r[S],int &lenR) {
fill(r,r+S,0);
lenR=lenA+lenB;
len=1,bit=0;
while (len<=lenA||len<=lenB)
len<<=1,bit++;
len<<=1,bit++;
memset(rev,0,sizeof rev);
rep(i,0,len-1) {
int x=0;
rep(j,0,bit-1)
if (i>>j&1)
x+=(1<<(bit-1-j));
rev[i]=x;
}
FFT(a,len,1);
FFT(b,len,1);
rep(i,0,len-1) r[i]=a[i]*b[i];
FFT(r,len,-1);
}
LL Trans(CP a) {
return (LL)(a.x+0.5);
}
int cmp(double x) {
if (fabs(x)<EPS) return 0;
return x<0?-1:1;
}
void IA(CP *a,int &lenA) {
fill(a,a+S,0);
rep(i,1,n) a[w[i]].x+=1.0;
lenA=C1;
while (lenA>=0&&!cmp(a[lenA].x))
lenA--;
}
void IA2(CP a[S],int &lenA) {
fill(a,a+S,0);
rep(i,1,n) a[w[i]+w[i]].x+=1.0;
lenA=C2;
while (lenA>=0&&!cmp(a[lenA].x))
lenA--;
}
void IA3(CP a[S],int &lenA) {
fill(a,a+S,0);
rep(i,1,n) a[w[i]+w[i]+w[i]].x+=1.0;
lenA=C3;
while (lenA>=0&&!cmp(a[lenA].x))
lenA--;
}
void Init(void) {
n=rd();
rep(i,1,n) w[i]=rd();
}
void Solve1(void) {
IA(a,lenA);
rep(i,1,lenA) cnt[i]+=Trans(a[i]);
}
void Solve2(void) {
memset(c,0,sizeof c);
IA(a,lenA); IA(b,lenB);
Mut(a,lenA,b,lenB,r,lenR);
rep(i,0,lenR) c[i]+=Trans(r[i]);
IA2(a,lenA);
rep(i,0,lenA) c[i]-=Trans(a[i]);
rep(i,0,S-1)
c[i]>>=1;
rep(i,0,S-1)
cnt[i]+=c[i];
}
void Solve3(void) {
memset(c,0,sizeof c);
// \sum(x,y,z)
IA(a,lenA); IA(b,lenB);
Mut(a,lenA,b,lenB,r,lenR);
IA(a,lenA);
Mut(a,lenA,r,lenR,b,lenB);
rep(i,0,lenB) c[i]+=Trans(b[i]);;
// \sum(x,x,y)
IA2(a,lenA); IA(b,lenB);
Mut(a,lenA,b,lenB,r,lenR);
rep(i,0,lenR) c[i]-=3*Trans(r[i]);
// \sum(x,x,x)
IA3(a,lenA);
rep(i,0,lenA) c[i]+=2*Trans(a[i]);
rep(i,0,S-1)
c[i]/=6;
rep(i,0,S-1)
cnt[i]+=c[i];
}
void Print(void) {
rep(i,0,S-1)
if (cnt[i]>0)
printf("%d %lld\n",i,cnt[i]);
}
int main(void) {
#ifndef ONLINE_JUDGE
freopen("xsy1728.in","r",stdin);
freopen("xsy1728.out","w",stdout);
#endif
Init();
Solve1();
Solve2();
Solve3();
Print();
return 0;
}
【bzoj3771】【xsy1728】Triple的更多相关文章
- 【BZOJ3771】Triple(生成函数,多项式运算)
[BZOJ3771]Triple(生成函数,多项式运算) 题面 有\(n\)个价值\(w\)不同的物品 可以任意选择\(1,2,3\)个组合在一起 输出能够组成的所有价值以及方案数. \(n,w< ...
- 【BZOJ3771】Triple 生成函数+FFT
[BZOJ3771]Triple Description 我们讲一个悲伤的故事. 从前有一个贫穷的樵夫在河边砍柴. 这时候河里出现了一个水神,夺过了他的斧头,说: “这把斧头,是不是你的?” 樵夫一看 ...
- 新鲜出炉的30个精美的 jQuery & CSS3 效果【附演示和教程】
新鲜出炉的30个精美的 jQuery & CSS3 效果[附演示和教程] 作为最流行的 JavaScript 开发框架,jQuery 在现在的 Web 开发项目中扮演着重要角色,它简化了 ...
- 【疯狂造轮子-iOS】JSON转Model系列之二
[疯狂造轮子-iOS]JSON转Model系列之二 本文转载请注明出处 —— polobymulberry-博客园 1. 前言 上一篇<[疯狂造轮子-iOS]JSON转Model系列之一> ...
- 【疯狂造轮子-iOS】JSON转Model系列之一
[疯狂造轮子-iOS]JSON转Model系列之一 本文转载请注明出处 —— polobymulberry-博客园 1. 前言 之前一直看别人的源码,虽然对自己提升比较大,但毕竟不是自己写的,很容易遗 ...
- 【原创分享·支付宝支付】HBuilder打包APP调用支付宝客户端支付
前言 最近有点空余时间,所以,就研究了一下APP支付.前面很早就搞完APP的微信支付了,但是由于时间上和应用上的情况,支付宝一直没空去研究.然后等我空了的时候,发现支付宝居然升级了支付逻辑,虽然目前还 ...
- 【AutoMapper官方文档】DTO与Domin Model相互转换(上)
写在前面 AutoMapper目录: [AutoMapper官方文档]DTO与Domin Model相互转换(上) [AutoMapper官方文档]DTO与Domin Model相互转换(中) [Au ...
- 【Win 10 应用开发】应用预启动
所谓预启动,其实你一看那名字就知道是啥意思了,这是直接译,也找不到比这个叫法更简练的词了.在系统资源允许的情况下(比如电池电量充足,有足够的内存空间),系统会把用户常用的应用程序在后台启动,但不会显示 ...
- 【Win 10 应用开发】启动远程设备上的应用
这个功能必须在“红石-1”(build 14393)以上的系统版中才能使用,运行在一台设备上的应用,可以通过URI来启动另一台设备上的应用.激活远程应用需要以下前提: 系统必须是build 14393 ...
随机推荐
- .Net使用微軟自帶的用戶驗證和登錄授權
使用微軟的用戶驗證,權限管理的方法 一.使用配置好的數據庫, 1.首先建立想定的數據庫(只填寫名字不加載任何表,如:Login數據庫) 2.使用vs兼容工具命令提示.如圖: 3.導入數據庫所需要的ta ...
- js对数组排序
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
- FreeSWITCH第三方库(视频)的简单介绍(二)
FreeSWITCH使用了大量的第三方库,本文档主要介绍视频相关库的信息: 音频相关库的信息介绍参考:http://www.cnblogs.com/yoyotl/p/5486753.html 其他相关 ...
- Lucky String
Lucky String -- 微软笔试 标签(空格分隔): 算法 A string s is LUCKY if and only if the number of different charact ...
- mongodb .net driver
1.介绍 The official MongoDB .NET Driver provides asynchronous interaction with MongoDB. Powering the d ...
- Freebie - Utility Form: Generate Excel Report From SQL Query In Oracle Forms 6i And 11g
Sharing a form to generate Excel file report from SQL query in Oracle Forms. This form can be used i ...
- Myeclipse 10.7 android(安卓) 开发环境搭建
1 下载并安装JDK,并且设置环境变量 2 下载安装 installer_r24.3.4-windows.exe (Android SDK Manager) 3 使用 Android SDK Mana ...
- bzoj 2242: [SDOI2011]计算器 BSGS+快速幂+扩展欧几里德
2242: [SDOI2011]计算器 Time Limit: 10 Sec Memory Limit: 512 MB[Submit][Status][Discuss] Description 你被 ...
- 【T-SQL系列】临时表、表变量
临时表临时表与永久表相似,只是它的创建是在Tempdb中,它只有在一个数据库连接结束后或者由SQL命令DROP掉,才会消失,否则就会一直存在.临时表在创建的时候都会产生SQL Server的系统日志, ...
- c++ IO的继承结构
#include <stdio.h> #include <iostream>//cin,cout #include <sstream>//ss transfer. ...