当前位置: 首页 > article >正文

信息学奥赛一本通 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:拓扑排序
#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;
}

http://www.kler.cn/a/523048.html

相关文章:

  • 小白爬虫冒险之反“反爬”:无限debugger、禁用开发者工具、干扰控制台...(持续更新)
  • Kotlin开发(六):Kotlin 数据类,密封类与枚举类
  • RocketMQ原理—5.高可用+高并发+高性能架构
  • git困扰的问题
  • 对游戏宣发的粗浅思考
  • AWTK 骨骼动画控件用法
  • c++多态
  • JavaScript逆向高阶指南:突破基础,掌握核心逆向技术
  • Nginx 开发总结
  • 《网络数据安全管理条例》施行,企业如何推进未成年人个人信息保护(上)
  • 深入探索C++17的std::any:类型擦除与泛型编程的利器
  • STM32 LED呼吸灯
  • pycharm(2)
  • noteboolm 使用笔记
  • 面向对象编程简史
  • Facebook如何应对全球范围内的隐私保护挑战
  • Python vLLM 实战应用指南
  • OpenCV:图像轮廓
  • 二叉树的最大深度(遍历思想+分解思想)
  • 算法随笔_28:最大宽度坡_方法2
  • Windows11无法打开Windows安全中心主界面
  • 【llm对话系统】大模型RAG之基本逻辑
  • python实现dbscan
  • 【huawei】云计算的备份和容灾
  • 基于vue和elementui的简易课表
  • skynet 源码阅读 -- 核心概念服务 skynet_context