题目描述

输入两个DFA,判断是否等价。

https://uva.onlinejudge.org/external/16/1671.pdf

输入

第一行T 可以接受的字母表

第二行N 状态数

接下去N行 每个状态的情况 第一个表示是否为接受状态 后面T个是接受字母i可以转移到的状态 -1表示不转移

输出

等价输出Yes,不等价输出No

样例输入

2

3

1 -1 1

0 -1 2

0 -1 0

2

1 -1 1

0 -1 0

3

4

1 -1 -1 1

1 -1 -1 2

1 -1 -1 3

1 -1 -1 1

2

1 -1 -1 1

1 -1 -1 0

0

样例输出

Case #1: No

Case #2: Yes

思路

参考刘神的思路,求DFA2的补,然后与DFA1测试相交。如果不相交,当然是等价。

求补的思路就是把接受状态全都取反。求相交的思路就是BFS,或者DFS。我用的是BFS,很快,复杂度O(n)。

一个重要的细节,我在这上面卡了很久,就是怎么处理-1的情况。最后的解法是设一个孤岛结点,孤岛的所有转移都指向自己,-1的情况就转移到孤岛,孤岛自己不是接受状态。这样做的正确性:原DFA可以接受的状态集没有改变。因为我只需要知道那些输入可以接受,其余的就是不接受的输入。

但是如果不要孤岛,怎么直接处理-1的情况?

首先,如果遇到-1直接pass,那么处理不了

DFA1: 1->2->3->4->5(ac)

DFA2: 1->2->3->4->5(ac)->6(ac)

或者

DFA2: 1->2->3->4->5(ac)->6->7-> ... ->n(ac)

而且,处理起来很麻烦,情况很多,所以还是加个孤岛结点吧。

实现

#include <iostream>
#include <algorithm>
#include <vector>
#include <queue>
#include <stack>
#include <cstdio>
#include <cstdlib>
#include <cstring>
using namespace std; const int maxn = 2010; // size of state
const int maxm = 30; // size of alphabet int T; // T <= 26
int n1, n2; // n <= 2000
bool is_final1[maxn], is_final2[maxn];
int dfa1[maxn][maxm], dfa2[maxn][maxm]; bool init()
{
scanf("%d", &T);
if (T == 0)
return false; scanf("%d", &n1);
for (int i = 0; i < n1; i++) {
int f1;
scanf("%d", &f1);
is_final1[i] = ((f1 == 1) ? true : false);
for (int j = 0; j < T; j++) {
scanf("%d", &dfa1[i][j]);
dfa1[n1][j] = n1;
if (dfa1[i][j] == -1) // 孤岛状态
dfa1[i][j] = n1;
}
}
is_final1[n1] = 0; // 孤岛 scanf("%d", &n2);
for (int i = 0; i < n2; i++) {
int f2;
scanf("%d", &f2);
is_final2[i] = ((f2 == 1) ? true : false);
for (int j = 0; j < T; j++) {
scanf("%d", &dfa2[i][j]);
dfa2[n2][j] = n2;
if (dfa2[i][j] == -1) // 孤岛状态
dfa2[i][j] = n2;
}
}
is_final2[n2] = 0;
return true;
} bool visit[maxn][maxn];
bool bfs(int a1[maxn][maxm], int a2[maxn][maxm], bool b1[maxn], bool b2[maxn])
{
memset(visit, false, sizeof visit);
queue<pair<int,int> > Q; Q.push(make_pair(0, 0));
visit[0][0] = true; while (!Q.empty()) {
pair<int,int> P = Q.front();
Q.pop();
if (b1[P.first] && !b2[P.second]) // 2求补集
return true; // 发现相交 for (int i = 0; i < T; i++) {
// bfs 遍历未访问转移状态
if (visit[a1[P.first][i]][a2[P.second][i]] == false) {
visit[a1[P.first][i]][a2[P.second][i]] = true;
Q.push(make_pair(a1[P.first][i], a2[P.second][i]));
}
}
}
return false; // 不相交
} int main()
{
int kase = 0;
while (init()) {
printf("Case #%d: ", ++kase);
if (!bfs(dfa1, dfa2, is_final1, is_final2)
&& !bfs(dfa2, dfa1, is_final2, is_final1))
printf("Yes\n");
else
printf("No\n");
}
return 0;
}

