描述

给定一个包含N个单词的字典:{W1, W2, W3, ... WN},其中第i个单词Wi有具有一个权值Vi。

现在小Hi要进行M次查询,每次查询包含一个前缀字符串Pi和一个后缀字符串Si。他希望知道同时以Pi为前缀并且以Si为后缀的单词中,权值最大的单词的权值是多少?

假设字典包含"hihocoder"、"hijacker"和"hiker",权值依次是30、20和10。

那么对于查询前缀="hi",后缀="er",答案应为30.

输入

第一行包两个整数N和M。(1 <= N <= M)

以下N行每行包含一个只包含小写字母的字符串Wi和对应的权值Vi。

再之后M行每行包含两个字符串Pi和Si。

对于30%的数据,1 <= N, M <= 100

对于100%的数据,1 <= N, M <= 50000, 1 <= |Wi|, |Pi|, |Si| <= 10, 1 <= Vi <= 100000

输出

输出最大的权值。如果没有符合条件的单词,输出-1。

样例输入

3 2
hihocoder 30
hijacker 20
hiker 10
hi er
hihoco hocoder

样例输出

30
30

初始代码:

思路是前缀查找,找到到Now节点,在Now的每个点(需要是单词尾)向前验证是否满足后缀要求,满足则更新最大值,无则输出-1。

这样显然是超时了,保证试一试的态度写了下。

#include<cstdio>
#include<cstdlib>
#include<iostream>
#include<algorithm>
#include<cstring>
#include<cmath>
#include<memory>
using namespace std;
const int maxn=; char s[maxn];
int ch[maxn][],fa[maxn],id[maxn],num[maxn],cnt,val;
int max(int a,int b){ if(a>b) return a; return b; }
void insert()
{
int Now=,L=strlen(s),x;
for(int i=;i<L;i++){
x=s[i]-'a';
if(!ch[Now][x]) ch[Now][x]=++cnt,fa[cnt]=Now,id[cnt]=x;
Now=ch[Now][x];
}
num[Now]=max(num[Now],val);
}
int Max; char a[],b[];
bool check(int Now)
{
int L2=strlen(b);
for(int i=L2-;i>=;i--){
if(Now==) return false;
if(b[i]-'a'!=id[Now]) return false;
Now=fa[Now];
} return true;
}
void dfs(int Now)
{
if(num[Now]) if(check(Now)) Max=max(Max,num[Now]);
for(int i=;i<;i++) {
if(ch[Now][i])
dfs(ch[Now][i]);
}
}
void query()
{
int Now=,L1=strlen(a),x;
for(int i=;i<L1;i++){
x=a[i]-'a';
if(!ch[Now][x]) { printf("-1\n");return ;}
Now=ch[Now][x];
}
Max=-; dfs(Now);
printf("%d\n",Max);
}
int main()
{
int n,m,i,L1,L2;
scanf("%d%d",&n,&m);
for(i=;i<=n;i++){
scanf("%s%d",s,&val);
insert();
}
for(i=;i<=m;i++){
scanf("%s%s",a,b);
L1=strlen(a);L2=strlen(b);
query();
}
return ;
}

改进:

既然对象是前缀和后缀,显然不需要后缀数组或者后缀自动机(二者对象是任意子串)来完成。还是考虑字典树,想个办法把后缀前缀一起查询了。

具体:

对单词abcd (权值为Val): 它的后缀是d,cd,bcd,abcd。把它的后缀连接到前缀上,成为4个单词(其他的最多也才10个),用'#'连接(后缀数组唱用)

对应     :   d#abcd,cd#abcd,bcd#abcd,abcd#abcd,权值都为Val。

#include<cstdio>
#include<cstdlib>
#include<iostream>
#include<algorithm>
#include<cstring>
#include<cmath>
#include<memory>
using namespace std;
const int maxn=; char s[maxn];
int ch[maxn<<][],num[maxn<<],cnt,Val;
int max(int a,int b){ if(a>b) return a; return b; }
void insert()
{
int L=strlen(s),x;
for(int j=L-;j>=;j--){
int Now=;
for(int i=j;i<L;i++){
x=s[i]-'a';
if(!ch[Now][x]) ch[Now][x]=++cnt;
Now=ch[Now][x];
}
x=;
if(!ch[Now][x]) ch[Now][x]=++cnt;
Now=ch[Now][x];
for(int i=;i<L;i++){
x=s[i]-'a';
if(!ch[Now][x]) ch[Now][x]=++cnt;
Now=ch[Now][x];
num[Now]=max(num[Now],Val);
}
}
}
char a[],b[];
void query()
{
int Now=,L1=strlen(a),L2=strlen(b),x;
for(int i=;i<L2;i++){
x=b[i]-'a';
if(!ch[Now][x]) { printf("-1\n");return ;}
Now=ch[Now][x];
}
Now=ch[Now][];if(!Now) { printf("-1\n");return ;}
for(int i=;i<L1;i++){
x=a[i]-'a';
if(!ch[Now][x]) { printf("-1\n");return ;}
Now=ch[Now][x];
}
printf("%d\n",num[Now]);
}
int main()
{
int n,m,L1,L2;
scanf("%d%d",&n,&m);
for(int i=;i<=n;i++){
scanf("%s%d",s,&Val);
insert();
}
for(int i=;i<=m;i++){
scanf("%s%s",a,b);
query();
}
return ;
}

