P4407 [JSOI2009] 电子字典

题目大意

详细题目传送门

给出 \(n\) 个互不相同字典串 \(S_i\)。和 \(m\) 个匹配串 \(Q_i\)。如果有字典串 \(S_i=Q_i\),输出 \(-1\)。三种变换操作:

  1. \(S_{i,j}\) 后添加任意一个字符
  2. 删除 \(S_{i,j}\)
  3. \(S_{i,j}\) 改成任意一个字符。

求每一个匹配串如果只进行 \(1\)操作有几个可以匹配的字典串。

\(n,m\leq 10^4\)

\(|S_i|,|Q_i|\leq 20\)

所有字符串都只有小写字母。

思路

考虑字典树。

发现时间复杂度猜一下是 \(O(26m|Q|)\) 的。其实发现哈希应该就能做,但是作者考场上没有写出来哈希,于是在这里补一个字典树做法。

先将所有字典串插入字典树,并记录结尾结点。对于每一个匹配串考虑深搜,用 \(f(u,l,c)\) 表示当前考虑到了字典树的 \(u\) 节点,是第 \(l\) 位,\(c\) 表示是否已经修改过。

如果已经修改过就一直到尾去记录是不是一个字典串。

之后对于三种操作:

添加任意一个字符就是对于 \(u\),进入到字符 \(x\) 中也就是搜索 \(f(t_{u,x},l,\text{true})\)。修改操作就是 \(f(t_{u,x},l+1,\text{true})\)。如果还能删就到 \(f_{t_{u,Q_{i,l}},l+1,\text{true}}\)。当然也可以选择这一位不修改。

代码

