提交记录 4240


用户 题目 状态 得分 用时 内存 语言 代码长度
mangoyang noi18c. 【NOI2018】你的名字 Accepted 100 1.976 s 570396 KB C++ 3.98 KB
提交时间 评测时间
2018-07-19 10:00:03 2020-07-31 22:45:36
/*pragram by mangoyang*/
#include<cstdio>
#include<cstring>
#include<vector>
#include<ctype.h>
#include<algorithm>
#define inf (0x7f7f7f7f)
#define Max(a, b) ((a) > (b) ? (a) : (b))
#define Min(a, b) ((a) < (b) ? (a) : (b))
typedef long long ll;
using namespace std;
template <class T>
inline void read(T &x){
    int f = 0, ch = 0; x = 0;
    for(; !isdigit(ch); ch = getchar()) if(ch == '-') f = 1;
    for(; isdigit(ch); ch = getchar()) x = x * 10 + ch - 48;
    if(f) x = -x;
}
int buff[40];
template<class M> inline void putd(M x){
    int p = 0;
    if(x < 0){ putchar('-'); x = -x; }
    do{
        buff[p++] = x % 10;
        x /= 10;
    }while(x);
    for(int i = p - 1; i >= 0; i--) putchar(buff[i] + '0');
    putchar('\n');
}
const int N = 3500005, MAXN = 1500005;
char s[N]; int rt[N], buf[N], n;
struct SegmentTree{
	int lc[MAXN*25], rc[MAXN*25], sz[MAXN*25], size;
	inline SegmentTree(){ size = 1; }
	inline void ins(int &u, int l, int r, int pos){
		if(!u) u = ++size;
		if(l == r) return (void) (sz[u]++);
		int mid = l + r >> 1; 
		if(pos <= mid) ins(lc[u], l, mid, pos);
		else ins(rc[u], mid + 1, r, pos); 
		sz[u] = sz[lc[u]] + sz[rc[u]];
	}
	inline int merge(int x, int y, int l, int r){
		if(!x || !y) return x + y;
		int mid = l + r >> 1, o = ++size;
		if(l == r) sz[o] = sz[x] + sz[y];
		else{
			lc[o] = merge(lc[x], lc[y], l, mid);
			rc[o] = merge(rc[x], rc[y], mid + 1, r);
			sz[o] = sz[lc[o]] + sz[rc[o]];
		}
		return o;
	}
	inline int query(int u, int l, int r, int pos){
		if(!sz[u]) return 0;
		if(l == r) return l;
		int mid = l + r >> 1;
		if(pos <= mid) return query(lc[u], l, mid, pos);
		int rans = query(rc[u], mid + 1, r, pos);
		return rans ? rans : query(lc[u], l, mid, pos);
	}
}Seg;
struct SuffixAutomaton{
	vector<int> g[N], v; ll dep[N];
	int ch[N][26], fa[N], vis[N], mx[N], mn[N], tail, size;
	inline SuffixAutomaton(){ size = tail = 1, rt[1] = 1; }
	inline int newnode(int x){ return dep[++size] = x, size; }
    inline void ins(int c, int ff, int pos){
        int p = tail, np = newnode(dep[p] + 1);
        if(ff) v.push_back(np); else{
        	Seg.ins(rt[np], 1, n, pos);
        	mx[np] = mn[np] = pos;
        }
        for(; p && !ch[p][c]; p = fa[p]) ch[p][c] = np;
        if(!p) return (void) (fa[np] = 1, tail = np);
        int q = ch[p][c];
        if(dep[q] == dep[p] + 1) fa[np] = q;
        else{
            int nq = newnode(dep[p] + 1);
            fa[nq] = fa[q], fa[np] = fa[q] = nq;
            if(ff) rt[nq] = rt[q], mx[nq] = mx[q], mn[nq] = mn[q];
            for(int i = 0; i < 26; i++) ch[nq][i] = ch[q][i];
            for(; p && ch[p][c] == q; p = fa[p]) ch[p][c] = nq;	
        }tail = np;
    }
	inline void addedge(){
		for(int i = 2; i <= size; i++) g[fa[i]].push_back(i);
	}
	inline void dfs(int u){
		for(int i = 0; i < g[u].size(); i++){
			int v = g[u][i];
			dfs(v), rt[u] = Seg.merge(rt[u], rt[v], 1, n);	
			mx[u] = Max(mx[u], mx[v]), mn[u] = Min(mn[u], mn[v]);
		}
	}
	inline void prepare(char *s){
		for(int i = 0; i < n; i++) ins(s[i] - 'a', 0, i + 1);
		addedge(), dfs(1);
	}	
	inline ll calc(char *s, int l, int r){
		tail = 1; v.clear(); 
		ll len = strlen(s), ans = 0, all = 0;
		for(int i = 0; i < len; i++) ins(s[i] - 'a', 1, 0);
		for(int i = 0; i < v.size(); i++){
			int u = v[i];
			for(int p = u; p > 1; p = fa[p]) {
				if(vis[p]) break; int OK = 0;
				all += dep[p] - dep[fa[p]], vis[p] = 1;
				if(rt[p]){
					if((l == 1 && r == n) || OK)
						{ ans += dep[p] - dep[fa[p]]; continue; }
					if(mx[p] < l || mn[p] > r) continue;
					int mxlen = mx[p] <= r ? mx[p] - l + 1 : Seg.query(rt[p], 1, n, r) - l + 1;
					if(mxlen > dep[fa[p]]) 
						ans += Min(dep[p], mxlen) - dep[fa[p]];
					if(mxlen >= dep[p]) OK = 1;
				}
			}
		}
		for(int i = 0; i < v.size(); i++){
			int u = v[i];
			for(int p = u; p > 1; p = fa[p]){
				if(!vis[p]) break; vis[p] = 0;
			}
		}
		return all - ans;
	}	
}van;
int main(){
	scanf("%s", s); n = strlen(s);
	int Q; read(Q), van.prepare(s);
	while(Q--){
		int l, r;
		scanf("%s", s), read(l), read(r);
		putd(van.calc(s, l, r));
	}
}

