JZOJ 4294【NOIP2015模拟11.2】复制&粘贴2

Description

文本编辑器的一个最重要的机能就是复制&粘贴。JOI社现在正在开发一款能够非常高速地进行复制&粘贴的文本编辑器,作为JOI社一名优秀的程序猿,你担负起了复制&粘贴功能的测试这一核心工作。整个JOI社的命运都系在你的身上,因此你无论如何都想写出一个正确且高速的程序来完成这项工作。
具体的做法如下所示。文件的内容是一个字符串S,对其进行N次复制&粘贴的操作,第i次操作复制位置Ai和位置Bi之间的所有文字,然后在位置Ci粘贴。这里位置x表示字符串的第x个字符的后面那个位置(位置0表示字符串的开头),例如字符串”copypaste”的位置6表示字符’a’和字符’s’之间的位置,位置9表示’e’后面的位置(即字符串的结尾)。不过,如果操作后的字符串长度超过了M,那么将超过的部分删除,只保留长度为M的前缀。
你的任务是写一个程序,输出N次操作后字符串的前K个字符。

Analysis

把操作倒过去推一下。由于K十分小,设 f[i][j](1<=j<=K) 为最后的K个字符经过 i 轮操作后的位置。分类讨论模拟COPY&PASTE的情况(分为位置不动,后移,被覆盖三种情况)。初始化 f[n][i]=i

Code

#include<cstdio>
#include<cstring>
#include<algorithm>
#define fo(i,a,b) for(int i=a;i<=b;i++)
using namespace std;
const int N=200010;
int f[2][N],x[N],y[N],pos[N];
char s[N];
int main()
{
    freopen("copypaste.in","r",stdin);
    freopen("copypaste.out","w",stdout);
    int k,m,q;
    scanf("%d %d %s %d",&k,&m,s+1,&q);
    int n=strlen(s+1);
    fo(i,1,q) scanf("%d %d %d",&x[i],&y[i],&pos[i]);
    fo(i,1,k) f[0][i]=i;
    int p=0;
    for(int i=q;i>0;i--,p^=1)
    {
        int l=pos[i]+1,r=pos[i]+y[i]-x[i];
        fo(j,1,k)
            if(f[p][j]<l) f[p^1][j]=f[p][j];
            else
            if(l<=f[p][j] && f[p][j]<=r) f[p^1][j]=x[i]+f[p][j]-l+1;
            else
            if(f[p][j]>r) f[p^1][j]=f[p][j]-r+l-1;
    }
    fo(i,1,k) printf("%c",s[f[p][i]]);
    fclose(stdin);fclose(stdout);
    return 0;
}

你可能感兴趣的:(递推,倒推)