1002. Phone Numbers

Time limit: 2.0 second
Memory limit: 64 MB
In the present world you frequently meet a lot of call numbers and they are going to be longer and longer. You need to remember such a kind of numbers. One method to do it in an easy way is to assign letters to digits as shown in the following picture:
1 ij    2 abc   3 def
4 gh 5 kl 6 mn
7 prs 8 tuv 9 wxy
0 oqz
This way every word or a group of words can be assigned a unique number, so you can remember words instead of call numbers. It is evident that it has its own charm if it is possible to find some simple relationship between the word and the person itself. So you can learn that the call number 941837296 of a chess playing friend of yours can be read as WHITEPAWN, and the call number 2855304 of your favourite teacher is read BULLDOG.
Write a program to find the shortest sequence of words (i.e. one having the smallest possible number of words) which corresponds to a given number and a given list of words. The correspondence is described by the picture above.

Input

Input contains a series of tests. The first line of each test contains the call number, the transcription of which you have to find. The number consists of at most 100 digits. The second line contains the total number of the words in the dictionary (maximum is 50 000). Each of the remaining lines contains one word, which consists of maximally 50 small letters of the English alphabet. The total size of the input doesn't exceed 300 KB. The last line contains call number −1.

Output

Each line of output contains the shortest sequence of words which has been found by your program. The words are separated by single spaces. If there is no solution to the input data, the line contains text “No solution.”. If there are more solutions having the minimum number of words, you can choose any single one of them.

Sample

input output
7325189087
5
it
your
reality
real
our
4294967296
5
it
your
reality
real
our
-1
reality our
No solution.
Problem Source: Central European Olympiad in Informatics 1999
 
题意:
给定一个数字串和若干个单词,根据手机按键用单词数将数字串拼出来。
求最少单词数对应的单词组合。
 
分析:
设数字串num的长度为l,前i个数字对应的最少单词数为f(i), 如果从i开始有一个单词word(j)能够匹配num(i+1, i+word(j).length),
则可以更新:f(i+word(j).length) = min(f(i+word(j).length), f(i)+1)
题目要求是要求出对应的单词组合,可以在递推过程中保存从f(i+word(j).length) 到 f(i)的指针以及对应的单词编号来记录,代码中使用了lw数组完成这项工作。
 
代码:
 //
// main.cpp
// phoneNumber2
//
// Created by apple on 16/1/30.
// Copyright © 2016年 apple. All rights reserved.
// #include <iostream>
#include <vector>
#include <stack>
#include <map>
//数字串长度
#define maxl 1000
//单词个数
#define maxn 100
//最多单词数
#define maxw 100
//最大值
#define inf 100000 using namespace std;
struct node {
node(int a, int b, bool v) {
fa = a;
wn = b;
vrfd = v;
}
node () {
vrfd = false;
}
int fa, wn;
bool vrfd;
};
//每个长度对应的最少单词数,初始化为inf
int f[maxl+];
//每个长度对应最少单词数的单词编号队
node lw[maxl+];
//单词们
vector<string> words;
stack<int> ans;
string num;
map<char, char> phone; int match(int pl, int wn) {
for (int i = ; i < words[wn].length(); i++) {
if (pl == num.length() || phone[words[wn][i]] != num[pl])
return -;
pl++;
}
return pl;
}
int main(int argc, const char * argv[]) {
phone['o'] = phone['q'] = phone['z'] = '';
phone['w'] = phone['x'] = phone['y'] = '';
phone['t'] = phone['u'] = phone['v'] = '';
phone['p'] = phone['r'] = phone['s'] = '';
phone['m'] = phone['n'] = '';
phone['k'] = phone['l'] = '';
phone['g'] = phone['h'] = '';
phone['d'] = phone['e'] = phone['f'] = '';
phone['a'] = phone['b'] = phone['c'] = '';
phone['i'] = phone['j'] = '';
int wn, i, j, tmp1;
string tmp;
lw[] = node(-, -, true);
while (cin >> num && num != "-1") {
words.clear();
cin >> wn;
while (wn--) {
cin >> tmp;
words.push_back(tmp);
}
f[] = ;
for (i = ; i <= maxl; i++) {
f[i] = inf;
}
for (i = ; i <= maxl; i++) {
lw[i] = node(-,-,false);
}
for (i = ; i < num.length(); i++) {
for (j = ; j < words.size(); j++) {
if (lw[i].vrfd && (tmp1 = match(i, j)) != - && f[tmp1] > f[i]+) {
f[tmp1] = f[i] + ;
lw[tmp1] = node(i, j, true);
}
}
}
if (!lw[num.length()].vrfd) {
cout << "No solution." << endl;
} else {
int itr = num.length();
while (itr != ) {
ans.push(lw[itr].wn);
itr = lw[itr].fa;
}
while (ans.size() != ) {
cout << words[ans.top()] << " ";
ans.pop();
}
cout << words[ans.top()] << endl;
ans.pop();
}
}
}
 

