【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
*/

转载于:https://www.cnblogs.com/scx2015noip-as-php/p/hdu6067.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值