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

相关文章

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

Linux使用cut进行文本提取的操作方法

《Linux使用cut进行文本提取的操作方法》Linux中的cut命令是一个命令行实用程序,用于从文件或标准输入中提取文本行的部分,本文给大家介绍了Linux使用cut进行文本提取的操作方法,文中有详... 目录简介基础语法常用选项范围选择示例用法-f:字段选择-d:分隔符-c:字符选择-b:字节选择--c

C#使用DeepSeek API实现自然语言处理,文本分类和情感分析

《C#使用DeepSeekAPI实现自然语言处理,文本分类和情感分析》在C#中使用DeepSeekAPI可以实现多种功能,例如自然语言处理、文本分类、情感分析等,本文主要为大家介绍了具体实现步骤,... 目录准备工作文本生成文本分类问答系统代码生成翻译功能文本摘要文本校对图像描述生成总结在C#中使用Deep

Mybatis官方生成器的使用方式

《Mybatis官方生成器的使用方式》本文详细介绍了MyBatisGenerator(MBG)的使用方法,通过实际代码示例展示了如何配置Maven插件来自动化生成MyBatis项目所需的实体类、Map... 目录1. MyBATis Generator 简介2. MyBatis Generator 的功能3

通过C#获取PDF中指定文本或所有文本的字体信息

《通过C#获取PDF中指定文本或所有文本的字体信息》在设计和出版行业中,字体的选择和使用对最终作品的质量有着重要影响,然而,有时我们可能会遇到包含未知字体的PDF文件,这使得我们无法准确地复制或修改文... 目录引言C# 获取PDF中指定文本的字体信息C# 获取PDF文档中用到的所有字体信息引言在设计和出

Java操作xls替换文本或图片的功能实现

《Java操作xls替换文本或图片的功能实现》这篇文章主要给大家介绍了关于Java操作xls替换文本或图片功能实现的相关资料,文中通过示例代码讲解了文件上传、文件处理和Excel文件生成,需要的朋友可... 目录准备xls模板文件:template.xls准备需要替换的图片和数据功能实现包声明与导入类声明与