把电话号码转换成为词典中能够记忆的的单词的组合,找到最短的组合。

我这道题应用到的知识点:

1 Trie数据结构

2 map的应用

3 动态规划法Word Break的知识

4 递归剪枝法

思路:

1 建立Trie字典树。方便查找, 可是字典树不是使用字符来建立的。而是把字符转换成数字。建立一个数字字典树。 然后叶子节点设置一个容器vector<string>装原单词。

2 动态规划建立一个表,记录能够在字典树中找到的字符串的前缀子串

3 假设找到整个串都在字典树中,那么就能够直接返回这个单词。假设无法直接找到。那么就要在表中找到一个前缀子串,然后后面部分在字典树中查找,看是否找到包括这个子串的单词,而且要求找到的单词长度最短。- 这里能够使用剪枝法提高效率。

原题:http://acm.timus.ru/problem.aspx?space=1&num=1002

作者:靖心 - http://blog.csdn.net/kenden23

#include <iostream>
#include <string>
#include <vector>
#include <cmath>
#include <algorithm>
#include <unordered_map> using namespace std; class PhoneNumber1002_2
{
static const int SIZE = 10;
struct Node
{
vector<string> words;
Node *children[SIZE];
explicit Node () : words()
{
for (int i = 0; i < SIZE; i++)
{
children[i] = NULL;
}
}
}; struct Trie
{
Node *emRoot;
int count;
explicit Trie(int c = 0) : count(c)
{
emRoot = new Node;
}
~Trie()
{
deleteTrie(emRoot);
}
void deleteTrie(Node *root)
{
if (root)
{
for (int i = 0; i < SIZE; i++)
{
deleteTrie(root->children[i]);
}
delete root;
root = NULL;
}
}
}; void insert(Trie *trie, string &keys, string &keyWords)
{
int len = (int)keys.size(); Node *pCrawl = trie->emRoot;
trie->count++; for (int i = 0; i < len; i++)
{
int k = keys[i] - '0';
if (!pCrawl->children[k])
{
pCrawl->children[k] = new Node;
}
pCrawl = pCrawl->children[k];
}
pCrawl->words.push_back(keyWords);
} Node *search(Node *root, string &keys)
{
int len = (int)keys.size(); Node *pCrawl = root;
for (int i = 0; i < len; i++)
{
int k = keys[i] - '0';
if (!pCrawl->children[k])
{
return NULL;//没走全然部keys
}
pCrawl = pCrawl->children[k];
}
return pCrawl;
} void searchLeft(Node *leaf, Node *r, int len, int &prun)
{
if (len >= prun) return; if (leaf->words.size())
{
r = leaf;
prun = len;
return;
} for (int i = 0; i < SIZE; i++)
{
searchLeft(leaf->children[i], r, len+1, prun);
}
} void wordsToKey(string &keys, string &keyWords,
unordered_map<char, char> &umCC)
{
for (int i = 0; i < (int)keyWords.size(); i++)
{
keys.push_back(umCC[keyWords[i]]);
}
} void charsToMap(const string phdig[], unordered_map<char, char> &umCC)
{
for (int i = 0; i < 10; i++)
{
for (int k = 0; k < (int)phdig[i].size(); k++)
{
umCC[phdig[i][k]] = i + '0';
}
}
} string searchComb(Trie *trie, string &num)
{
vector<string> tbl(num.size());
for (int i = 0; i < (int)num.size(); i++)
{
string s = num.substr(0, i+1);
Node *n = search(trie->emRoot, s);
if (n && n->words.size())
{
tbl[i].append(n->words[0]);
continue;//这里错误写成break! 。
}
for (int j = 1; j <= i; j++)
{
if (tbl[j-1].size())
{
s = num.substr(j, i-j+1);
n = search(trie->emRoot, s);
if (n && n->words.size())
{
tbl[i].append(tbl[j-1]);
tbl[i].append(" ");
tbl[i].append(n->words[0]);
break;
}
}
}
} if (tbl.back().size())
{
return tbl.back();
} string ans;
for (int i = 0; i < (int)tbl.size() - 1; i++)
{
if (tbl[i].size())
{
string tmp = tbl[i];
string keys = num.substr(i+1);
Node *n = search(trie->emRoot, keys); if (!n) continue; Node *r = NULL;
int prun = INT_MAX;
searchLeft(n, r, 0, prun); tmp += r->words[0]; if (ans.empty() || tmp.size() < ans.size())
{
ans = tmp;
}
}
}
return ans.empty()? "No solution." : ans;
} //測试函数。不使用解题
void printTrie(Node *n)
{
if (n)
{
for (int i = 0; i < SIZE; i++)
{
printTrie(n->children[i]);
for (int j = 0; j < (int)n->words.size(); j++)
{
cout<<n->words[j]<<endl;
}
}
}
}
public:
PhoneNumber1002_2()
{
const string phdig[10] =
{"oqz","ij","abc","def","gh","kl","mn","prs","tuv","wxy"};
unordered_map<char, char> umCC;
charsToMap(phdig, umCC); int N; string num, keys, keyWords;
while ((cin>>num) && "-1" != num)
{
cin>>N; Trie trie;
while (N--)
{
cin>>keyWords;
wordsToKey(keys, keyWords, umCC); insert(&trie, keys, keyWords); keys.clear();//别忘记清空
} cout<<searchComb(&trie, num)<<endl;
}
}
};

