第六集,想不到你这个浓眉大眼的都叛变革命了

题意:

给你两个只包含01的字符串S和T,问你在允许一次错误的情况下,T是否能成为S的子串

思路:

这个问题的解法挺多,我是用fft匹配的,也比较简单,针对0和1匹配两次,第一次针对0就是把S串和T串中等于0的位置都标记成1,然后reverse一个串后进行fft,如果这两个位置都是0,就会出现1*1=1的情况,代表有一个位置匹配上了,0这样做一次,1这样做一次,他们的和就是匹配成功的次数,所以允许一次错误就是判断和是否大于len-1。

还有一个做法是指数哈希,判断两个串的哈希值的差是否是2^n,如果是的话check一下,就做出来了,2^n可以塞到hash或者map里。

还有exkmp等其他做法你们自行了解一下

代码实现

给出fft的做法(我只写了fft)

#include <iostream>

#include <string>

#include <cstring>

#include <algorithm>

using namespace std;

typedef long long ll;

const int  N = ;

ll a[N],b[N];

const ll PMOD=(<<)+;

const ll PR=;

static ll qp[];

ll res[N];

struct NTT__container

{

    NTT__container()

    {

        int  t,i;

        for( i=; i<; i++)///注意循环上界与2n次幂上界相同

        {

            t=<<i;

            qp[i]=quick_pow(PR,(PMOD-)/t);

        }

    }

    ll quick_pow(ll x,ll n)

    {

        ll ans=;

        while(n)

        {

            if(n&)

                ans=ans*x%PMOD;

            x=x*x%PMOD;

            n>>=;

        }

        return ans;

    }

    int get_len(int n)///计算刚好比n大的2的N次幂

    {

        int i,len;

        for(i=(<<); i; i>>=)

        {

            if(n&i)

            {

                len=(i<<);

                break;

            }

        }

        return len;

    }

    inline void NTT(ll F[],int len,int type)

    {

        int id=,h,j,k,t,i;

        ll E,u,v;

        for(i=,t=; i<len; i++)

        {

            if(i>t)   swap(F[i],F[t]);

            for(j=(len>>); (t^=j)<j; j>>=);

        }

        for( h=; h<=len; h<<=)

        {

            id++;

            for( j=; j<len; j+=h)

            {

                E=;

                for(int k=j; k<j+h/; k++)

                {

                    u=F[k];

                    v=(E*F[k+h/])%PMOD;

                    F[k]=(u+v)%PMOD;

                    F[k+h/]=((u-v)%PMOD+PMOD)%PMOD;

                    E=(E*qp[id])%PMOD;

                }

            }

        }

        if(type==-)

        {

            int i;

            ll inv;

            for(i=; i<len/; i++)

                swap(F[i],F[len-i]);

            inv=quick_pow(len,PMOD-);

            for( i=; i<len; i++)

                F[i]=(F[i]%PMOD*inv)%PMOD;

        }

    }

    inline void inv(ll *a,int len)///答案存在res中

    {

        if(len==)

        {

            res[]=quick_pow(a[],PMOD-);

            return ;

        }

        inv(a,len>>);///递归

        static ll temp[N];

        memcpy(temp,a,sizeof(ll)*(len>>));

        NTT(temp,len,);

        NTT(res,len,);

        int i;

        for(i=; i<len; i++)

            res[i]=res[i]*(-temp[i]*res[i]%PMOD+PMOD)%PMOD;

        NTT(res,len,-);

        memset(res+(len>>),,sizeof(ll)*(len>>));

    }

    void mul(ll x[],ll y[],int len)///答案存在x中

    {

        int i;

        NTT(x,len,);///先映射到频域上

        NTT(y,len,);///先映射到频域上

        for(i=; i<len; i++)

            x[i]=(x[i]*y[i])%PMOD;///在频域上点积

        NTT(x,len,-);///再逆变换回时域

    }

} cal;

ll x[N],y[N],z[N];

int main()

{

    ios::sync_with_stdio(false);

    cin.tie();

    int N;

    cin>>N;

    while(N--){

        string s1,s2;

        cin>>s1>>s2;

        if(s2.length()>s1.length()){

            puts("NO");

            continue;

        }

        int len=cal.get_len(s1.length());

        //做1匹配

        memset(x,,len*sizeof(ll));

        memset(y,,len*sizeof(ll));

        memset(z,,len*sizeof(ll));

        for(int i=;i<s1.length();i++){

            x[i]=s1[i]-'';

        }

        for(int i=;i<s2.length();i++){

            y[i]=s2[i]-'';

        }

        reverse(y,y+s2.length());

        cal.mul(x,y,len);

        //for(int i=0;i<len;i++)

        for(int i=s2.length()-;i<s1.length();i++)

            z[i]+=x[i];

        //做0匹配

        memset(x,,len*sizeof(ll));

        memset(y,,len*sizeof(ll));

        for(int i=;i<s1.length();i++){

            x[i]=-(s1[i]-'');

        }

        for(int i=;i<s2.length();i++){

            y[i]=-(s2[i]-'');

        }

        reverse(y,y+s2.length());

        cal.mul(x,y,len);

        //for(int i=0;i<len;i++)

        for(int i=s2.length()-;i<s1.length();i++)

            z[i]+=x[i];

        bool flag=;

        for(int i=s2.length()-;i<s1.length();i++){

            if(z[i]>=s2.length()-){

                flag=;

                break;

            }

        }

        if(flag){

            puts("YES");

        }

        else {

            puts("NO");

        }

    }

}

