题意

题目链接(Virtual Judge):Edit Step Ladders - UVA 10029

题意:

如果单词 \(x\) 能通过添加、删除或修改一个字母变换为单词 \(y\),则称单词 \(x\) 到单词 \(y\) 的变换为一个 edit step。

Edit step ladder 指的是一个按字典序排列的单词序列 \(w_1,w_2,\ldots,w_n\),每个 \(w_{i+1}\) 都由 \(w_i\) 经一个 edit step 变换而来。

给出一个按字典序排列的单词序列,问其中符合 edit step ladder 要求的最长子序列的长度。

思路

虽然这题目又是字典序,又是子序列什么的,但其实跟字符串没多少关系,做法是建图跑最长路,将单词视为图节点,单词之间的变换视为有向边。想出建图这个思路后,剩下的就好办了。

首先是怎样建图的问题。如果采用枚举所有单词对的方法,时间复杂度 \(O(n^2)\),很可能会超时。我们注意到单词的长度非常短,所以不妨换一种思路:对于一个单词,枚举它能变换出的所有单词,这样便能实现 \(O(n)\) 建图。

具体做法如下:用哈希表保存单词及其对应下标。对于一个单词,枚举它经过一步 edit step 变换后的所有单词,看变换后在不在哈希表里。如果在,且下标大于当前单词(为了满足字典序要求),则构造一条有向边。

接着是怎样求解最长路的问题。显然此图是个有向无环图,所以我们可以用动态规划的方法 \(O(n)\) 求出最长路。

用 \(\mathrm{step}(u)\) 表示从 \(u\) 出发的最长路的长度(路径上的节点数),状态转移方程如下:

