bzoj1030 [JSOI2007]文本生成器

2024-01-10 02:49

本文主要是介绍bzoj1030 [JSOI2007]文本生成器,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

传送门
Description
  JSOI交给队员ZYX一个任务,编制一个称之为“文本生成器”的电脑软件:该软件的使用者是一些低幼人群,他们现在使用的是GW文本生成器v6版。该软件可以随机生成一些文章―――总是生成一篇长度固定且完全随机的文章—— 也就是说,生成的文章中每个字节都是完全随机的。如果一篇文章中至少包含使用者们了解的一个单词,那么我们说这篇文章是可读的(我们称文章a包含单词b,当且仅当单词b是文章a的子串)。但是,即使按照这样的标准,使用者现在使用的GW文本生成器v6版所生成的文章也是几乎完全不可读的?。ZYX需要指出GW文本生成器 v6生成的所有文本中可读文本的数量,以便能够成功获得v7更新版。你能帮助他吗?
Input
  输入文件的第一行包含两个正整数,分别是使用者了解的单词总数N (<= 60),GW文本生成器 v6生成的文本固定长度M;以下N行,每一行包含一个使用者了解的单词。这里所有单词及文本的长度不会超过100,并且只可能包含英文大写字母A..Z
Output
  一个整数,表示可能的文章总数。只需要知道结果模10007的值。
Sample Input
2 2
A
B
Sample Output
100

题解

看到有字符串的题目首先想到的就是能不能用AC自动机,但是这道题要求出现至少一个字符串的方案数,如果每次都判断是否有一个字符串或者它的一个前缀在生成串中出现过几乎是不可能完成的事情,所以要转化一下思路,变成求不合法的字符串的个数,然后用26的m次方减就好了。这样的题几乎都是dp,所以我们就可以尝试推dp方程了。
如果一个字符串在文本串中出现过,那么如果在AC自动机上看,一定是到达了某个打过结束标记的节点。为了防止这种情况的发生,如果我们用f[i][j]表示在生成的文本串中进行到第i个位置,在AC自动机上匹配到第j个位置,那么只要我们不用带有结束标记的节点更新别的节点就好了。这样dp方程就可以推出来了。
这道题有几个需要注意的地方:一开始要将0号节点的所有儿子都赋为非0的值,否则会出现儿子更新0号节点的情况;还有就是在建立失配指针的时候,如果一个节点的Fail指针指向的节点有结束标记,那么这个点也必须有结束标记,否则就有可能出现某个给定的单词成为某个单词的后缀的情况。

CODE:

#include<queue>
#include<cstdio>
#include<cstring>
using namespace std;
const int M=1e4+7;
struct AC
{int ch[26];int fail;bool isend;
}a[10000];
char s[101];
int f[105][10005];
int n,m,tot,ans;
inline void insert()
{int len=strlen(s),now=0;for(int i=0;i<len;i++){if(!a[now].ch[s[i]-'A']) a[now].ch[s[i]-'A']=++tot;now=a[now].ch[s[i]-'A'];}a[now].isend=1;
}
inline void makefail()
{queue<int>q;for(int i=0;i<26;i++)if(a[0].ch[i]) q.push(a[0].ch[i]);while(!q.empty()){int tmp=q.front();q.pop();for(int i=0;i<26;i++){if(!a[tmp].ch[i]){a[tmp].ch[i]=a[a[tmp].fail].ch[i];continue;}a[a[tmp].ch[i]].fail=a[a[tmp].fail].ch[i];if(a[a[a[tmp].ch[i]].fail].isend) a[a[tmp].ch[i]].isend=1;q.push(a[tmp].ch[i]);}}
}
inline int pow(int a,int b)
{int ans=1;for(;b;b>>=1,a=(1ll*a*a)%M)if(b&1) ans=(1ll*ans*a)%M;return ans;
}
int main()
{scanf("%d%d",&n,&m);for(int i=1;i<=n;i++)scanf("%s",s),insert();for(int i=0;i<26;i++)if(!a[0].ch[i]) a[0].ch[i]=++tot;makefail();f[0][0]=1;for(int i=1;i<=m;i++)for(int j=0;j<=tot;j++)if(!a[j].isend)for(int k=0;k<26;k++)f[i][a[j].ch[k]]=(f[i][a[j].ch[k]]+f[i-1][j])%M;ans=pow(26,m);for(int i=1;i<=tot;i++)if(!a[i].isend) ans=(ans-f[m][i]+M)%M;printf("%d",ans);return 0;
}

