【hdu 6067】Big Integer
题意
给你一个 \((k-1)\times (n+1)\) 的 \(01\) 矩阵 \(g\),求满足下列条件的 \(k(k\le 10)\) 进制整数的数量:
1. 不超过 \(n\) 位且数的最高位非 \(0\)
2. 没有出现 \(0\)
3. 对于 \(0\) 以外的数字 \(i\),对于 \(j∈[0,n]\),若 \(g(i,j)=1\),则允许数字 \(i\) 恰好出现 \(j\) 次;若 \(g(i,j)=0\),则不允许数字 \(i\) 恰好出现 \(j\) 次。
这个问题太简单了,于是有 \(m\) 次修改操作,每次将 \(g(i,j)\) 单点取反。让你求修改前及每次修改操作后的答案之和 \(\mod 786433\)。
\(786433=2^{18}\times 3 + 1\),是个质数。
\(k\le 10,\space n\le 14000,\space m\le 200\)
题解
所以放一个这么明显的 \(\text{NTT}\) 模数是什么意思
前置普及组知识:你有 \(x_1\) 个 \(1\),\(x_2\) 个 \(2\),……,\(x_n\) 个 \(n\),用这 \(x_1+x_2+\cdots+x_n\) 个数构成的不同排列数为 \(\frac{(x_1+x_2+\cdots+x_n)!}{x_1! x_2! \cdots x_n!}\)。
构造指数生成函数 \(f_i(x) = \sum\limits_{j=0}^{n} g(i,j) \frac{x^j}{j!}\),将这 \(k-1\) 个多项式卷积成一个生成函数后,记 \(i\) 次项系数为 \(a_i\),则答案为 \(\sum\limits_{i=1}^n a_i i!\)。
可以用 \(\text{NTT}\) 在 \(O(nk^2\log (nk))\) 的复杂度内预处理出初始答案。
下面考虑修改。注意到我们只关心所有答案的和,故可以在 \(\text{DFT}\) 意义下直接累加答案,最后再将结果 \(\text{IDFT}\) 回来。
对于单点修改操作,可以看成是给某个多项式 \(A\) 叠加上一个只有一项系数不为 \(0\) 的多项式 \(B\)。
因为 \(A\) 正处于点值表示法,所以我们把 \(B\) 也转化成点值表示法(其长度需要扩到与 \(A\) 相等)。这需要 \(O(nk\log(nk))\) 的时间由于只有一项系数不为 \(0\),我们考虑暴力 \(\text{DFT}\)。
观察指数生成函数 \(\text{NTT}\) 的公式:
$$y_n = \sum\limits_{i=0}^{d-1}\frac{x_n}{n!}\times (g\frac{p-1}{d}){ni}\mod p$$
那么之前的多项式 \(B\) 在 \(\text{DFT}\) 后的结果是一个等比数列,故直接对原始多项式叠加一个等比数列即可。
然而如果直接暴力叠加的话,修改部分的复杂度是 \(O(nmk^2)\)(不过实测能卡过)。
发现把 \(k-1\) 个 \(\text{DFT}\) 后的多项式放成 \(k-1\) 行,依次对齐每次项,由于点值表示法下,这些多项式卷起来的第 \(i\) 位是这些多项式第 \(i\) 位的乘积,显然只要有一个是 \(0\),这一列就废了。考虑记录 \(\text{DFT}\) 下每一列 \(0\) 的数量以及非 \(0\) 数的乘积,这样每次单点修改时就只需要修改该点所在的一行多项式的信息。
复杂度 \(O(nk^2\log(nk) + nmk)\)。
#include<bits/stdc++.h>
#define ll long long
#define N 140010
#define mod 786433
#define G 10
#define invG 235930
using namespace std;
inline int read(){
int x=0; bool f=1; char c=getchar();
for(;!isdigit(c);c=getchar()) if(c=='-') f=0;
for(; isdigit(c);c=getchar()) x=(x<<3)+(x<<1)+c-'0';
if(f) return x; return 0-x;
}
int Pow(int x, int y){
int ret=1;
while(y){
if(y&1) ret=(ll)ret*x%mod;
x=(ll)x*x%mod;
y>>=1;
}
return ret;
}
struct Poly{
int n,bit,r[N];
void init(int x){
for(n=1,bit=0; n<x; n<<=1,++bit);
for(int i=1; i<n; ++i) r[i]=(r[i>>1]>>1)|((i&1)<<(bit-1));
//cout<<"n:"<<n<<endl;
}
void dft(int *a, int f){
for(int i=0; i<n; ++i) if(i<r[i]) swap(a[i],a[r[i]]);
for(int i=1; i<n; i<<=1){
int wn = Pow(f==1 ? G : invG, (mod-1)/(i<<1));
for(int j=0; j<n; j+=(i<<1)){
int w=1,x,y;
for(int k=0; k<i; ++k,w=(ll)w*wn%mod)
x=a[j+k], y=(ll)w*a[j+i+k]%mod,
a[j+k]=(x+y)%mod, a[j+i+k]=(x-y+mod)%mod;
}
}
if(f==-1){
int mul=Pow(n,mod-2);
for(int i=0; i<n; ++i) a[i]=(ll)a[i]*mul%mod;
}
}
}NTT;
char c[11][N];
int k,n,m,e[11][N],f[11][N],g[N],ans;
int inv[mod+5],jc[N],jcn[N];
int mul[N],zero_cnt[N];
int main(){
k=read(), n=read(), m=read();
int num=(k-1)*n;
inv[0]=inv[1]=jc[0]=jcn[0]=1;
for(int i=2; i<mod; ++i) inv[i]=(ll)(mod-mod/i)*inv[mod%i]%mod;
for(int i=1; i<=num; ++i) jc[i]=(ll)jc[i-1]*i%mod, jcn[i]=inv[jc[i]];
for(int i=1; i<k; ++i){
scanf("%s",c[i]);
for(int j=0; j<=n; ++j){
e[i][j]=c[i][j]-'0';
f[i][j] = e[i][j] ? jcn[j] : 0;
}
}
NTT.init(num+1);
for(int i=0; i<NTT.n; ++i) mul[i]=1, g[i]=1;
for(int i=1; i<k; ++i){
NTT.dft(f[i],1);
for(int j=0; j<NTT.n; ++j){
g[j]=(ll)g[j]*f[i][j]%mod;
if(!f[i][j]) ++zero_cnt[j];
else mul[j]=(ll)mul[j]*f[i][j]%mod;
//cout<<f[i][j]<<endl;
}
}
//for(int i=0; i<NTT.n; ++i) cout<<g[i]<<endl;
int x,y;
while(m--){
x=read(), y=read();
e[x][y]^=1;
//for(int i=0; i<NTT.n; ++i) cout<<mul[i]<<' '<<zero_cnt[i]<<endl;
for(int i=0; i<NTT.n; ++i){
if(f[x][i]) mul[i]=(ll)mul[i]*inv[f[x][i]]%mod;
else --zero_cnt[i];
}
int val=jcn[y], tol=Pow(G,(mod-1)/NTT.n*y%(mod-1));
if(!e[x][y]) val=mod-val;
for(int i=0; i<NTT.n; ++i){
f[x][i]=(f[x][i]+val)%mod;
//cout<<f[x][i]<<endl;
val=(ll)val*tol%mod;
}
for(int i=0; i<NTT.n; ++i){
if(f[x][i]) mul[i]=(ll)mul[i]*f[x][i]%mod;
else ++zero_cnt[i];
if(!zero_cnt[i]) g[i]=(g[i]+mul[i])%mod;
}
}
NTT.dft(g,-1);
for(int i=1; i<=num; ++i){
ans=(ans+(ll)g[i]*jc[i]%mod)%mod;
//cout<<f[k-1][i]<<' '<<jc[i]<<endl;
}
cout<<ans<<endl;
return 0;
}
/*
3 2 0
101
010
3 2 0
111
010
3 2 0
110
010
3 2 1
101
010
1 1
*/
【hdu 6067】Big Integer的更多相关文章
- 【数位dp】【HDU 3555】【HDU 2089】数位DP入门题
[HDU 3555]原题直通车: 代码: // 31MS 900K 909 B G++ #include<iostream> #include<cstdio> #includ ...
- 【HDU 5647】DZY Loves Connecting(树DP)
pid=5647">[HDU 5647]DZY Loves Connecting(树DP) DZY Loves Connecting Time Limit: 4000/2000 MS ...
- -【线性基】【BZOJ 2460】【BZOJ 2115】【HDU 3949】
[把三道我做过的线性基题目放在一起总结一下,代码都挺简单,主要就是贪心思想和异或的高斯消元] [然后把网上的讲解归纳一下] 1.线性基: 若干数的线性基是一组数a1,a2,a3...an,其中ax的最 ...
- 【HDU 2196】 Computer(树的直径)
[HDU 2196] Computer(树的直径) 题链http://acm.hdu.edu.cn/showproblem.php?pid=2196 这题可以用树形DP解决,自然也可以用最直观的方法解 ...
- 【HDU 2196】 Computer (树形DP)
[HDU 2196] Computer 题链http://acm.hdu.edu.cn/showproblem.php?pid=2196 刘汝佳<算法竞赛入门经典>P282页留下了这个问题 ...
- 【HDU 5145】 NPY and girls(组合+莫队)
pid=5145">[HDU 5145] NPY and girls(组合+莫队) NPY and girls Time Limit: 8000/4000 MS (Java/Other ...
- 【hdu 2955】Robberies
Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Total Submission(s) ...
- 【hdu 5996】dingyeye loves stone
Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Others) Total Submission(s) ...
- 【hdu 2486】A simple stone game
Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Total Submission(s) ...
随机推荐
- 系统启动热键(Boot Hotkey)
1.HP ENVY 13 F1 --> 系统信息F2 --> 系统检测F9 --> 启动设备选项F10 --> 设置BIOSF11 --> 系统恢复ENTER --> ...
- 【Linux】常用基础命令
修改时间 date -s 月/日/年 例如:date -s 07/31/2019 date -s 时:分:秒 例如:date -s 23:56:50 hwclock -w 将时间写到bois,防止重启 ...
- 【MM系列】SAP S/4 HANA BP创建客户/供应商的一点想法
公众号:SAP Technical 本文作者:matinal 原文出处:http://www.cnblogs.com/SAPmatinal/ 原文链接:[MM系列]SAP S/4 HANA BP创建客 ...
- 对scrapy进行单元测试 -- 使用betamax
使用betamax进行单元测试 爬虫代码 测试代码 对于scrapy的单元测试,官方文档并没有提到,只是说有一个Contract功能.但是相信我,这个东西真的不好用,甚至scrapy的作者在一个iss ...
- Linux常用命令详解(1)
基础命令: ls man pwd cd mkdir echo touch cp mv rm rmdir cat more less head tail clear poweroff reboot 命令 ...
- limit和offset、切片操作
#encoding: utf-8 from sqlalchemy import create_engine,Column,Integer,String,Float,func,and_,or_,\ Da ...
- 复杂json格式转化为javabean
工具阿里巴巴的fastjson包 <!-- https://mvnrepository.com/artifact/com.alibaba/fastjson --><dependenc ...
- Laravel验证问题记录
1.当购物车提交时,POST传来一个对象{address:2,item:{ {ksu_id:2,count:2},{ksu_id:2,count:2}, } 验证方法: public function ...
- 3.Java和hadoop的安装
先创建目录 [hadoop@node1 opt]$ cd /opt [hadoop@node1 opt]$ sudo mkdir /opt/softwares [hadoop@node1 opt]$ ...
- (5.8)mysql高可用系列——MySQL中的GTID复制(实践篇)
一.基于GTID的异步复制(一主一从)无数据/少数据搭建 二.基于GTID的无损半同步复制(一主一从)(mysql5.7)基于大数据量的初始化 正文: [0]概念 [0.5]GTID 复制(mysql ...