解法

这题比赛时过的人很多,我却没思路,糊里糊涂写了个强联通分量,得了 80 分。

这题思路是这样的。

一个替换操作可以看做一个有向边,所以题目实际上给出了一个有向图 $G$,一个节点代表一个字母。

注意题目要求每个操作都必须执行一次。

关于自环

首先注意到自环是没有意义的,因此处理输入时把自环忽略掉。

这里需要特别说明自环的问题,题目描述中并没有说明 $X_i \ne Y_i$。不过似乎可以合理地假设输入中不存在 $X_i = Y_i$ 的操作。

有些 AC 的代码并没有判断自环,比如冰心水蜜桃的提交。当输入中有自环时,这个代码是有 BUG 的。

关于重边

实际上重边也是没有意义的,但是我们不必特别处理它。


用 $\mathsf{h}$ 表示 $G$ 中对应于字符 ‘h’ 的节点。

设字符 $x$ 在串 $S$ 中出现过且 $x$ 不是 ‘h’ 将 $x$ 出现的次数记做 $c_x$ 。则这 $c_x$ 个 $x$ 能转变为 ‘h’ 的充要条件是「图 $G$ 中存在一条从 $x$ 到 ‘h’ 的简单路径」。

证明:不失一般性,设 $x\to x_1 \to x_2 \to \mathsf{h}$ 是一条从 $x$ 到 ‘h’ 的简单路径,则我们可以按下述方法将 $x$ 变为 ‘h’。

首先将 $x$ 变为 $x_1$,这个操作用掉了 $(x\to x_1)$ 这条边;再将剩下的从 $x$ 发出的边全部用掉,这些边将不换改变当前的字符串,然后把所有从 $x$ 发出的边从图 $G$ 中删除。以此类推。

不难注意到若 ‘h’ 有出边,则上述论证是有问题的。

‘h’ 的出度为零的情形是平凡的。考虑 ‘h’ 的出度不为零的情形。此时若图 $G$ 中不存在「从 ‘h’ 到 ‘h’ 的回路」,则若初始字符串 $S$ 中有 ‘h’,则这些 ‘h’ 终将变成别的字符。因此在这种情况下我们可以将 ‘h’ 的所有出边先执行一遍,并把这些边从图 $G$ 中删除。

这样就完成了上述论证。

不过至此我们只是针对一个字符 $x$ 进行论证。实际上对多个字符,结论是一样的,证明留给读者。


现在来考虑图 $G$ 中存在从 ‘h’ 到 ‘h’ 的回路的情形。注意这样的回路一定不是自环。任取一个从 ‘h’ 到 ‘h’ 的回路 $C$,我们可以先把 'h' 变成回路 $C$ 上 'h' 的后继,得到一个新字符串 $S'$,并把图 $G$ 中其他的 ‘h’ 的出边删除。这样就把问题规约为上一段所描述的情形。

实现

我们需要判断的是,对于字符 $x\in S$ 且 $x\ne\mathsf{h}$,图 $G$ 中是否有一条从 $x$ 到 $\mathsf{h}$ 的简单路径,这可以通过 DFS 完成。另外当 $\mathsf{h}$ 的出度不为零时我们需要判断图 $G$ 中是否存在一条从 $\mathsf{h}$ 到 $\mathsf{h}$ 的回路。先进行DFS,确保字符串 $S$ 中所有字符都被访问过。遍历每一条以 $\mathsf{h}$ 为起点的边 $(\mathsf{h} \to x)$,判断图 $G$ 中是否存在从 $x$ 到 $\mathsf{h}$ 的简单路径。

也可以把边反向以后建图,这样只要对 $\mathsf{h}$ 调用一次 DFS 就可以了。

Implementation

#include <bits/stdc++.h>
using namespace std; int main() {
//freopen("main.in", "r", stdin);
int n; cin >> n;
string s; cin >> s;
vector<int> cnt(26);
for(auto ch: s) cnt[ch-'a']++;
vector<vector<int>> g(26);
vector<bool> to_h(26);
bool flag = false;
while (n--) {
char x, y; cin >> x >> y;
if (x != y) {
g[y-'a'].push_back(x-'a');
if(x == 'h') {
to_h[y-'a'] = true;
flag = true;
}
}
} function<void(int)> dfs;
vector<bool> vis(26);
dfs = [&](int u) {
vis[u] = true;
for(auto v: g[u]) {
if(!vis[v]) dfs(v);
}
};
int h = 'h' - 'a';
dfs(h); int ans = 0;
for (int i = 0; i < 26; i++)
if(vis[i]) ans += cnt[i];
if (flag) {
for (int i = 0; i < 26; i++)
if (to_h[i] && vis[i]) {
flag = false;
break;
}
if (flag) ans -= cnt[h];
}
cout << ans << endl;
return 0;
}

