1 条题解

  • 0
    @ 2023-6-21 19:59:14

    C++ :

    #include<cstdio>
    #include<algorithm>
    #define N 280005
    #define H 19
    #define A 26
    #define S 317
    using namespace std;
    int read()
    {
    	int r = 0; char c = getchar();
    	for(; c < '0' || c > '9'; c = getchar());
    	for(; c >='0' && c <='9'; r = r*10+c-'0', c = getchar());
    	return r;
    }
    namespace runzhe2000
    {	
    	typedef long long ll;
    
    	int happy, n, m, q, k, block;
    	char s[N];
    	
    	struct inter
    	{
    		int l, r, len;
    		bool operator < (const inter &that) const {return l == that.l ? r < that.r : l < that.l;}
    		bool operator == (const inter &that) const {return l == that.l && r == that.r;}
    	}in[N];
    
    //////////
    	
    	int tot, pos[N], log[N], lm[N][H], rm[N][H], pre[N], a[N], b[N];
    	int t1[N<<1], t2[N<<1], sa[N<<1], rank[N<<1], height[N<<1], sum[N<<1];
    	
    	void build_SA()
    	{
    		int *x = t1, *y = t2, m = q + 128;
    		for(int i = 1; i <= tot; i++) sum[x[i] = s[i] + q]++;
    		for(int i = 1; i <= m; i++) sum[i] += sum[i-1];
    		for(int i = 1; i <= tot; i++) sa[sum[x[i]]--] = i;
    		for(int k = 1; k < tot; k <<= 1)
    		{
    			int p = 0;
    			for(int i = tot-k+1; i <= tot; i++) y[++p] = i;
    			for(int i = 1; i <= tot; i++) if(sa[i]-k >= 1) y[++p] = sa[i]-k;
    			for(int i = 1; i <= m; i++) sum[i] = 0;
    			for(int i = 1; i <= tot; i++) sum[x[i]]++;
    			for(int i = 1; i <= m; i++) sum[i] += sum[i-1];
    			for(int i = tot; i >= 1; i--) sa[sum[x[y[i]]]--] = y[i];
    			swap(x, y);
    			for(int i = 1; i <= tot; i++)
    				x[sa[i]] = x[sa[i-1]] + (y[sa[i]] == y[sa[i-1]] && y[sa[i]+k] == y[sa[i-1]+k] ? 0 : 1);
    			m = x[sa[tot]]; if(m == n) break;
    		}
    		for(int i = 1; i <= tot; i++) rank[sa[i]] = i;
    		for(int i = 1, k = 0; i <= tot; height[rank[i++]] = k, k ? k-- : 0)
    			for(; s[i+k] == s[sa[rank[i]-1]+k]; k++);
    	}
    	
    	void init_ST()
    	{
    		log[1] = 0; for(int i = 2; i <= tot; i++) log[i] = log[i >> 1] + 1;
    		for(int i = 1; i <= tot; i++) lm[i][0] = rm[i][0] = height[i];
    		for(int h = 1; h < H; h++)
    			for(int i = 1; i <= tot; i++)
    			{
    				lm[i][h] = min(lm[i][h-1], lm[i+(1<<(h-1))][h-1]);
    				1 <= i-(1<<(h-1))
    					? rm[i][h] = min(rm[i][h-1], rm[i-(1<<(h-1))][h-1])
    					: rm[i][h] = min(rm[i-1][h], height[i]);
    			}
    	}
    	int ST(int l, int r)
    	{
    		int d = log[r-l+1];
    		return min(lm[l][d], rm[r][d]);
    	}
    	
    	int find_l(int p, int lim)
    	{
    		int l = 1, r = p;
    		for(; l < r; )
    		{
    			int mid = (l+r)>>1;
    			if(ST(mid+1,p) < lim) l = mid + 1;
    			else r = mid;
    		}
    		return l;
    	}
    	int find_r(int p, int lim)
    	{
    		int l = p, r = tot;
    		for(; l < r; )
    		{
    			int mid = (l+r+1)>>1;
    			if(ST(p+1, mid) < lim) r = mid - 1;
    			else l = mid;
    		}
    		return l;
    	}
    	
    	void solve1()
    	{
    		tot = n; s[++tot] = 0;
    		for(int i = 1; i <= q; i++)
    		{
    			happy = scanf("%s",s+tot+1);
    			pos[i] = tot + 1;
    			s[tot += k + 1] = -i;
    			a[i] = read()+1, b[i] = read()+1;
    		}
    		build_SA();
    		for(int i = 1; i <= tot; i++) pre[i] = pre[i-1] + (sa[i] <= n);
    		init_ST();
    		for(int i = 1; i <= q; i++)
    		{
    			ll f = 0;
    			for(int j = a[i]; j <= b[i]; j++)
    			{
    				int l = find_l(rank[pos[i]+in[j].l-1], in[j].len), r = find_r(rank[pos[i]+in[j].l-1], in[j].len);
    				f += pre[r] - pre[l-1];
    			}
    			printf("%lld\n",f);
    		}
    	}
    	
    //////////
    
    	struct sam
    	{
    		sam *next[A], *fail;
    		int len, right;
    	}mem[N], *totp, *null, *last, *root, *que[N];
    	sam *newsam()
    	{
    		sam *x = ++totp; *x = *null; 
    		return x;
    	}
    	void init_SAM()
    	{
    		null = totp = mem; for(int i = 0; i < A; i++) null->next[i] = null, null->fail = null; null->len = null->right = 0;
    		last = root = newsam();
    	}
    	void extend(int i)
    	{
    		sam *p = last, *np = last = newsam(); np->len = p->len+1, np->right = 1; int v = s[i] - 'a';
    		for(; p->next[v] == null && p != null; p = p->fail) p->next[v] = np;
    		if(p == null) np->fail = root;
    		else
    		{
    			if(p->next[v]->len == p->len+1) np->fail = p->next[v];
    			else
    			{
    				sam *q = p->next[v], *nq = newsam(); *nq = *q; nq->len = p->len + 1; nq->right = 0;
    				q->fail = np->fail = nq;
    				for(; p->next[v] == q && p != null; p = p->fail) p->next[v] = nq;
    			}
    		}
    	}
    	bool cmp(sam *a, sam *b){return a->len < b->len;}
    	
    	struct event
    	{
    		int id, p, type;
    		bool operator < (const event &that) const
    		{
    			return p < that.p;
    		}
    	}eve[N<<1]; 
    	ll ans[N];
    	int evecnt, tmp[N];
    	char str[100001][S];
    	
    	struct seg
    	{
    		seg *ch[2]; int v;
    	}memt[N*16], *tott, *nullt, *ro[S];
    	seg *newseg()
    	{
    		seg *x = ++tott; *x = *nullt;
    		return x;
    	}
    	void init_seg()
    	{
    		nullt = tott = memt; nullt->ch[0] = nullt->ch[1] = nullt;
    		nullt->v = 0; for(int i = 1; i <= k; i++) ro[i] = newseg();
    	}
    	void insert(seg *x, int l, int r, int p)
    	{
    		if(l == r) {x->v++; return;} int mid = (l+r)>>1;
    		if(p <= mid) 
    		{
    			if(x->ch[0] == nullt) x->ch[0] = newseg();
    			insert(x->ch[0], l, mid, p);
    		}
    		else
    		{
    			if(x->ch[1] == nullt) x->ch[1] = newseg();
    			insert(x->ch[1], mid+1, r, p);
    		}
    	}
    	void push(seg *x, int l, int r)
    	{
    		if(l == r) {tmp[l] = x->v; return;}; int mid = (l+r)>>1;
    		push(x->ch[0], l, mid);
    		push(x->ch[1], mid+1, r);
    	}
    	
    	void solve2()
    	{
    		init_seg(); init_SAM(); for(int i = 1; i <= n; i++) extend(i);
    		for(sam *i = totp; i != mem; i--) que[i-mem] = i;
    		sort(que+1, que+1+(totp-mem), cmp);
    		for(int i = totp-mem; i > 1; i--) que[i]->fail->right += que[i]->right;
    		for(int i = 1; i <= q; i++) 
    		{
    			happy = scanf("%s",str[i]+1);
    			int a = read(), b = read();
    			eve[++evecnt] = (event){i, a   , -1};
    			eve[++evecnt] = (event){i, b+1 ,  1};
    		}
    		sort(eve+1, eve+1+evecnt);
    		for(int i = 0, cur = 1; i <= m; i++)
    		{
    			if(i) insert(ro[in[i].l], 1, k, in[i].r);
    			for(; cur <= evecnt && eve[cur].p == i; cur++)
    			{
    				ll f = 0;
    				for(int l = 1; l <= k; l++)
    				{
    					push(ro[l], 1, k); sam *p = root;
    					for(int r = l; r <= k; r++)
    					{
    						p = p->next[str[eve[cur].id][r]-'a'];
    						f += p->right * tmp[r];
    					}
    				}
    				ans[eve[cur].id] += eve[cur].type * f;
    			}
    		}
    		for(int i = 1; i <= q; i++) printf("%lld\n",ans[i]);
    	}
    	
    //////////
    	
    	void main()
    	{
    		n = read(), m = read(), q = read(), k = read();
    		happy = scanf("%s",s+1);
    		
    		block = 316;
    		
    		for(int i = 1; i <= m; i++) in[i].l = read()+1, in[i].r = read()+1, in[i].len = in[i].r-in[i].l+1;
    		if(q < block) solve1();
    		else solve2();
    	}
    }
    int main()
    {
    	runzhe2000::main();
    }
    
    • 1

    信息

    ID
    1010
    时间
    5000ms
    内存
    512MiB
    难度
    10
    标签
    递交数
    2
    已通过
    2
    上传者