1002 Phone Numbers 解题报告的更多相关文章

  1. 【九度OJ】题目1442:A sequence of numbers 解题报告

    [九度OJ]题目1442:A sequence of numbers 解题报告 标签(空格分隔): 九度OJ 原题地址:http://ac.jobdu.com/problem.php?pid=1442 ...

  2. 【LeetCode】129. Sum Root to Leaf Numbers 解题报告(Python)

    [LeetCode]129. Sum Root to Leaf Numbers 解题报告(Python) 标签(空格分隔): LeetCode 题目地址:https://leetcode.com/pr ...

  3. 【LeetCode】386. Lexicographical Numbers 解题报告(Python)

    [LeetCode]386. Lexicographical Numbers 解题报告(Python) 标签(空格分隔): LeetCode 作者: 负雪明烛 id: fuxuemingzhu 个人博 ...

  4. 【LeetCode】165. Compare Version Numbers 解题报告(Python)

    [LeetCode]165. Compare Version Numbers 解题报告(Python) 标签(空格分隔): LeetCode 作者: 负雪明烛 id: fuxuemingzhu 个人博 ...

  5. USACO Section2.2 Runaround Numbers 解题报告 【icedream61】

    runround解题报告---------------------------------------------------------------------------------------- ...

  6. [POJ 1002] 487-3279 C++解题报告

        487-3279 Time Limit: 2000MS   Memory Limit: 65536K Total Submissions: 228365   Accepted: 39826 D ...

  7. 【LeetCode】1022. Sum of Root To Leaf Binary Numbers 解题报告(Python)

    作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu.cn/ 目录 题目描述 题目大意 解题方法 DFS 日期 题目地址:https://leetco ...

  8. 【LeetCode】628. Maximum Product of Three Numbers 解题报告(Python)

    作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu.cn/ 目录 题目描述 题目大意 解题方法 方法一:排序 日期 题目地址:https://lee ...

  9. 洛谷 CF55D Beautiful numbers 解题报告

    CF55D Beautiful numbers 题意 \(t(\le 10)\)次询问区间\([l,r](1\le l\le r\le 9\times 10^{18})\)中能被每一位上数整除的数的个 ...

随机推荐

  1. 【Alpha】Daily Scrum Meeting第三次

    本次随笔调换了展示顺序,把重要的内容放前面. 一.本次Daily Scrum Meeting主要内容 说明要完成alpha版本还需要哪些功能 对这些功能进行分析和实现方式的讨论 强调编码规范和变量命名 ...

  2. npm link 安装本地模块,将本地模块cli化

    第三方学习地址 http://mp.weixin.qq.com/s?__biz=MzAxMTU0NTc4Nw==&mid=2661157390&idx=1&sn=6d96e54 ...

  3. SharePoint Configuration Wizard - Unable to upgrade SharePoint Products and Technologies because an upgrade is already in progress

    故障描述 当要运行SharePonit Products and Technologies Configuration Wizard的时候,出现了如下图所示的错误提示. 错误信息为: Unable t ...

  4. Java实现Oracle数据库备份

    今天将很早就实现的一个Oracle数据库备份功能粘贴出来,这个功能是在大学做阶段设计时写的: import java.io.File; import java.io.IOException; /** ...

  5. centos部署gitlab服务器

    gitlab的安装和配置非常简单,关于git,这里摘抄一下百度百科: Git是一款免费.开源的分布式版本控制系统,用于敏捷高效地处理任何或小或大的项目.Git是一个开源的分布式版本控制系统,可以有效. ...

  6. CPP - sort

    #include "stdafx.h" #include <iostream> #include <string> using namespace std; ...

  7. Mybatis与Spring整合,使用了maven管理项目,作为初学者觉得不错,转载下来

    转载自:http://www.cnblogs.com/xdp-gacl/p/4271627.html 一.搭建开发环境 1.1.使用Maven创建Web项目 执行如下命令: mvn archetype ...

  8. Java 枚举7常见种用法

    DK1.5引入了新的类型--枚举.在 Java 中它虽然算个"小"功能,却给我的开发带来了"大"方便. 用法一:常量 在JDK1.5 之前,我们定义常量都是:  ...

  9. vert.x学习(八),用JDBCClient配合c3p0操作数据库

    今天学习了下vert.x的JDBCClient,我这里将今天的学习笔记记录下来.这次学习中使用了c3p0. 用使用JDBCClient和c3p0得现在pom.xml文件里面导入对应的依赖,下面贴出xm ...

  10. merge 本地 master 分支代码提示 “Already up-to-date”

    在使用 git 的过程中由于误操作,导致从本地 master 分支 merge 代码到当前分支失败,虽然当前分支和 master 分支代码不同步,但是仍然提示 Already up-to-date. ...