CompilationN/AN/ACompile OKScore: N/A

Testcase #118.025 ms87 MB + 700 KBAcceptedScore: 4

Testcase #218.069 ms88 MB + 72 KBAcceptedScore: 4

Testcase #318.386 ms88 MB + 368 KBAcceptedScore: 4

Testcase #4135.213 ms178 MB + 544 KBAcceptedScore: 4

Testcase #5131.09 ms175 MB + 656 KBAcceptedScore: 4

Testcase #6605.904 ms509 MB + 252 KBAcceptedScore: 4

Testcase #7606.15 ms509 MB + 264 KBAcceptedScore: 4

Testcase #8173.151 ms170 MB + 588 KBAcceptedScore: 4

Testcase #9112.205 ms167 MB + 300 KBAcceptedScore: 4

Testcase #10382.212 ms262 MB + 532 KBAcceptedScore: 4

Testcase #11251.789 ms262 MB + 848 KBAcceptedScore: 4

Testcase #12660.777 ms359 MB + 124 KBAcceptedScore: 4

Testcase #13415.806 ms359 MB + 780 KBAcceptedScore: 4

Testcase #14908.479 ms457 MB + 788 KBAcceptedScore: 4

Testcase #15595.116 ms456 MB + 776 KBAcceptedScore: 4

Testcase #161.169 s556 MB + 748 KBAcceptedScore: 4

Testcase #17780.939 ms551 MB + 612 KBAcceptedScore: 4

Testcase #181.301 s363 MB + 76 KBAcceptedScore: 4

Testcase #191.4 s426 MB + 308 KBAcceptedScore: 4

Testcase #201.684 s491 MB + 156 KBAcceptedScore: 4

Testcase #211.657 s556 MB + 728 KBAcceptedScore: 4

Testcase #221.976 s556 MB + 924 KBAcceptedScore: 4

Testcase #231.933 s556 MB + 468 KBAcceptedScore: 4

Testcase #241.756 s557 MB + 28 KBAcceptedScore: 4

Testcase #251.639 s556 MB + 308 KBAcceptedScore: 4


Judge Duck Online | 评测鸭在线
Server Time: 2025-01-19 02:42:40 | Loaded in 2 ms | Server Status
个人娱乐项目,仅供学习交流使用 | 捐赠