QOJ.ac
QOJ
ID | 题目 | 提交者 | 结果 | 用时 | 内存 | 语言 | 文件大小 | 提交时间 | 测评时间 |
---|---|---|---|---|---|---|---|---|---|
#597470 | #9434. Italian Cuisine | ucup-team5071# | WA | 0ms | 38744kb | C++20 | 5.2kb | 2024-09-28 17:53:12 | 2024-09-28 17:53:13 |
Judging History
answer
#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N = 1.5e6 + 10;//2*strlen
struct Suffix{
int ht[N],rk[N],sa[N],y[N],c[N];
int n,m;
char s[N];
int st[22][N];
void init(){
n=strlen(s+1);
m=300;
for(int i=0;i<=m;i++) c[i]=0;
for(int i=0;i<=2*n;i++) y[i]=0;
for(int i=1;i<=n;i++) c[rk[i]=s[i]]++;
for(int i=1;i<=m;i++) c[i]+=c[i-1];
for(int i=n;i>=1;i--) sa[c[rk[i]]--]=i;
for(int k=1;k<=n;k<<=1){
int p=0;
for(int i=n-k+1;i<=n;i++) y[++p]=i;
for(int i=1;i<=n;i++){
if(sa[i]>k){
y[++p]=sa[i]-k;
}
}
for(int i=0;i<=m;i++) c[i]=0;
for(int i=1;i<=n;i++) c[rk[i]]++;
for(int i=1;i<=m;i++) c[i]+=c[i-1];
for(int i=n;i>=1;i--) sa[c[rk[y[i]]]--]=y[i];
for(int i=0;i<=n;i++) swap(rk[i],y[i]);
rk[sa[1]]=p=1;
for(int i=2;i<=n;i++){
rk[sa[i]]=(y[sa[i]] == y[sa[i-1]] && y[sa[i]+k] == y[sa[i-1]+k] ? p : ++p);
}
if(p>=n) break;
m=p;
}
for(int i=1,k=0;i<=n;i++){
if(k)k--;
int j=sa[rk[i]-1];
while(s[i+k] == s[j+k] )k++;
ht[rk[i]] = k;
}
for(int i=1;i<=n;i++)st[0][i]=ht[i];
for(int j=1;j<22;j++){
for(int i=1;i+(1<<j)-1<=n;i++)st[j][i]=min(st[j-1][i],st[j-1][i+(1<<(j-1))]);
}
}
int get(int l,int r){
int g=__lg(r-l+1);
return min(st[g][l],st[g][r-(1<<g)+1]);
}
int lcp(int x,int y){
x=rk[x],y=rk[y];
if(x==y)return n-x+1;
if(x>y)swap(x,y);
return get(x+1,y);
}
int query(int l1,int r1,int l2,int r2){
int len=lcp(l1,l2);
len=min({len,r1-l1+1,r2-l2+1});
if(len==min(r1-l1+1,r2-l2+1)){
if(r1-l1>r2-l2)return 1;
if(r1-l1<r2-l2)return -1;
if(r1-l1==r2-l2)return 0;
}
char p=s[l1+len],q=s[l2+len];
if(p>q)return 1;
if(p==q)return 0;
return -1;
}
void writ()
{
printf("%s\n",s+1);
for(int i=1;i<=n;i++)cout<<sa[i]<<" ";;cout<<"\n";
for(int i=1;i<=n;i++)cout<<ht[i]<<" ";;cout<<"\n";
for(int i=1;i<=n;i++)cout<<rk[i]<<" ";;cout<<"\n";
}
};
Suffix suf;
struct Trie {
int tree[N][26],tot=0;
int e[N],id[N],sum[N],dep[N],dfn[N];
int now=0;
Trie(){
memset(e,0,sizeof(e));
memset(id,-1,sizeof(id));
}
void clear(){
for(int i=1;i<=tot;i++){
memset(tree[i],0,sizeof(tree[i]));
e[i]=0;
id[i]=-1;
sum[i]=0;
dep[i]=0;
}
tot=0;now=0;
}
int insert(string &t,int idd,int v){
int x=0;
for(int i=0;i<t.size();i++){
if(!tree[x][t[i]-'a']){
tree[x][t[i]-'a']=++tot;
dep[tot]=dep[x]+1;
}
x=tree[x][t[i]-'a'];
sum[x]+=v;
}
e[x]+=v;
if(id[x]=-1&&v>0)id[x]=idd;
return dfn[x];
}
void dfs(int x){
dfn[x]=++now;
for(int i=0;i<26;i++)if(tree[x][i])dfs(tree[x][i]);
}
};
Trie tri;
ll solve()
{
int n;cin>>n;
tri.clear();
vector<string>s(n);
vector<int> l(n),r(n);
ll ans1=0,ans2=0,ans3=0,ans4=0;
int now=0;
vector<int> id_tree(n);
for(int i=0;i<n;i++){
cin>>s[i];
id_tree[i]=tri.insert(s[i],i,0);
l[i]=now+1;
for(auto &it:s[i]){
now++;
suf.s[now]=it;
}
r[i]=now;
suf.s[++now]='#';
}
tri.dfs(0);
for(int i=0;i<n;i++)id_tree[i]=tri.insert(s[i],i,0);
vector<int> id(n);iota(id.begin(),id.end(),0);
sort(id.begin(),id.end(),[&](int x,int y){
return id_tree[x]<id_tree[y];
});
for(int i=0;i<n;i++)tri.insert(s[id[i]],i,1);
suf.s[now+1]=0;
suf.init();
suf.writ();
int havvis=0;
auto dfs = [&](auto&& self,int x,vector<int> &lis) -> void {
if(tri.e[x]){
for(auto it:lis){
int rlen=tri.dep[x]-tri.dep[it];
int rL=r[id[tri.id[x]]]-rlen+1,rR=r[id[tri.id[x]]];
if(suf.query(l[id[tri.id[it]]],r[id[tri.id[it]]],rL,rR)){
ans2+=tri.e[it];
}
int ql=0,qr=tri.id[x]-1,res=-1;
while(ql<=qr){
int mid=(ql+qr)/2;
if(suf.query(l[id[mid]],r[id[mid]],rL,rR)>0){
res=mid;ql=mid+1;
}
else qr=mid-1;
}
ans1+=(ll)(res+1)*(tri.e[it]);
}
lis.push_back(x);
if(tri.e[x]>=3)ans3+=(ll)tri.e[x]*(tri.e[x]-1)*(tri.e[x]-2)/6;
ans4+=(ll)havvis*tri.e[x]*(tri.e[x]-1)/2;
}
havvis+=tri.e[x];
for(int i=0;i<26;i++)if(tri.tree[x][i])self(self,tri.tree[x][i],lis);
if(tri.e[x])lis.pop_back();
};
vector<int>lis;
dfs(dfs,0,lis);
cout<<"ans1="<<ans1<<" ans2"<<ans2<<" ans3="<<ans3<<" ans4="<<ans4<<endl;
return (ans1-ans2)/2+ans2+ans3+ans4;
}
int main()
{
ios::sync_with_stdio(false);
cin.tie(0);
int T;cin>>T;
while(T--)cout<<solve()<<"\n";
return 0;
}
詳細信息
Test #1:
score: 0
Wrong Answer
time: 0ms
memory: 38744kb
input:
3 5 1 1 1 0 0 1 0 5 0 3 3 0 5 6 2 4 1 2 0 4 0 6 3 4 6 2 6 0 3 4 3 3 1 3 0 6 3 3 6 0 3
output:
10 8 6 4 2 9 7 5 3 1 0 1 3 1 3 0 2 0 2 4 10 5 9 4 8 3 7 2 6 1 ans1=0 ans20 ans3=0 ans4=0 0 2 1 0 0 2 1 ans1=0 ans20 ans3=0 ans4=0 0 10 6 4 2 8 1 7 5 3 9 0 1 1 3 1 0 2 0 2 0 6 4 9 3 8 2 7 5 10 1 ans1=0 ans20 ans3=0 ans4=0 0 1#1#1#0#0# 0# 0#3#3#0#5#
result:
wrong answer 1st numbers differ - expected: '5', found: '10'