题干:
Ivan had string s consisting of small English letters. However, his friend Julia decided to make fun of him and hid the string s. Ivan preferred making a new string to finding the old one.
Ivan knows some information about the string s. Namely, he remembers, that string ti occurs in string s at least ki times or more, he also remembers exactly ki positions where the string ti occurs in string s: these positions are xi, 1, xi, 2, ..., xi, ki. He remembers n such strings ti.
You are to reconstruct lexicographically minimal string s such that it fits all the information Ivan remembers. Strings ti and string s consist of small English letters only.
Input
The first line contains single integer n (1 ≤ n ≤ 105) — the number of strings Ivan remembers.
The next n lines contain information about the strings. The i-th of these lines contains non-empty string ti, then positive integer ki, which equal to the number of times the string ti occurs in string s, and then ki distinct positive integers xi, 1, xi, 2, ..., xi, ki in increasing order — positions, in which occurrences of the string ti in the string s start. It is guaranteed that the sum of lengths of strings ti doesn't exceed 106, 1 ≤ xi, j ≤ 106, 1 ≤ ki ≤ 106, and the sum of all ki doesn't exceed 106. The strings ti can coincide.
It is guaranteed that the input data is not self-contradictory, and thus at least one answer always exists.
Output
Print lexicographically minimal string that fits all the information Ivan remembers.
Examples
Input
3 a 4 1 3 5 7 ab 2 1 5 ca 1 4
Output
abacaba
Input
1 a 1 3
Output
aaa
Input
3 ab 1 1 aba 1 3 ab 2 3 5
Output
ababab
题目大意:
有一个字符串,最长1e6,现在给你若干个它的子字符串,每个子字符串长度不超过1e6,最多给你1e5个子字符串,告诉你第i个子字符串出现了ki次,并告诉你它每次出现的第一个字符的位置。所有子字符串的长度之和不超过1e6,出现的次数总和也不超过1e6。然后问你最短的满足要求的字符串中字典序最小的那个是什么。
输入是这样的:
第一行一个数n,代表有n个子字符串。
接下来n行,第i行 先是一个子字符串,然后一个数ki,然后ki个数,每个数代表第i个子字符串的第一个数分别出现的位置。
输入合法。
解题报告:
因为构造出的字符串最多1e6且输入合法,所以直接在每个位置填充就可以了,当然暴力填充的话可能会有很多重复填充,所以用并查集维护一下第i个字符后面第一个空位。
AC代码:
#include<cstdio>
#include<iostream>
#include<algorithm>
#include<queue>
#include<stack>
#include<map>
#include<vector>
#include<set>
#include<string>
#include<cmath>
#include<cstring>
#define FF first
#define SS second
#define ll long long
#define pb push_back
#define pm make_pair
using namespace std;
typedef pair<int,int> PII;
const int MAX = 5e6 + 5;
int mx,f[MAX];
char s[MAX],ans[MAX];
int getf(int v) {return f[v] == v ? v : f[v] = getf(f[v]);
}
int main()
{int n;cin>>n;for(int i = 1; i<MAX; i++) f[i] = i,ans[i] = 'a';for(int num,x,len,i = 1; i<=n; i++) {scanf("%s%d",s+1,&num);len=strlen(s+1);for(int j = 1; j<=num; j++) {scanf("%d",&x);mx = max(mx,x+len-1);for(int k = x; k<=x+len-1; k=getf(k+1)) {ans[k] = s[k-x+1];f[k] = k+1;//=f[k+1]或者getf(k+1)其实都可以,只要记得在使用的时候getf一下就好,这题是巧了k就是根节点了,所以可以这样操作。也就是并查集使用的时候要保证,操作的节点必须是根节点,但是不一定接在人家的根节点上。}}} for(int i = 1; i<=mx; i++) printf("%c",ans[i]);return 0 ;
}