HihoCoder1656 : 前缀后缀查询([Offer收割]编程练习赛39)(字典树+小技巧)的更多相关文章

  1. [Offer收割]编程练习赛39

    公平分队 #pragma comment(linker, "/STACK:102400000,102400000") #include<stdio.h> #includ ...

  2. HihoCoder1665方块游戏([Offer收割]编程练习赛40)(线段树)

    时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 小Ho在玩一款类似俄罗斯方块的游戏.与原版俄罗斯方块不同的是,落下方块都是长度不一的横向长条,并且不能移动也不能变成竖直方 ...

  3. HihoCoder1655 : 第K小最简真分数([Offer收割]编程练习赛39)(唯一分解+容斥定理+二分)(不错的数学题)

    描述 给一个整数N,请你求出以N为分母的最简(既约)真分数中第K小的是多少? 输入 两个整数N个K. 对于30%的数据,1 <= N <= 1000000 对于100%的数据,1 < ...

  4. HihoCoder1654: XY游戏([Offer收割]编程练习赛39)(好久没写搜索)(已经超级简短了)

    描述 如下图所示,在4x4的棋盘上有X和Y两种棋子各若干枚:O表示空格. OXXY YOOX XOOY XOXX 小Hi每次可以选择任意一枚棋子,将它移动到上下左右相邻的空格中. 小Hi想知道最少移动 ...

  5. HihoCoder1653 : 公平分队([Offer收割]编程练习赛39)(贪心)

    描述 小Hi和小Ho在玩一个战争游戏.游戏中2N个战斗单位,其中第i个单位的战斗力是Ai. 现在小Hi和小Ho要各选N个单位组成队伍,当然他们都希望自己队伍的总战斗力越大越好. 为了使分队更加公平,经 ...

  6. 【[Offer收割]编程练习赛 14 A】小Hi和小Ho的礼物

    [题目链接]:http://hihocoder.com/problemset/problem/1505 [题意] [题解] 考虑Meet in the middle. 因为两个数的和不是很大; 直接用 ...

  7. hihocoder [Offer收割]编程练习赛61

    [Offer收割]编程练习赛61 A:最小排列 给定一个长度为m的序列b[1..m],再给定一个n,求一个字典序最小的1~n的排列A,使得b是A的子序列. 贪心即可,b是A的子序列,把不在b中的元素, ...

  8. hihocoder [Offer收割]编程练习赛4

    描述 最近天气炎热,小Ho天天宅在家里叫外卖.他常吃的一家餐馆一共有N道菜品,价格分别是A1, A2, ... AN元.并且如果消费总计满X元,还能享受优惠.小Ho是一个不薅羊毛不舒服斯基的人,他希望 ...

  9. [Offer收割]编程练习赛46

    [Offer收割]编程练习赛46赛后题解 A.AEIOU 分析

随机推荐

  1. 扩展MongoDB C# Driver的QueryBuilder

    扩展MongoDB C# Driver的QueryBuilder 因为不想直接hardcode "ClassA.MemberA.MemberB" 这种字符串 .写了下面几个类,用于 ...

  2. A008-drawable资源

    关于drawable资源笔者之前有写过两篇文章: Android-自己定义图像资源的使用(1) Android-自己定义图像资源的使用(2) 这里笔者就不做过多的赘述.我们从实际开发的角度去理解这个知 ...

  3. Highcharts使用表格数据绘制图表

    Highcharts使用表格数据绘制图表 在Highcharts中,同意用户使用网页中现有的表格数据作为数据来源,然后依据该数据来源绘制图表.对于一个典型的HTML表格.当中,第一列的数据会作为x轴刻 ...

  4. 关于C++项目指针对象未被初始化的问题(0xcdcdcd)

    http://blog.csdn.net/devfun/article/details/6900086 昨天我试图将一个封装好的模块加入到正在开发的项目中,这个模块不是单独的类,而且对应的声明和实例. ...

  5. Delphi列表控件TListView定位到某一行。

    ListView1.Item[100].Focused = true; //定位到索引为100的行ListView1.Item[100].Selected = true; ListView1.Item ...

  6. Android API Guides---Supporting Tablets and Handsets

    在Android平台上的各种屏幕尺寸的执行和系统调整大小正常应用程序的用户界面.以适应每一个人. 通常情况下,你须要做的是设计你的UI是灵活的,并通过提供替代资源(如又一次定位的一些看法观点或替代尺寸 ...

  7. javaScript 深层复制

    在工作中遇到了深浅复制的问题,所以详细总结一下: 深复制和浅复制只针对像 Object, Array 这样的复杂对象的.简单来说,浅复制只复制一层对象的属性,而深复制则递归复制了所有层级. var o ...

  8. 【Unity 3D】学习笔记三十:游戏元素——游戏地形

    游戏地形 在游戏的世界中,必然会有非常多丰富多彩的游戏元素融合当中. 它们种类繁多.作用也不大同样.一般对于游戏元素可分为两种:经经常使用.不经经常使用.经常使用的元素是游戏中比較重要的元素.一般须要 ...

  9. RTSP Windows专用播放器EasyPlayer : 稳定、兼容、高效、超低延时

    EasyPlayer RTSP Windows专用播放器 EasyPlayer RTSP Windows 播放器是由EasyDarwin团队开发和维护的一个完善的RTSP流媒体播放器项目,视频编码支持 ...

  10. EasyDSS高性能流媒体服务器前端重构(五)- webpack + vue-router 开发单页面前端实现按需加载 - 副本

    为了让页面更快完成加载, 第一时间呈现给客户端, 也为了帮助客户端节省流量资源, 我们可以开启 vue-router 提供的按需加载功能, 让客户端打开页面时, 只自动加载必要的资源文件, 当客户端操 ...