\[\mathrm{step}(u)=\left\{
\begin{array}{ll}
1 & \text{if}\; u \;\text{is a leaf} \\
1+\max\{\mathrm{step}(v) \mid (u,v)\in E\} & \text{otherwise} \\
\end{array}
\right.
\]

代码

注意:输出答案后还要再输出一个换行符,否则 UVA 会判你 WA(而不是 PE),非常的毒瘤。

#include <iostream>
#include <string>
#include <vector>
#include <unordered_map>
#include <algorithm>
#include <cassert> using namespace std; const int maxn = int(25000 + 5);
string word_arr[maxn];
unordered_map<string, int> word_dict;
vector<int> G[maxn];
int step_arr[maxn]; // 将 s 的第 i 个字符换成 ch
string change_letter(const string &s, int i, char ch)
{
string ret = s;
ret[i] = ch;
return ret;
} // 将 s 的第 i 个字符移除
string remove_letter(const string &s, int i)
{
string ret;
int len = int(s.length());
ret.reserve(len - 1);
for (int j = 0; j < len; j++)
if (j != i)
ret.push_back(s[j]);
return ret;
} // 在 s 的第 i 个字符前面插入字符 ch
string insert_letter(const string &s, int i, char ch)
{
string ret;
int len = int(s.length());
ret.reserve(len + 1);
if (i == len)
{
ret = s;
ret.push_back(ch);
return ret;
}
for (int j = 0; j < len; j++)
{
if (j == i)
ret.push_back(ch);
ret.push_back(s[j]);
}
return ret;
} // 添加有向边 u -> v
void add_edge(int u, int v)
{
G[u].push_back(v);
} // 从点 u 出发的最长路的长度
int longest_step(int u)
{
if (step_arr[u] > 0)
return step_arr[u];
int max_son_step = 0;
for (int v : G[u])
max_son_step = max(max_son_step, longest_step(v));
step_arr[u] = max_son_step + 1;
return step_arr[u];
} int main()
{
ios_base::sync_with_stdio(false);
string word;
int index = 1;
while (cin >> word)
{
word_arr[index] = word;
word_dict[word] = index;
index++;
}
int n = index - 1; for (int u = 1; u <= n; u++)
{
const string &cur = word_arr[u];
int len = int(cur.length());
// 修改字符
for (int i = 0; i < len; i++)
{
for (char ch = cur[i] + 1; ch <= 'z'; ch++)
{
string gen = change_letter(cur, i, ch);
if (word_dict.count(gen) > 0)
{
int v = word_dict[gen];
assert(u < v);
add_edge(u, v);
}
}
}
// 移除字符
for (int i = 0; i < len; i++)
{
string gen = remove_letter(cur, i);
if (word_dict.count(gen) > 0)
{
int v = word_dict[gen];
if (u < v)
add_edge(u, v);
}
}
// 插入字符
for (int i = 0; i <= len; i++)
{
for (char ch = 'a'; ch <= 'z'; ch++)
{
string gen = insert_letter(cur, i, ch);
if (word_dict.count(gen) > 0)
{
int v = word_dict[gen];
if (u < v)
add_edge(u, v);
}
}
}
} int ans = 0;
for (int u = 1; u <= n; u++)
ans = max(ans, longest_step(u));
cout << ans << '\n'; return 0;
}

Edit Step Ladders - UVA 10029的更多相关文章

  1. UVA - 10029 Edit Step Ladders (二分+hash)

    Description Problem C: Edit Step Ladders An edit step is a transformation from one word x to another ...

  2. UVa 10029 - Edit Step Ladders

    題目:已知一些字典序排列的單詞,問能從中找到最大的一個有序單詞集合, 使得集合中的單詞每一個是有上一個單詞經過一次變換得來的(增.刪.改). 分析:dp,LIS.最大遞增子序列,不過數據較大须要優化. ...

  3. UVA 10029 Edit Step Ladders ——(DAG求最长路)

    题意:升序的给出一本若干个单词,每个单词都可删除一个字母,添加一个字母或者改变一个字母,如果任意一个操作以后能变成另外一个字典中的单词,那么就连一条有向边,求最长的长度. 分析:DAG的最长路和最短路 ...

  4. uva 10026 Problem C: Edit Step Ladders

    http://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&page=show_problem&p ...

  5. POJ2564:Edit Step Ladders

    浅谈\(Trie\):https://www.cnblogs.com/AKMer/p/10444829.html 题目传送门:http://poj.org/problem?id=2564 记\(f[i ...

  6. UVa 10029 hash + dp

    题目链接:http://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&page=show_problem& ...

  7. ACM训练计划step 1 [非原创]

    (Step1-500题)UVaOJ+算法竞赛入门经典+挑战编程+USACO 下面给出的题目共计560道,去掉重复的也有近500题,作为ACMer Training Step1,用1年到1年半年时间完成 ...

  8. UVA题目分类

    题目 Volume 0. Getting Started 开始10055 - Hashmat the Brave Warrior 10071 - Back to High School Physics ...

  9. (Step1-500题)UVaOJ+算法竞赛入门经典+挑战编程+USACO

    http://www.cnblogs.com/sxiszero/p/3618737.html 下面给出的题目共计560道,去掉重复的也有近500题,作为ACMer Training Step1,用1年 ...

随机推荐

  1. SpringCloud分布式配置中心Config

    统一管理所有配置. 1.微服务下的分布式配置中心 简介:讲解什么是配置中心及使用前后的好处 什么是配置中心: 一句话:统一管理配置, 快速切换各个环境的配置 相关产品: 百度的disconf 地址:h ...

  2. Spring第一课:配置文件及IOC引入(一)

    Spring最核心的特点就是控制反转:(IOC)和面向切面(AOP) 首先作为一个Spring的项目需要导入的四个核心包,一个依赖: 四核心:core.context.beans.expression ...

  3. 转:C语言自增自減问题总结

    C语言自增自減问题总结 在程序设计中,经常遇到"i=i+1"和"i=i-1"这两种极为常用的操作.C语言为这种操作提供了两个更为简洁的运算符,即++和--,分别 ...

  4. [题解] Luogu P5446 [THUPC2018]绿绿和串串

    [题解] Luogu P5446 [THUPC2018]绿绿和串串 ·题目大意 定义一个翻转操作\(f(S_n)\),表示对于一个字符串\(S_n\), 有\(f(S)= \{S_1,S_2,..., ...

  5. vue 引用省市区三级联动(插件)

    vue 用省市区三级联动之傻瓜式教程(复制粘贴即用) npm 下载 npm install v-distpicker --save main.js //引入 省市区三级联动 import Distpi ...

  6. 一文读懂Lua元表

    元表 Lua语言中的每种类型的值都有一套可预见的操作集合.例如,我们可以将数字相加,可以连接字符串,还可以在表中插入键值对等,但是我们无法将两个表相加,无法对函数作比较,也无法调用一个字符串,除非使用 ...

  7. 洛谷P3768 简单的数学题解题报告

    $$\begin{eqnarray}&\sum_{i=1}^{n}\sum_{j=1}^{n}ij\gcd(i,j)\\&\sum_{d=1}^{n}\sum_{i=1}^{n}\su ...

  8. Docker(40)- docker 实战三之安装 ES+Kibana

    背景 参考了狂神老师的 Docker 教程,非常棒! https://www.bilibili.com/video/BV1og4y1q7M4?p=16 es 前言 es 暴露的端口很多 es 十分耗内 ...

  9. 史上最详细的信号使用说明(已被收藏和N次)

    Unix环境高级编程(第三版) 第10章 信号 文章目录 1. 引言 2. 信号的概念 2.1 信号操作之忽略信号 2.2 信号操作之捕捉信号 2.3 信号操作之执行系统默认操作 2.4 常见的信号 ...

  10. JS005. 拷贝引用数据类型Array使其指向不同堆的解决方案

    一个很常见的语法问题,但专注实现需求时经常会忘记去避免,导致最终问题的出现,再花时间排查.为此专门整理一篇解决方法的博客,也加强一下自己的记忆. TAG: JSON.parse() JSON.stri ...