Timus : 1002. Phone Numbers 题解的更多相关文章

  1. 1002 Phone Numbers 解题报告

    1002. Phone Numbers Time limit: 2.0 secondMemory limit: 64 MB In the present world you frequently me ...

  2. URAL 1002 Phone Numbers(KMP+最短路orDP)

    In the present world you frequently meet a lot of call numbers and they are going to be longer and l ...

  3. CF55D Beautiful numbers 题解

    题目 Volodya is an odd boy and his taste is strange as well. It seems to him that a positive integer n ...

  4. Hdoj 1905.Pseudoprime numbers 题解

    Problem Description Fermat's theorem states that for any prime number p and for any integer a > 1 ...

  5. Hdoj 1058.Humble Numbers 题解

    Problem Description A number whose only prime factors are 2,3,5 or 7 is called a humble number. The ...

  6. [LeetCode] Add Two Numbers题解

    Add Two Numbers: You are given two non-empty linked lists representing two non-negative integers. Th ...

  7. poj 1995 Raising Modulo Numbers 题解

    Raising Modulo Numbers Time Limit: 1000MS   Memory Limit: 30000K Total Submissions: 6347   Accepted: ...

  8. Timus 1009. K-based Numbers

    1009. K-based Numbers Time limit: 0.5 secondMemory limit: 64 MB Let’s consider K-based numbers, cont ...

  9. Timus 1712. Cipher Grille 题解

    版权声明:本文作者靖心,靖空间地址:http://blog.csdn.net/kenden23/.未经本作者同意不得转载. https://blog.csdn.net/kenden23/article ...

随机推荐

  1. 兼容浏览器 回车键 keydown事件

    $("body").keydown(function(event){   if(event.keyCode==13){   //body   } }); 重点:$("bo ...

  2. 【转】 [UnityUI]UGUI射线检测

    http://blog.csdn.net/lyh916/article/details/50947026 1.Graphic Raycaster 主要用于UI上的射线检测,挂有这个组件的物体,必须要挂 ...

  3. 原生js实现tooltip提示框的效果

    在js的世界里面,每一个小的特效都那么微不足道,却又那么的令人向往与好奇.前端工程师的任务特别高大上,因为他们的一个小小的设计就会激发别人的求知欲.比如说我,只是随机一瞟,便看到了这个tooltip的 ...

  4. bzoj 2503 相框 分类讨论

    题目大意:给定一张无向图,每次可以进行以下两种操作: 1.将一个点分裂成一些点,原先这个点连接的每条边任选一个新点进行连接 2.将两个度数为1的点合并为1个点 求将这个图变成一个环的最小操作次数 我们 ...

  5. react当中子组件改变父组件的状态

    子组件直接改变父组件传入的props值是不被允许的, 当需要在子组件当中改变父组件的某一个状态, 父组件传入一个改变状态的函数,然后在子组件当中调用函数即可

  6. 【03】Vue 之列表渲染及条件渲染

    3.1. 条件渲染 有时候我们要根据数据的情况,决定标签是否进行显示或者有其他动作.最常见的就是,表格渲染的时候,如果表格没有数据,就显示无数据.如果有数据就显示表格数据. Vue帮我们提供了一个v- ...

  7. hdu 2147 kiki's game 组合游戏 找规律

    题目链接 题意 两人轮流将硬币从\((n,m)\)移动到\((1,1)\),每次只能向下或向左或向左下移动一格,最后无法移动者输.问先手会赢还是会输. 思路 找规律 -- P N P N P N P ...

  8. android中提示&对话框----Notification

    Notification(状态栏通知) 一.Notification用于状态栏显示通知的控件,在不同的设备上面Notification是不一样的 二.Notification的基本布局 元素组成: I ...

  9. [SaltStack] Minion-conf自动更新

    minion-conf配置文件自动更新, 加载 minion-conf是每个minion自身以来的配置, 为了方便我们在中心管控机上(Master)统一配置, 然后下发文件, 进而使得Minion能够 ...

  10. ubuntu下使用OBS开斗鱼直播

    系统环境:ubuntu 15.10,OBS Studio 0.13.1 OBS是可以在linux,windows,mac下直播的开源软件,官方地址:https://obsproject.com/ 斗鱼 ...