[NOI2018] 你的名字

题目大意:给出一个串\(S\)\(q\)次询问每次询问字符串\(T\)不在\(S[l,r]\)出现的本质不同的子串数目。

0.68

有条件\(l=1,r=|S|\)。考虑\(SAM_T\)上的parent树的结构特点,不难可以得出:若设\(lim_i\)\(T[1,i]\)\(S\)中出现的最长后缀长度(在\(SAM_S\)上直接跑整个\(T\)就能预处理),\(tar_x\)为集合\(right_x\)中任一元素,答案可以表示为

\[\sum_{x\in SAM_T} |[0,lim_{tar_x}]\cap[mxl_{prt_x}+1,mxl_x]| \]

(其实画图想了好久。。)

1.00

将上一个做法中的预处理部分做出如下修改:若目标状态的right集合存在位于\([l+len-1,r]\)中的元素则转移(len是已上一个后缀的lim值),否则挨个考虑时len-=1,当len=prt的mxl时转向prt节点。(对于len满足|对于len-1满足?)

这样使用可持久化线段树+线段树合并求出right集合来做。


感谢 QAQ

恼惭做了60h。

代码 :模板线段树合并求right集合

#include <cstdio>
#include <cstring>
#include <algorithm>
#include <vector>

const int N=1e6+10; //faq

namespace SGT {
	int ch[N*30][2],cnt;
	bool exi[N*30];
	void insert(int&x,int l,int r,int p) {
		exi[x=++cnt]=1;
		if(l==r) return;
		int mid=(l+r)>>1;
		if(p<=mid) insert(ch[x][0],l,mid,p);
		else insert(ch[x][1],mid+1,r,p);
	}
	int merge(int x,int y,int l,int r) {
		if(!x||!y) return x|y;
		int nd=++cnt,mid=(l+r)>>1;
		ch[nd][0]=merge(ch[x][0],ch[y][0],l,mid);
		ch[nd][1]=merge(ch[x][1],ch[y][1],mid+1,r);
		exi[nd]=exi[ch[nd][0]]|exi[ch[nd][1]];
		return nd;
	}
	bool query(int x,int l,int r,int L,int R) {
		//printf("[%d,%d,%d,%d,%d]\n",x,l,r,L,R);
		if(!x||R<L) return 0;
		if(L<=l&&r<=R) return exi[x];
		int mid=(l+r)>>1;
		if(L<=mid&&query(ch[x][0],l,mid,L,R)) return 1;
		if(mid<R&&query(ch[x][1],mid+1,r,L,R)) return 1;
		return 0;
	}
}

int n,m,root[N*2];
std::vector<int> e[N*2];

struct SAM {
	struct Node {
		int ch[26],prt,mxl,tar;
	} t[N*2];
	int cnt,lst;
	inline void clear() {
		for(int i=1; i<=cnt; ++i) memset(&t[i],0,sizeof t[i]);
		cnt=lst=1;
	}
	inline int extend(int c,int pos) {
		int p=lst,x=lst=++cnt;
		t[x].tar=pos;
		t[x].mxl=t[p].mxl+1;
		while(p&&!t[p].ch[c]) t[p].ch[c]=x,p=t[p].prt;
		if(!p) t[x].prt=1;
		else {
			int q=t[p].ch[c];
			if(t[q].mxl==t[p].mxl+1) t[x].prt=q;
			else {
				int k=++cnt; 
				t[k]=t[q]; 
				t[k].mxl=t[p].mxl+1;
				t[q].prt=t[x].prt=k;
				while(p&&t[p].ch[c]==q) t[p].ch[c]=k,p=t[p].prt;
			}
		}
		return x;
	}
	inline void dfs(int x) {
		for(int y:e[x]) {
			dfs(y);
			root[x]=SGT::merge(root[x],root[y],1,N);
		}
	}
	inline void build(char *s,int N,bool sgt) {
		clear();
		for(int i=1; i<=N; ++i) {
			int x=extend(s[i]-'a',i);
			if(sgt) SGT::insert(root[x],1,N,i);
		}
		if(sgt) {
			for(int i=2; i<=cnt; ++i) e[t[i].prt].push_back(i);
			dfs(1);
		}
	}
	void key() {
		printf("total nodes: %d\n",cnt);
		for(int i=1; i<=cnt; ++i) {
			printf(" %d %d %d\n",t[i].mxl,t[i].tar,t[i].prt);
		}
	}
} S,T;

char s[N];
int L,R,Q,lim[N];

inline long long calc() {
	scanf("%s%d%d",s+1,&L,&R);
	T.build(s,m=strlen(s+1),0);
	//T.key();
	int now=1,len=0;
	for(int i=1; i<=m; ++i) {
		int c=s[i]-'a';
		while(1) {
			//printf("<%d %d> ",now,len);
			if(SGT::query(root[S.t[now].ch[c]],1,n,L+len,R)) {
				//printf(" ent ");
				len++;
				now=S.t[now].ch[c];
				break;
			} else {
				if(now==1) break;
				--len;
				if(len==S.t[S.t[now].prt].mxl) now=S.t[now].prt; 
			}
		}
		lim[i]=len;
	}
	long long ans=0;
	for(int i=1; i<=T.cnt; ++i) {
		ans+=T.t[i].mxl-T.t[T.t[i].prt].mxl;
		ans-=std::max(0,std::min(T.t[i].mxl,lim[T.t[i].tar])-T.t[T.t[i].prt].mxl);
	}
	return ans;
}

int main() {
	freopen("name.in","r",stdin);
	freopen("name.out","w",stdout);
	scanf("%s%d",s+1,&Q);
	S.build(s,n=strlen(s+1),1); 
	//S.key();
	while(Q--) printf("%lld\n",calc());
	return 0;
}
posted @ 2019-05-04 21:36  nosta  阅读(235)  评论(0编辑  收藏  举报