总结

一定要考虑到所有的情况;注意转换角度思考;注意细节。

这篇关于bzoj1030 [JSOI2007]文本生成器的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/589312

相关文章

Python中图片与PDF识别文本(OCR)的全面指南

《Python中图片与PDF识别文本(OCR)的全面指南》在数据爆炸时代,80%的企业数据以非结构化形式存在,其中PDF和图像是最主要的载体,本文将深入探索Python中OCR技术如何将这些数字纸张转... 目录一、OCR技术核心原理二、python图像识别四大工具库1. Pytesseract - 经典O

苹果macOS 26 Tahoe主题功能大升级:可定制图标/高亮文本/文件夹颜色

《苹果macOS26Tahoe主题功能大升级:可定制图标/高亮文本/文件夹颜色》在整体系统设计方面,macOS26采用了全新的玻璃质感视觉风格,应用于Dock栏、应用图标以及桌面小部件等多个界面... 科技媒体 MACRumors 昨日(6 月 13 日)发布博文,报道称在 macOS 26 Tahoe 中

Python实现精准提取 PDF中的文本,表格与图片

《Python实现精准提取PDF中的文本,表格与图片》在实际的系统开发中,处理PDF文件不仅限于读取整页文本,还有提取文档中的表格数据,图片或特定区域的内容,下面我们来看看如何使用Python实... 目录安装 python 库提取 PDF 文本内容:获取整页文本与指定区域内容获取页面上的所有文本内容获取

详解如何使用Python从零开始构建文本统计模型

《详解如何使用Python从零开始构建文本统计模型》在自然语言处理领域,词汇表构建是文本预处理的关键环节,本文通过Python代码实践,演示如何从原始文本中提取多尺度特征,并通过动态调整机制构建更精确... 目录一、项目背景与核心思想二、核心代码解析1. 数据加载与预处理2. 多尺度字符统计3. 统计结果可

JavaScript实战:智能密码生成器开发指南

本文通过JavaScript实战开发智能密码生成器,详解如何运用crypto.getRandomValues实现加密级随机密码生成,包含多字符组合、安全强度可视化、易混淆字符排除等企业级功能。学习密码强度检测算法与信息熵计算原理,获取可直接嵌入项目的完整代码,提升Web应用的安全开发能力 目录

C#TextBox设置提示文本方式(SetHintText)

《C#TextBox设置提示文本方式(SetHintText)》:本文主要介绍C#TextBox设置提示文本方式(SetHintText),具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑... 目录C#TextBox设置提示文本效果展示核心代码总结C#TextBox设置提示文本效果展示核心代

Python 迭代器和生成器概念及场景分析

《Python迭代器和生成器概念及场景分析》yield是Python中实现惰性计算和协程的核心工具,结合send()、throw()、close()等方法,能够构建高效、灵活的数据流和控制流模型,这... 目录迭代器的介绍自定义迭代器省略的迭代器生产器的介绍yield的普通用法yield的高级用法yidle

使用Python实现文本转语音(TTS)并播放音频

《使用Python实现文本转语音(TTS)并播放音频》在开发涉及语音交互或需要语音提示的应用时,文本转语音(TTS)技术是一个非常实用的工具,下面我们来看看如何使用gTTS和playsound库将文本... 目录什么是 gTTS 和 playsound安装依赖库实现步骤 1. 导入库2. 定义文本和语言 3

Python实现常用文本内容提取

《Python实现常用文本内容提取》在日常工作和学习中,我们经常需要从PDF、Word文档中提取文本,本文将介绍如何使用Python编写一个文本内容提取工具,有需要的小伙伴可以参考下... 目录一、引言二、文本内容提取的原理三、文本内容提取的设计四、文本内容提取的实现五、完整代码示例一、引言在日常工作和学

Java实现将Markdown转换为纯文本

《Java实现将Markdown转换为纯文本》这篇文章主要为大家详细介绍了两种在Java中实现Markdown转纯文本的主流方法,文中的示例代码讲解详细,大家可以根据需求选择适合的方案... 目录方法一:使用正则表达式(轻量级方案)方法二:使用 Flexmark-Java 库(专业方案)1. 添加依赖(Ma