hihoCoder #1902 字符替换的更多相关文章

  1. strtr和str_replace字符替换函数

    (一)strtr是字符替换函数 (1)单个字符替换: <?php echo strtr("abba", "ab", "10"),&qu ...

  2. AC日记——字符替换 openjudge 1.7 08

    08:字符替换 总时间限制:  1000ms 内存限制:  65536kB 描述 把一个字符串中特定的字符全部用给定的字符替换,得到一个新的字符串. 输入 只有一行,由一个字符串和两个字符组成,中间用 ...

  3. SQL 字符替换

    --匹配所有字符替换 )),'被替换','替换') --匹配给定位子替换 update 表名 set 列=stuff(列名,从一开始数位数,往后数几位,替换)

  4. Oracle的字符替换函数translate用法

    参考文档如下:http://www.banping.com/2009/05/18/oracle_function_translate/ Oracle提供了一个字符替换函数translate,不同于re ...

  5. R语言实现两文件对应行列字符替换(解决正负链统一的问题)

    假设存在文件file1.xlsx,其内容如下: 存在文件file2.xlsx,其内容如下: 现在我想从第七列开始,将file2所有的字符替换成file1一样的,即第七.八.九.十列不需要改变,因为fi ...

  6. Python replace() 和 re.sub() 字符串字符替换

    Python replace() 和 re.sub() 字符串字符替换 replace() testStr = 'aa:bb[cc' testStr.replace(':','_') 每次只能替换一个 ...

  7. js中字符替换函数String.replace()使用技巧

    定义和用法 replace() 方法用于在字符串中用一些字符替换另一些字符,或替换一个与正则表达式匹配的子串. 语法 stringObject.replace(regexp/substr,replac ...

  8. PHP基础--strtr和str_replace字符替换函数

    (一)strtr是字符替换函数 (1)单个字符替换: <?php echo strtr("abba", "ab", "10"),&qu ...

  9. C# Replace字符替换函数

    它可以将字串内的字符替换为别的字符,可以嵌套使用,如下: 需要注意的是,它可以把字符替换为空,但不可以替换空字符,当不确定字符串是否为空时,可以进行以下判断,再替换: 示例的完整代码: string ...

随机推荐

  1. iOS新浪微博OAuth2.0认证代码

    #import "ViewController.h" #import "AFNetworking.h" @interface ViewController () ...

  2. cordova创建工程添加插件

    创建工程 phonegap创建工程 代码 用以创建自己需要的  工程名   ; 报名  ;类名 ; 应用名 cordova create hello com.example.hello HelloWo ...

  3. spring-AspectJ

    动态代理 ProxyFactoryBean织入切面数量太多不利于围护 BeanNameAutoProxyCreater-------------根据Bean名称创建代理 DefaultAdvisorA ...

  4. Docker自学纪实(六)搭建docker私有仓库

    docker的镜像仓库分两种:一种是从官方公有仓库拉取:还有就是自己搭建私有仓库.官方的镜像仓库是面对整个应用市场的:私有仓库一般用于公司内部,就是公司项目自身所需的镜像.搭建私有仓库有什么好处?私有 ...

  5. Linux终端显示控制字符

    在Linux中, 我们时常要将一个命令的输出作为另外一个命令的输入进行下一步处理操作. 有时, 如果一个命令的输出中有不可见的控制字符时, 有可能会导致后续操作出错. 而这些控制字符很可能是不可打印的 ...

  6. webmin纯web界面管理linux系统

    关键字: 摘要:从Windows环境的管理转到Linux环境的管理时所面临的挑战之一是,您需要去学习利用新的工具.作为一个管理员,您希望理解操作系统的细节以发挥它的最大功效.但是,当您还处在学习阶段时 ...

  7. python简介,数据类型,input,if语句

      1. python的起源 python的创始人为吉多·范罗苏姆(龟叔Guido van Rossum),1989年的圣诞节期间,龟叔为了在阿姆斯特丹打发时间 决心开发一个新的脚本程序解释器,作为A ...

  8. 10.1.2 Document类型【JavaScript高级程序设计第三版】

    JavaScript 通过Document 类型表示文档.在浏览器中,document 对象是HTMLDocument(继承自Document 类型)的一个实例,表示整个HTML 页面.而且,docu ...

  9. VIM安装YCM插件

    折腾了两天,终于好了 1.配置VIM (1)下载相关插件 sudo apt-get install git sudo apt-get install build-essential cmake sud ...

  10. 翻译 | “扩展asm”——用C表示操作数的汇编程序指令

    本文翻译自GNU关于GCC7.2.0版本的官方说明文档,第6.45.2小节.供查阅讨论,如有不当处敬请指正…… 通过扩展asm,可以让你在汇编程序中使用C中的变量,并从汇编代码跳转到C语言标号.在汇编 ...