Luogu P1245 电话号码

谨以此题解致敬我的初中英语老师孙菡老师,她带的班默写不过的人数总是像电话号码那样屈指可数

玄学问题?

本题的SPJ似乎已经基本没有问题了,只要 文末没有多余的空格和回车 就能正常评测。


审题

本题给出了26个字母与数字的对应关系,要求将一串数字翻译为几个单词。

那么我们是不需要关注各个单词中的字母具体是什么的,只需要存起来输出的时候用一下就行了,翻译过程中完全可以转换为纯数字操作。

具体地说:the->732she->732,对于密码中的732,它们是完全等价的。所以我们建立一个数组用于转换:

const char st[26]={1,1,1,2,2,2,3,3,3,4,4,4,5,5,6,6,6,7,7,7,8,8,8,9,9,9};

然后题目变成了:给定目标串,要求将它拆解为多个模式串首尾相连的结果。

AC自动机字典树+深度优先搜索


Trie

先上代码:

inline void init(char *a)
{
register int i=0;
for(;*a;++i,++a)
{
b[i]=st[*a-'a'];
}
b[i]='\0';
return;
}
inline void insert(char *a,const int &id)
{
register trie *p=head;
for(;*a;++a)
{
if(p->son[*a])
{
p=p->son[*a];
}
else
{
p=p->son[*a]=++mtp;
}
}
p->end=id;
return;
}

我先使用init(str[i])str[i][](第i个单词)转换为纯数字保存在b[]中,然后使用insert(b,i)执行插入操作。为了方便搜到答案之后输出,我将单词的编号i传递进函数,用编号作为Trie中字符串的结尾标记Trie::end,这样就可以方便地一边深搜一边统计答案。

另外,比如the->732&she->732,就会执行两次完全相同的插入过程,但是我们不需要考虑这个问题,因为按照题意,它们完全等价,end中会保存较晚插入的那个单词的编号,而题目只要求输出一组可行解。


值得一提的是insert(b,i)前我还进行了特判if(*b)

  for(i=1;i<=n;++i)
{
init(str[i]);
if(*b)insert(b,i);
}

这个操作的用意是判断b[]是否是空串,理论上来说不可能会有这种情况,可是我切这道题是遇到了Trie的根节点莫名其妙被标记为字符串末尾的现象,就这样解决了......目前还没有查出问题根源,如果大佬们AC了这道题,请劳驾前往这里帮我看看是不是我哪个地方写了。。


DFS

先上代码:

void dfs(int x)
{
register trie *p=head;
for(;a[x];)
{
if(p->end)
{
ans[++cnt]=p->end;
dfs(x);
--cnt;
}
if(p->son[a[x]])
{
p=p->son[a[x]];
++x;
}
else
{
return;
}
}
if(p->end)
{
ans[++cnt]=p->end;
out();
}
return;
}
变量说明

x为当前待匹配的数字的下标;

指针p用于检索Trie树;

ans[]用于保存当前搜到的状态。

DFS过程

我们使用Trie将所有单词记录后,对密码串从前到后扫描,并在Trie中查询,每查到一个单词的末尾标记,就说明这个地方是有可能断开成为一个单词的,就保存这个单词的编号(即每次搜到p->end!=0说明可能要从这里划断,就先ans[++cnt]=p->end,保存这个词的编号,dfs(x)把当前这个待匹配位置留给下一层搜索,如果回溯回来了,cnt--清除即可),从这个地方再递归一层考虑是否可行。如果一直这样到了密码串末尾,就找到了一组解,输出即可;如果到某个地方Trie上匹配不到了,就说明之前某个地方划分错了,应该回溯,return即可。如果一直退回到main()都没找到解,说明No Solution!

具体地讲(我是指针选手,希望看得惯,话说这道题使用指针影响不大),每次调用dfs()时都先让一个指针p指向head,使用for从参数指定的位置向后扫描。每次循环先看此处是否是一个单词的结尾(if(p->end)),如果是就ans[++cnt]=p->end;dfs(x);进行下一层的匹配,否则查看是否存在后继节点是数字a[x],如果有就p=p->son[a[x]];x++;匹配下一个数字,否则匹配失败GG了,return。就这样一直到密码串末尾(a[x]=='\0'),还没完,需要判断当下这个p->end是不是0,如果是0说明最后面配到的这一小段并不是某个单词的结尾,只能被迫return继续搜索;如果非0,那么我们终于找到了一组答案ans[++cnt]=p->end;保存最后这个单词的编号,out();输出找到的单词序列~~

代码

//P1245 电话号码
#include<cstdio>
#include<cstdlib>
#include "memory.h"
#define MAXN 110
using namespace std;
const char st[26]={1,1,1,2,2,2,3,3,3,4,4,4,5,5,6,6,6,7,7,7,8,8,8,9,9,9};
struct trie
{
int end;
trie *son[17];
inline trie();
}mmp[MAXN*MAXN],*mtp=mmp,*head=NULL;
char str[MAXN][MAXN],a[MAXN],b[MAXN];
int ans[MAXN],cnt=0,n=0;
inline void read(int&);
void write(const int&);
inline void insert(char*,const int&);
inline void init(char*);
void dfs(int);
inline void out();
int main()
{
register int i=0;
head=++mtp;
scanf("%d",&n);
scanf("%s",a);
for(i=0;a[i];a[i++]^=48);
for(i=1;i<=n;++i)
{
scanf("%s",str[i]);
}
for(i=1;i<=n;++i)
{
init(str[i]);
if(*b)insert(b,i);
}
dfs(0);
printf("No Solutions!");
return 0;
}
inline trie::trie()
{
end=0;
memset(son,0,sizeof(son));
return;
}
inline void out()
{
register int i=0;
for(i=1;i<cnt;++i)
{
printf("%s ",str[ans[i]]);
}
printf("%s",str[ans[cnt]]);
exit(0);
}
void dfs(int x)
{
register trie *p=head;
for(;a[x];)
{
if(p->end)
{
ans[++cnt]=p->end;
dfs(x);
--cnt;
}
if(p->son[a[x]])
{
p=p->son[a[x]];
++x;
}
else
{
return;
}
}
if(p->end)
{
ans[++cnt]=p->end;
out();
}
return;
}
inline void init(char *a)
{
register int i=0;
for(;*a;++i,++a)
{
b[i]=st[*a-'a'];
}
b[i]='\0';
return;
}
inline void insert(char *a,const int &id)
{
register trie *p=head;
for(;*a;++a)
{
if(p->son[*a])
{
p=p->son[*a];
}
else
{
p=p->son[*a]=++mtp;
}
}
p->end=id;
return;
}