#include <bits/stdc++.h>
#define endl "\n"
using namespace std;
typedef long long ll;
const ll MAXN=1e5+5;
ll trie[MAXN*20][26],tot,ed[MAXN*20];
void insert(string s){ll u=0;for(auto c:s){int x=c-'a';if(!trie[u][x]){trie[u][x]=++tot;}u=trie[u][x];}ed[u]=1;
}
string s;
bool vis[MAXN];
ll vu[MAXN],ans;
set<string>se;
void dfs(ll u,ll len,bool change){if(len==s.size()&&ed[u]){if(change&&!vis[u]){vu[++ans]=u;vis[u]=true;}return;}if(len>=s.size()){if(!change){for(int i=0;i<26;++i){if(!trie[u][i]){continue;}dfs(trie[u][i],len,true);}}return;}ll x=s[len]-'a';if(trie[u][x]){dfs(trie[u][x],len+1,change);}if(!change){dfs(u,len+1,true);for(int i=0;i<26;++i){if(!trie[u][i]){continue;}dfs(trie[u][i],len,true);dfs(trie[u][i],len+1,true);}}
}
signed main(){ios::sync_with_stdio(false);cin.tie(0);cout.tie(0);ll n,Q;cin>>n>>Q;for(int i=1;i<=n;++i){cin>>s;se.insert(s);insert(s);}while(Q--){cin>>s;if(se.count(s)){cout<<-1<<endl;continue;}ans=0;dfs(0,0,false);if(ans<0){cout<<-1<<endl;continue;}cout<<ans<<endl;while(ans){vis[vu[ans]]=false;ans--;}}return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/855596.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

组合数学+ybt题解

加法原理 乘法原理 排列数 从 \(n\) 个数中任取 \(m\) 个元素的排列的方案数,表示为 \(A^m_n=\frac{n!}{(n-m)!}\) \(0!=1\) 全排列 \(A^n_n\) 组合数 从 \(n\) 个元素中取出 \(m\) 个元素的组合的个数,表示为 \(\dbinom{n}{m}= \frac{A^m_n}{m!}=\frac{n!}{m!(n-m)!}\) 如何…

苍穹外卖day02

JWT令牌、ThreadLocal、分页查询bug记录知识点记录新增员工新增员工需要填写创建人id和修改人id两个属性,这两个属性应该填本账户的id。 通过拦截器可以解析出JWT令牌中包含的登录员工id信息,但是该如何传递给Service的save方法? ThreadLocal并非一个Thread,而是Thread的局…

年底裁员开始了,大家做好准备吧!

各大互联网公司的接连裁员,政策限制的行业接连消失,让今年的求职雪上加霜,想躺平却没有资本,还有人说软件测试岗位饱和了,对此很多求职者深信不疑,因为投出去的简历回复的越来越少了。 另一面企业招人真的变得容易了吗?有企业HR吐槽,简历确实比以前多了好几倍,其实是变…

2024-2025-1 20241401 《计算机基础与程序设计》 第十三周学习总结

班级链接 2024计算机基础与程序设计作业要求 第十三周作业教材学习内容总结 《C语言程序设计》第12章结构体的定义和使用: 结构体类型的定义,以及结构体变量的创建和使用。结构体允许将不同数据类型的成员组合成一个整体,以便于管理和引用。 结构体变量的初始化: 结构体变量…

golang的互斥锁和读写锁

golang mutex(互斥锁) 1.锁最本质的作用 保证原子性 2.mutex 使用原则 适用于并发编程,尽量减少加锁区域的逻辑 3.mutex的局限性 仅限于单个进程内操作 sema(信号量,semaphore的简称)是一种用于并发控制的机制 资源计数:信号量维护一个资源计数。这个计数表示当前可用的资源数…

「测试人员」年终总结及来年规划

年底一般企业都会针对项目写总结,一般流程是组员呈交给组长,组长递交给测试经理,所以每个层级写各自的总结,会有不同的侧重点。 今天分别来聊聊测试工程师、测试组长以及测试经理年终总结的关注点在什么地方以及采用什么样的方法及策略使年终总结有实际意义且高大上。📝 …

【硬件测试】基于FPGA的2FSK调制解调系统开发与硬件片内测试,包含信道模块,误码统计模块,可设置SNR

1.算法仿真效果 本文是之前写的文章基于FPGA的2FSK调制解调系统,包含testbench,高斯信道模块,误码率统计模块,可以设置不同SNR_fpga 2fsk-CSDN博客的硬件测试版本。在系统在仿真版本基础上增加了ila在线数据采集模块,vio在线SNR设置模块,数据源模块。硬件ila测试结果如下:(…

征程 6 云端 Ota 升级方案设计与实现

01 引言 在开发、调测过程中,经常通过 Ota 技术升级应用软件、底软。目前,项目上已将 Matrix6 的 Demo 部署在实车上,每次软件、底软需要版本更替或版本升级时,都需要研发测试同学上到车上使用工控机进行手动升级。 这流程存在耗费人力多,花费时间长,流程易出错的问题,并…

vxe-table 树表格新增数据,插入指定节点位置操作

vxe-table 新增数据,插入指定节点位置操作 官网:https://vxetable.cn npm install vxe-pc-ui@4.3.37 vxe-table@4.9.31// ... import VxeUI from vxe-pc-ui import vxe-pc-ui/lib/style.css import VxeUITable from vxe-table import vxe-table/lib/style.css // ...createAp…

less5

1.在URL处输入 ?id=1 ?id=1 ?id=1" ?id=1/1发现回显只有you are in......和报错 报错:判断回显是布尔类型 使用order by发现回显有三栏 ?id=1 order by 4%23 ?id=1 order by 1%23 ?id=1 order by 3%23判断数据库长度:8位 ?id=1 and length((select database()))&g…

Java-递归查询部门下所有子部门(包括本部门)

Java-递归查询部门下所有子部门(包括本部门),会得到一个部门id的集合:List deptIds具体代码如下: //递归1public List<Long> queryAllSubInstitutionIds(Long institutionId) {List<Long> subInstitutionIds = new ArrayList<>();querySubInstitutionIds…