[uva] 1671 History of Languages的更多相关文章

  1. UVa 111 - History Grading (by 最长公共子序列 )

     History Grading  Background Many problems in Computer Science involve maximizing some measure accor ...

  2. uva 111 History Grading(最长公共子序列)

    题目连接:111 - History Grading 题目大意:给出一个n 代表序列中元素的个数, 然后是一个答案, 接下来是若干个同学的答案(直到文件结束为止), 求出两个序列的最长公共子序列, 注 ...

  3. UVA, 10336 Rank the Languages

    难点在于:递归函数和输出: #include <iostream> #include <vector> #include <algorithm> #include ...

  4. UVA 111 History Grading

    读题读了好久,其实就是在输入数据时要对数据的位置进行相应的改变 #include<iostream> #include<cstring> #include<cstdio& ...

  5. uva 111 - History Grading (dp, LCS)

    题目链接 题意:给N,第二行是答案,n个数c1---cn, 代表第一个的顺序是c1,第二个数顺序是c2; 下面每一行是学生的答案,格式同上. 注意:这个给的顺序需要处理一下,不能直接用. 思路:LCS ...

  6. UVa 111 History Grading (简单DP,LIS或LCS)

    题意:题意就是坑,看不大懂么,结果就做不对,如果看懂了就so easy了,给定n个事件,注意的是, 它给的是第i个事件发生在第多少位,并不是像我们想的,第i位是哪个事件,举个例子吧,4 2 3 1, ...

  7. uva 111 History Grading(lcs)

    题目描述 在信息科学中有一些是关于在某些条件限制下,找出一些计算的最大值. 以历史考试来说好了,学生被要求对一些历史事件根据其发生的年代顺序来排列.所有事件顺序都正确的学生无疑的可以得满分.但是那些没 ...

  8. UVA 10887 Concatenation of Languages 字符串hash

    题目链接:传送门 题意: 给你两个集合A,B,任意组合成新的集合C(去重) 问你最后C集合大小 题解: 暴力 组成的新串hash起来 #include<bits/stdc++.h> usi ...

  9. UVa 1671 语言的历史——判断两个DFA是否等价

    题意 一个DFA可以用一个5元组  $((Q, \sum , \delta , q_0, F))$ 表示,其中 $Q$ 为状态集,$\sum$ 为字母表,$\delta$ 为转移函数,$q_0$ 为起 ...

随机推荐

  1. 【Win10】一些零碎不好归档的小总结(原谅我这个该死的标题吧)

    一.同步方式获取设备的屏幕分辨率 public static class ScreenResolution { /// <summary> /// 获取屏幕高度. /// </sum ...

  2. C#模拟请求,模拟登录,Cookie设置、文件上传等问题汇总

    由于业务需求,最近需要模拟完成登陆某个网站,并上传所需要的文件.在开发途中,遇到了很多问题,现在,就我遇到的一些问题及解决办法说明如下,希望对遇到同样问题的人有所帮助.因为技术有限,可能有些内容并不完 ...

  3. [ASP.NET]JQuery直接调用asp.net后台WebMethod方法

    在项目开发碰到此类需求,特此记录下经项目验证的方法总结. 利用JQuery的$.ajax()可以很方便的调用asp.net的后台方法. [WebMethod] 命名空间 1.无参数的方法调用 注意:方 ...

  4. K8S+GitLab-自动化分布式部署ASP.NET Core(一) 部署环境

    一.部署流程介绍 开发人员通过Git上传asp.net core 项目到Gilab,并编写好.gitlab-ci.yml , GitLab-Runner 自动拉取代码,然后进行Build,编译,单元测 ...

  5. 使用filter进行登录验证,并解决多次重定向问题

    最近在做关于filter登录验证的功能,防止未登录的用户直接通过地址进入系统 LoginFilter类:继承Filter接口 package com.ss.filter; import java.io ...

  6. forname,newInstance的作用及使用

    Forname可以获得类名对应的class对象: String classname=“java.util.Date” Class cl=Class.forName(className); newIns ...

  7. Hector 入门

    导入jar包 为了能够使用hector操作Cassandra数据库,首先导入hector的jar包,注意根据实际情况修改版本号 <dependency> <groupId>me ...

  8. [JavaScript] iframe加载完成事件

    //iframe加载完成后,对其子元素进行操作 var iframe = document.getElementById("re-img"); if (iframe.attachE ...

  9. 【javascript】iOS Safari 中点击事件失效的解决办法

    问题描述 当使用委托给一个元素添加click事件时,如果事件是委托到 document 或 body 上,并且委托的元素是默认不可点击的(如 div, span 等),此时 click 事件会失效. ...

  10. zookeeper集群环境搭建详细图文教程

    zookeeper集群环境搭建详细图文教程 zhoubang @ 2018-01-02 [文档大纲] 友情介绍 软件环境 注意点 环境安装 1. 新建用于存储安装包以及软件安装的目录 2. 下载安装z ...