信息学奥赛一本通 1396:病毒(virus)

【题目链接】

ybt 1396:病毒(virus)

【题目考点】

1. 图论:拓扑排序

【解题思路】

原字典中的单词是按照字母顺序排列的,指的就是原文件中的单词都是按照字典序顺序排列的。

既然是字典,可以认为其中不存在相同的单词。

也就是下面的单词在字典序意义下一定大于上面的单词。

假设存在被病毒感染后的字符a、b,分别是图中的一个顶点。如果a被病毒感染前的字符的ASCII码小于b被病毒感染前的字符的ASCII码,那么认为顶点a到顶点b有一条有向边。

读入被病毒感染后的一个字符串s,和上一次读入的字符串ls进行比较,假设二者第一个不同的字符(包括字符串末尾的'\0')是第i个字符,也就是s[i]ls[i],那么根据字典序的定义,s[i]在被病毒感染前的字符的ASCII码一定大于ls[i]在被病毒感染前的ASCII码,也就是顶点ls[i]到顶点s[i]有一条有向边。

遍历被病毒感染后的字典,根据以上方法建图。

要想得到完整的感染前后所有字符的一一对应关系,那么该有向图无环且其拓扑排序序列应该是唯一的。(如果初始情况或删掉某个顶点后,入度为0的顶点始终只有1个,那么拓扑排序是唯一的)

得到的拓扑排序序列即为a~z字符序列被感染后变成的字符序列。

注意:给定的感染后的字符只是a~z中的部分字符,将这些字符保存在一个set中。访问顶点时,遍历这个set就可以访问到所有的顶点。

对于最后输入的待还原的字符串,也要先判断其中的字符是否都是保存感染后字符的set中的字符,如果存在其它字符,直接输出0。

【题解代码】

解法1:拓扑排序
cpp 复制代码
#include<bits/stdc++.h>
using namespace std;
#define N 30
int edge[N][N], deg[N];
set<int> st;//保存被感染后的所有字符 
map<char, char> mp;//mp[i]:感染后的字符i原来是什么字符 
bool topoSort()//如果返回值为false,则该有向图没有拓扑排序或没有唯一拓扑排序 
{
	int ct = 0;
	char alph = 'a';
	queue<int> que;
	for(int v : st) if(deg[v] == 0)
	{
		que.push(v);
		ct++;
	}
	if(ct > 1)//入度为0的应该只有1个 
		return false;
	while(!que.empty())
	{
		int u = que.front();
		que.pop();
		mp[u+'a'] = alph++;//拓扑排序序列对应a,b,c... 
		ct = 0;
		for(int v : st) if(edge[u][v] && --deg[v] == 0)
		{
			que.push(v);
			ct++;
		}
		if(ct > 1)//删掉u后,入度为0的应该只有1个 
			return false;
	}
	return mp.size() == st.size();//st中每个字符都有对应的字符,图中才没有环,是完成了拓扑排序 
}
int main()
{
	string s, ls, t;//ls:上一个字符串
	int k;
	cin >> k;
	for(int j = 1; j <= k; ++j)
	{
		cin >> s;
		for(char c : s)//把所有感染后的字符串中的字符加入st 
			st.insert(c-'a');
		if(j == 1)//第一次循环只记录ls
		{
			ls = s; 
			continue;
		}
		for(int i = 0; i < s.length(); ++i) 
		{
			if(s[i] != ls[i])
			{
				if(ls[i] != '\0')
				{
					edge[ls[i]-'a'][s[i]-'a'] = 1;//ls[i]到s[i]有一条边
					deg[s[i]-'a']++;
				}
				break;
			}
		}
		ls = s;
	}
	cin >> t;//待转换字符串
	bool hasAns = topoSort();
	for(char c : t) if(st.count(c-'a') == 0)//如果出现不存在的字符 
		hasAns = false;
	if(hasAns)
		for(char c : t)
			cout << mp[c];
	else
		cout << 0;
	return 0;
}
相关推荐
洛水水7 分钟前
【力扣100题】33.验证二叉搜索树
算法·leetcode·职场和发展
SimpleLearingAI19 分钟前
聚类算法详解
算法·数据挖掘·聚类
刀法如飞1 小时前
Go 字符串查找的 20 种实现方式,用不同思路解决问题
算法·面试·程序员
Dlrb12113 小时前
C语言-指针数组与数组指针
c语言·数据结构·算法·指针·数组指针·指针数组·二级指针
WL_Aurora3 小时前
Python 算法基础篇之集合
python·算法
平行侠3 小时前
A15 工业路由器IP前缀高速检索与内存压缩系统
网络·tcp/ip·算法
阿旭超级学得完4 小时前
C++11包装器(function和bind)
java·开发语言·c++·算法·哈希算法·散列表
li星野4 小时前
位运算 & 数学 & 高频进阶九题通关(Python + C++)
c++·python·学习·算法
hnjzsyjyj4 小时前
洛谷 T145300:这是一棵树吗? ← 图论握手定理
图论·握手定理
jerryinwuhan4 小时前
hello算法,简单讲(1)
算法·排序算法