福建工程学院第十四届ACM校赛J题题解的更多相关文章

  1. 福建工程学院第十四届ACM校赛M题题解 fwt进阶,手推三进制fwt

    第九集,结束亦是开始 题意: 大致意思就是给你n个3进制的数字,让你计算有多少对数字的哈夫曼距离等于i(0<=i<=2^m) 思路: 这个是一个防ak题,做法是要手推公式的fwt 大概就这 ...

  2. 福建工程学院第十四届ACM校赛G题题解

    外传:编剧说了不玩游戏不行 题意: 有n个石堆,我每次只能从某一堆中取偶数个石子,你取奇数个,我先手,先不能操作的人输.问最后谁能赢. 思路: 这个题仔细想想,就发现,取奇数的人有巨大的优势,因为假设 ...

  3. 福建工程学院第十四届ACM校赛B题题解

    第二集,未来的我发量这么捉急的吗 题意: 有n个数,请问有多少对数字(i,j)(1<=i<j<=n),满足(a[i]^a[j])+((a[i]&a[j])<<1) ...

  4. 福建工程学院第十四届ACM程序设计大赛 - E - 外传:小晋逃生记

    http://www.fjutacm.com/Contest.jsp?cid=705#P4 其实想清楚了就很简单,之前想了很多种方法,以为是二分什么的,看起来就像是一个单峰函数.但是发现直接暴力一波就 ...

  5. 2016 华南师大ACM校赛 SCNUCPC 非官方题解

    我要举报本次校赛出题人的消极出题!!! 官方题解请戳:http://3.scnuacm2015.sinaapp.com/?p=89(其实就是一堆代码没有题解) A. 树链剖分数据结构板题 题目大意:我 ...

  6. 湖南大学第十四届ACM程序设计新生杯(重现赛)G a+b+c+d=? (16进制与LL范围)

    链接:https://ac.nowcoder.com/acm/contest/338/G来源:牛客网 时间限制:C/C++ 1秒,其他语言2秒空间限制:C/C++ 32768K,其他语言65536K6 ...

  7. 湖南大学第十四届ACM程序设计新生杯(重现赛)I:II play with GG(博弈论||DP)

    链接:https://ac.nowcoder.com/acm/contest/338/I 来源:牛客网 题目描述 IG won the S championship and many people a ...

  8. 湖南大学第十四届ACM程序设计新生杯(重现赛)

    RANK  0 题数 0 期末复习没有参加,补几道喜欢的题. A: AFei Loves Magic  签到 思路 :不需考虑 碰撞 直接计算最终状态即可. #include<bits/stdc ...

  9. 湖南大学第十四届ACM程序设计新生杯 E.Easy Problem

    E.Easy Problem Description: Zghh likes number, but he doesn't like writing problem description. So h ...

随机推荐

  1. Eclipse在线安装插件进度缓慢问题

    最近在学习Maven的过程中需要安装m2e 插件,在线安装的缓慢速度实在是让人抓狂,故将自己最后的解决方案记录下来,以供其他人参考. 最终的原因是安装时同时检查更新了其他插件的最新版,所以安装插件时注 ...

  2. sql把一段时间分割成周,月,季度,年的时间段

    --本周 select TO_CHAR(CREATE_DATE ,'yyyy-MM-dd')as NEW_DATE , TO_CHAR(trunc(CREATE_DATE, ,'yyyy-MM-dd' ...

  3. orale数据库的SQL查询

      创建学生表,成绩表,教师表,课程表,分别添加数据信息 create table student( sno ) primary key, sname ), sage ), ssex ) ); cre ...

  4. Qt新安装之后出现Error while building/deploying (kit: Desktop Qt 5.7.0 GCC 64bit) When executing step "Make”

      Ubuntu14.04初次安装Qt之后可能出现Error while building/deploying project *** (kit: Desktop Qt 5.7.0 GCC 64bit ...

  5. 一、初识Spring Boot框架

    一.搭建Spring Boot环境 1.选择Project,选择Spring Initializr 2.选择Sdk与默认url 3.点击 Next 4.修改一下Group信息和Artifacet,Ne ...

  6. 利用Viewpager和Fragment实现UI框架的搭建实现

    package com.loaderman.uiframedemo; import android.os.Bundle; import android.support.v4.app.Fragment; ...

  7. 机器学习之K-Mean聚类算法

    知识点: # coding = utf-8 import numpy as np import pandas as pd from sklearn.cluster import KMeans &quo ...

  8. TypeScript 编译目标(target)设置

    TypeScript的编译配置文件tsconfig.json中有许多配置项,本文简单对比编译目标环境的配置项(target)的设置.模块(module)不在本文讨论之内,是另外一个大话题. 实验对于t ...

  9. linux 基础 yum 安装

    ls /dev/cdrom mkdir /mnt/cdrom mount -r /dev/cdrom /mnt/cdrom

  10. debian中安装pip

    sudo apt upate sudo apt-get install python3-pip