码迷,mamicode.com
首页 > 其他好文 > 详细

luogu4173 残缺的字符串

时间:2020-03-01 00:11:48      阅读:60      评论:0      收藏:0      [点我收藏+]

标签:多项式   cpp   its   tor   eps   bitset   push   pac   串匹配   

对于一类带有通配符的字符串匹配问题,我们考虑构造匹配函数,通过匹配函数的值来判断匹配的位置。

先考虑一个不带通配符的问题:给定两个字符串\(A,B\),判断\(B\)的哪些位置能与\(A\)匹配。

除了kmp,我们同样可以考虑构造匹配函数来解决匹配问题,首先将\(A\)串翻转同时在其末尾补\(0\),构造函数\(f_i=\sum_{j=0}^i(A_i-B_{i-j})^2\),那么\(B\)中在第\(i\)个位置结尾的长度为\(|A|\)的子串能与\(A\)匹配当且仅当\(f_i=0\)。将函数展开后得到\(\sum_{j=0}^i(A_i^2-2A_iB_{i-j}+B_{i-j}^2)\),只需要一次FFT便能得到结果。

现在考虑有通配符的情况,那么原来对匹配函数的定义显然是不完备的,考虑第\(i\)位有通配符时直接算作匹配,故可以在第\(i\)位为通配符时令\(A_i=0\),那么便可将匹配函数的定义完善为\(f_i=\sum_{j=0}^i(A_i-B_{i-j})^2A_iB_{i-j}\).

展开后得到\(f_i=\sum_{j=0}^i(A_i^3B_{i-j}-2A_i^2B_{i-j}^2+A_iB_{i-j}^3)\).

做三次多项式乘法即可。

#include<iostream>
#include<string.h>
#include<string>
#include<stdio.h>
#include<algorithm>
#include<vector>
#include<bitset>
#include<math.h>
#include<stack>
#include<queue>
#include<set>
#include<map>
using namespace std;
typedef long long ll;
typedef long double db;
typedef pair<int,int> pii;
const int dx[]={-1,0,1,0},dy[]={0,1,0,-1};
const int N=100000;
const db pi=acos(-1.0);
#define lowbit(x) (x)&(-x)
#define sqr(x) (x)*(x)
#define rep(i,a,b) for (register int i=a;i<=b;i++)
#define per(i,a,b) for (register int i=a;i>=b;i--)
#define go(u,i) for (register int i=head[u];i;i=sq[i].nxt)
#define fir first
#define sec second
#define mp make_pair
#define pb push_back
#define maxd 998244353
#define eps 1e-8
inline int read()
{
    int x=0,f=1;char ch=getchar();
    while ((ch<'0') || (ch>'9')) {if (ch=='-') f=-1;ch=getchar();}
    while ((ch>='0') && (ch<='9')) {x=x*10+(ch-'0');ch=getchar();}
    return x*f;
}

namespace My_Math{
    #define N 100000

    int fac[N+100],invfac[N+100];

    int add(int x,int y) {return x+y>=maxd?x+y-maxd:x+y;}
    int dec(int x,int y) {return x<y?x-y+maxd:x-y;}
    int mul(int x,int y) {return 1ll*x*y%maxd;}
    ll qpow(ll x,int y)
    {
        ll ans=1;
        while (y)
        {
            if (y&1) ans=mul(ans,x);
            x=mul(x,x);y>>=1;
        }
        return ans;
    }
    int inv(int x) {return qpow(x,maxd-2);}

    int C(int n,int m)
    {
        if ((n<m) || (n<0) || (m<0)) return 0;
        return mul(mul(fac[n],invfac[m]),invfac[n-m]);
    }

    int math_init()
    {
        fac[0]=invfac[0]=1;
        rep(i,1,N) fac[i]=mul(fac[i-1],i);
        invfac[N]=inv(fac[N]);
        per(i,N-1,1) invfac[i]=mul(invfac[i+1],i+1);
    }
    #undef N
}
using namespace My_Math;
struct qnode{int x,y,pre,now;}ask[2002000];
int n,m,q,tot=0,a[310][310],fa[3003000],id[310][310],ans[2002000];

int find(int x) 
{
    if (fa[x]==x) return fa[x];
    fa[x]=find(fa[x]);
    return fa[x];
}

int work(int x,int y)
{
    int cnt=1;
    rep(i,0,3)
    {
        int nx=x+dx[i],ny=y+dy[i];
        if ((nx<1) || (nx>n) || (ny<1) || (ny>m)) continue;
        if (a[x][y]==a[nx][ny])
        {
            int fx=find(id[x][y]),fy=find(id[nx][ny]);
            if (fx!=fy) {fa[fx]=fy;cnt--;}
        }
    }
    return cnt;
}

void solve1()
{
    rep(i,1,q)
    {
        if (ask[i].pre==ask[i].now) continue;
        int x=ask[i].x,y=ask[i].y;
        id[x][y]=(++tot);a[x][y]=ask[i].now;fa[tot]=tot;
        ans[i]+=work(x,y);
    }
}
    
void solve2()
{
    tot=0;
    rep(i,1,n) rep(j,1,m) {id[i][j]=(++tot);fa[tot]=tot;}
    rep(i,1,n) rep(j,1,m) work(i,j);
    per(i,q,1)
    {
        if (ask[i].pre==ask[i].now) continue;
        int x=ask[i].x,y=ask[i].y;
        id[x][y]=(++tot);fa[tot]=tot;a[x][y]=ask[i].pre;
        ans[i]-=work(x,y);
    }
}

int main()
{
    n=read();m=read();q=read();
    rep(i,1,q)
    {
        int x=read(),y=read(),w=read();
        ask[i]=(qnode){x,y,a[x][y],w};
        a[x][y]=w;
    }
    rep(i,1,n) rep(j,1,m) a[i][j]=0;
    solve1();solve2();ans[0]=1;
    rep(i,1,q) ans[i]+=ans[i-1];
    rep(i,1,q) printf("%d\n",ans[i]);
    return 0;
}

luogu4173 残缺的字符串

标签:多项式   cpp   its   tor   eps   bitset   push   pac   串匹配   

原文地址:https://www.cnblogs.com/encodetalker/p/12387381.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!