Luogu P1245 电话号码的更多相关文章

  1. [LeetCode] Valid Phone Numbers 验证电话号码

    Given a text file file.txt that contains list of phone numbers (one per line), write a one liner bas ...

  2. [LeetCode] Letter Combinations of a Phone Number 电话号码的字母组合

    Given a digit string, return all possible letter combinations that the number could represent. A map ...

  3. 【代码笔记】iOS-替换电话号码中间4位为-号

    一,效果图. 二,代码. RootViewController.m - (void)viewDidLoad { [super viewDidLoad]; // Do any additional se ...

  4. 国内固定电话正则验证:'tel': [/0\d{2,3}-\d{7,8}(|([-\u8f6c]{1}\d{1,5}))$/, "请填写有效的电话号码"],

    // 验证字段 $('#info_form').validator({ rules : { checkMobile : function(ele) { return checkMobile(ele); ...

  5. C#中使用正则表达式验证电话号码、手机号、身份证号、数字和邮编

      验证电话号码的主要代码如下: public bool IsTelephone(string str_telephone) { return System.Text.RegularExpressio ...

  6. Luogu 魔法学院杯-第二弹(萌新的第一法blog)

    虽然有点久远  还是放一下吧. 传送门:https://www.luogu.org/contest/show?tid=754 第一题  沉迷游戏,伤感情 #include <queue> ...

  7. 防止在iOS设备中的Safari将数字识别为电话号码

    在测试中发现iPad上的Safari总会把长串数字识别为电话号码,文字变成蓝色,点击还会弹出菜单添加到通讯录. 别的地方倒也罢了,如果在用户名中出现数字(手机注册新浪微博的话用户名就是“手机用户xxx ...

  8. java 验证手机号码、电话号码(包括最新的电信、联通和移动号码)

    一.目前的号码段(2016-12-8更新)   二.代码 package com.test; import java.util.regex.Pattern; public class CheckPho ...

  9. php电话号码正则表达式常用例子

    电话号码正则表达式(支持手机号码,3-4位区号,7-8位直播号码,1-4位分机号) 02   03 ((\d{11})|^((\d{7,8})|(\d{4}|\d{3})-(\d{7,8})|(\d{ ...

随机推荐

  1. Linux如何实现进程监控和守护

    最近新搭建的亚马逊EC2服务器, 上面部署了一个静态的WEB, 启动了一个nginx做代理.最近发现一个问题: Nginx进程隔一段时间就莫名的挂掉了, 然后就出现了网站无法打开的窘境.. 为了防止这 ...

  2. Asp.net MVC WebApi项目的自动接口文档及测试功能打开方法

    https://blog.csdn.net/foren_whb/article/details/78866133

  3. SWIT2019无线通信和信息技术国际研讨会(上海)

    无线通信和信息技术国际研讨会(SWIT 2019)将于2019年6月29日至30日在中国上海皇冠晶品酒店举行.本次会议将讨论无线通信和信息技术问题.它致力于创造一个交流最新研究成果和分享先进研究方法的 ...

  4. MFC关于.rc文件 .rc2文件

    .rc文件和.rc2文件 c和rc2都是资源文件,包含了应用程序中用到的所有的资源. 两者不同在于:rc文件中的资源可以直接在VC集成环境中以可视化的方法进行编辑和修改; 而rc2中的资源不能在VC的 ...

  5. 关于表情的战争APP隐私政策网址

    本软件尊重并保护所有使用服务用户的个人隐私权.为了给您提供更准确.更有个性化的服务,本软件会按照本隐私权政策的规定使用和披露您的个人信息.但本软件将以高度的勤勉.审慎义务对待这些信息.除本隐私权政策另 ...

  6. Centos7下使用Ceph-deploy快速部署Ceph分布式存储-操作记录

    之前已详细介绍了Ceph分布式存储基础知识,下面简单记录下Centos7使用Ceph-deploy快速部署Ceph环境: 1)基本环境 192.168.10.220 ceph-admin(ceph-d ...

  7. Canvas Snippets

    ========================================== Example: 1. To revel "fillStyle" property, type ...

  8. Linux Centos 删除除某(多)个文件之外的所有文件

    好久没有写东西了.通常我们通过rm -rf  *可以直接强制删除当前文件夹里面的所有内容,但是有些时候我们需要保留一些文件,就比如,网站转移更新需要保留程序压缩包等就需要用到在linux centos ...

  9. Back up and restore information in Firefox profiles

    Click the menu button , click Help and select Troubleshooting Information. The Troubleshooting Infor ...

  10. Machine Learning 第三周

    ML week3 逻辑回归 Logistic Function h_\theta(x)=g(\theta^Tx) g(t)=\frac{1}{1+e^{-z}} 当t大于0, 即下面公式成立时,y=1 ...