Luogu P3294 【[SCOI2016]背单词】
阅读理解题
。。。。。。
$Trie$
后缀问题不好处理,我们把它转化为前缀问题,用字典树解决问题
贪心
容易想到,一个串的后缀要先于它插入
对于一个串和其若干后缀串,容易想到,我们要先插入后缀串
然后递归进入$size$最小的子串
bool cmp(const int &x,const int &y)
{
return size[x]<size[y];
}
void makes(int x)
{
size[x]=;
for(int i=;i<t[x].size();i++)
{
makes(t[x][i]);
size[x]+=size[t[x][i]];
}
sort(t[x].begin(),t[x].end(),cmp);
}
void dfs(int x)
{
id[x]=tot++;
for(int i=;i<t[x].size();i++)
{
ans+=tot-id[x];
dfs(t[x][i]);
}
}
注意
求$size$要重构树,只保留关键点
因为我太蒻了,并不会指针,所以提供一个并查集重构树的方法
在建$Trie$时给所有串的结尾和$Trie$树的根节点标号,表示新树中点的编号
void insert(const string &s,int id)
{
int now=,l=len[id];
for(int i=;i<l;i++)
{
int c=idx(s[i]);
now=tr[now][c]?tr[now][c]:tr[now][c]=++cnt;
}
val[now]=id;
}
然后遍历$Trie$树,如果一个节点的子节点没有被标号,就把它并入当前节点的集合;否则把这个子节点作为当前节点所在集合的根的儿子(就是连一条边)
void make(int x)
{
for(int v,i=;i<;i++)
if(v=tr[x][i])
{
if(!val[v])
f[v]=find(x);
else
t[val[find(x)]].push_back(val[v]);
make(v);
}
}
代码:
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<vector>
#define int long long
using namespace std;
const int maxl=,maxn=1e5+;
int n,tr[maxl][],val[maxl],cnt,len[maxn],size[maxn],tot,f[maxl],id[maxn],ans;
vector<int>t[maxn];
string st[maxn];
inline int find(int x)
{
return x==f[x]?x:f[x]=find(f[x]);
}
inline int idx(char c)
{
return c-'a';
}
void insert(const string &s,int id)
{
int now=,l=len[id];
for(int i=;i<l;i++)
{
int c=idx(s[i]);
now=tr[now][c]?tr[now][c]:tr[now][c]=++cnt;
}
val[now]=id;
}
void make(int x)
{
for(int v,i=;i<;i++)
if(v=tr[x][i])
{
if(!val[v])
f[v]=find(x);
else
t[val[find(x)]].push_back(val[v]);
make(v);
}
}
bool cmp(const int &x,const int &y)
{
return size[x]<size[y];
}
void makes(int x)
{
size[x]=;
for(int i=;i<t[x].size();i++)
{
makes(t[x][i]);
size[x]+=size[t[x][i]];
}
sort(t[x].begin(),t[x].end(),cmp);
}
void dfs(int x)
{
id[x]=tot++;
for(int i=;i<t[x].size();i++)
{
ans+=tot-id[x];
dfs(t[x][i]);
}
}
signed main()
{
scanf("%lld",&n);
for(int i=;i<=n;i++)
{
cin>>st[i];
len[i]=st[i].length();
for(int j=;j<len[i]/;j++)
swap(st[i][j],st[i][len[i]-j-]);
insert(st[i],i);
}
for(int i=;i<=cnt;i++)
f[i]=i;
make(),makes(),dfs();
printf("%lld\n",ans);
return ;
}
Luogu P3294 【[SCOI2016]背单词】的更多相关文章
- [luogu] P3294 [SCOI2016]背单词 (贪心)
题目描述 Lweb 面对如山的英语单词,陷入了深深的沉思,"我怎么样才能快点学完,然后去玩三国杀呢?".这时候睿智的凤老师从远处飘来,他送给了 Lweb 一本计划册和一大缸泡椒,他 ...
- P3294 [SCOI2016]背单词
P3294 [SCOI2016]背单词 Trie+贪心 倒插进树+取出重建+子树处理+贪心遍历 倒插进树:把后缀转化为前缀,所以把字符串倒着插进Trie中 取出重建:重新建立一棵以单词为节点的树,如果 ...
- 洛谷P3294 [SCOI2016]背单词——题解
题目传送 阅读理解题题意解释可以看这位大佬的博客. 发现求后缀与倒序求前缀是等价的,而找前缀自然就想到了trie树.将所有字符串翻转后再建入trie树中,再对每一个字符串翻转后从trie树中找前缀,就 ...
- BZOJ4567[Scoi2016]背单词
4567: [Scoi2016]背单词 Time Limit: 10 Sec Memory Limit: 256 MB Submit: 304 Solved: 114 [Submit][Status] ...
- 4567: [Scoi2016]背单词
4567: [Scoi2016]背单词 https://www.lydsy.com/JudgeOnline/problem.php?id=4567 题意: 题意看了好久,最后在其他人的博客里看懂了的. ...
- 【BZOJ4567】[Scoi2016]背单词 Trie树+贪心
[BZOJ4567][Scoi2016]背单词 Description Lweb 面对如山的英语单词,陷入了深深的沉思,“我怎么样才能快点学完,然后去玩三国杀呢?”.这时候睿智 的凤老师从远处飘来,他 ...
- 【bzoj4567】[Scoi2016]背单词
4567: [Scoi2016]背单词 Time Limit: 10 Sec Memory Limit: 256 MBSubmit: 1123 Solved: 476[Submit][Status][ ...
- [SCOI2016]背单词 题解
背单词 https://www.luogu.com.cn/problem/P3294 前言: Trie树的省选题(瑟瑟发抖QAQ) 问题汇总:(请忽略) (1)对Trie字典树的运用不熟练 (2)没想 ...
- [SCOI2016]背单词——trie树相关
题目描述 Lweb 面对如山的英语单词,陷入了深深的沉思,”我怎么样才能快点学完,然后去玩三国杀呢?“.这时候睿智的凤老师从远处飘来,他送给了 Lweb 一本计划册和一大缸泡椒,他的计划册是长这样的: ...
随机推荐
- marshaller unmarshaller解析xml和读取xml
JAXB(Java Architecture for XML Binding) 是一个业界的标准,是一项可以根据XML Schema产生Java类的技术.该过程中,JAXB也提供了将XML实例文档反向 ...
- SpringBoot使用redis缓存List<Object>
一.概述 最近在做性能优化,之前有一个业务是这样实现的: 1.温度报警后第三方通讯管理机直接把报警信息保存到数据库 2.我们在数据库中添加触发器,(BEFORE INSERT)根据这条报警信息处理业务 ...
- JVM总结(三):类文件结构
这一节我们来总结一下类文件结构方面的知识.目录如下: 类文件结构 字节码的意义 Class类文件的结构 Class类文件的存储形式 Class文件的格式 Class类文件结构详解 举例详解 一.写程序 ...
- Java实现DOS中的Copy命令
import java.io.*; import java.util.Scanner; public class fileCopy { public static void main(String [ ...
- 《Two Dozen Short Lessons in Haskell》所有习题的索引
<Two Dozen Short Lessons in Haskell>(Copyright © 1995, 1996, 1997 by Rex Page,有人翻译为Haskell二十四学 ...
- form表单提交onclick和onsubmit
onsubmit只能表单上使用,提交表单前会触发, onclick是按钮等控件使用, 用来触发点击事件. 在提交表单前,一般都会进行数据验证,可以选择在submit按钮上的onclick中验证,也可以 ...
- XQuartz简介
这是一个类似于中转的软件,比如现在在Mac上,YY语音还没有官方版的,但其实在Mac上,有了XQuartz就可以实现运行YY了,下载这个从Windows上移植过来的软件,然后打开的时候,Mac会提醒你 ...
- string中substr,find函数使用
2.string函数 find:某子串的起始位(0开始),函数的第二个参数使用代表从该位开始的后缀 substr:1) x开始的连续y位 2) x开始的后缀 #include<bits/stdc ...
- JavaJavaScript之内存与变量初始化
0.搞清三个概念:预加载与执行期:js变量存储(栈区与堆区):js变量的类型(引用类型(对象)与基本数据类型); JS在预编译时,对于函数的预加载方面,浏览器仅仅选择编译声明式函数(function ...
- [C++]指针和指向数组的指针[一维数组与指针]
1.一维数组与指针 形如:int型 数组 a[10] 1)&a[0] 地址常量;地址类型:int *型 ; 存储数组a的首地址 ...