Nested Dfs算法——判断Büchi自动机接受语言是否为空

2023-12-10 19:40

本文主要是介绍Nested Dfs算法——判断Büchi自动机接受语言是否为空,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

  • 前言
  • Büchi自动机判断接受语言是否为空
  • 算法伪码
  • 算法解释
  • 代码实现


前言

本文是写给学生看的系统分析与验证笔记(十二)——验证ω-正则属性的代码实现篇,因为想把课程笔记与自己的实现代码分离,所以就单独列出来

该算法在《Model Checking》与《Principles of Model Checking》上均给出了解释与伪码,实现由我自己实现,部分基础性的代码来自《算法》(第四版)的图部分。

Büchi自动机判断接受语言是否为空

根据前文所说,假设p是一个Büchi自动机的可接受运行,那么p中肯定会经过接受状态F无限多次,因此,p中肯定有一部分的序列重复了无限多次,这重复的状态肯定处在一个环里面,也就是说,这些个状态节点组成了强连通图。

基于上述分析,我们将Büchi自动机A接受的语言是否空的问题,转化成如下的形式

如果语言L(A)是非空的,当且仅当有接受状态在一个环里面出现

Tarjan提出的查找强连通图上的深度优先遍历算法(DFS)可以在时间复杂度O(|Q|+|Δ|)内判定一个Büchi自动机是否为空,下面介绍的是一种更加有效的算法,该算法采用双DFS来查找带有接受状态的环,因为嵌套地使用了DSF,该算法被称为Nested Dfs。

算法伪码

在这里插入图片描述

算法解释

算法输入:

  • Büchi自动机A所代表的的图
  • 初始状态集
  • 接受状态集

算法输出:

  • 是否存在接受状态在图中的环上

我把算法的流程分为如下几个步骤:
1、从初始状态出发,利用深度优先算法搜索某个接受状态是否是可达的
2、如果接受状态是可达的,那么从该接受状态出发,利用第二个深度优先算法搜索该接受状态是否处于在一个环上,同时,标记搜索过的状态
3、如果发现该接受状态处于环上,则终止搜索,算法输出True以及对应的环的路径,如果该接受状态不在环上,则继续第一个搜索,看看下一个接受状态是否可达,并尝试找环
4、直到找到了某个接受状态处于环上,或者搜索完了所有的接受状态,算法结束

代码实现

Digraph.java 带权有向图的数据结构类,采用的是邻接表的方式

import java.util.ArrayList;
import java.util.LinkedHashMap;
import java.util.List;/*** Created on 2020/12/4.* Description:带权有向图类*/
public class Digraph {private int V;  // 有向图的顶点private int E;  // 有向图的边private LinkedHashMap<Integer, List<Integer>> adj;  // 邻接表/*** 使用顶点数初始化有向图** @param V* @throws IllegalArgumentException if(@code V < 0)*/public Digraph(int V) {if (V < 0) throw new IllegalArgumentException("Number of vertices in a Digraph must be nonnegative");this.V = V;this.E = 0;adj = new LinkedHashMap<>();for (int v = 0; v < V; v++)adj.put(v, new ArrayList<>());}/*** 使用另一个图来初始化图** @param digraph*/public Digraph(Digraph digraph) {this.V = digraph.V();this.E = digraph.E();this.adj = digraph.adj();}/*** 返回顶点数** @return*/public int V() {return V;}/*** 返回边的数量** @return*/public int E() {return E;}/*** 返回邻接表** @return*/public LinkedHashMap adj() {return adj;}/*** 添加一条边,id1指向id2** @param id1,id2*/public void addEdge(Integer id1, Integer id2) {// 判断该边是否已经存在for (Integer i: (List<Integer>)adj(id1)) {// 如果已经存在,那么直接返回if (i.equals(id2) )return;}// 如果不存在则加入该边((List<Integer>) adj(id1)).add(id2);}/*** 查找某个顶点的邻接链表** @param id* @return*/public Iterable adj(Integer id) {validateVertex(id);return adj.get(id);}/*** 使用id验证顶点是否存在于图中** @param id*/private void validateVertex(Integer id) {if (adj.get(id) == null) {throw new IllegalArgumentException("vertex " + V + "not in the graph");}}@Overridepublic String toString() {return adj.toString();}
}

Nested Dfs算法类,用于处理算法的计算逻辑

  • marked是用于标记第一个Dfs算法遍历过的图节点
  • cycleMarked是用于标记第二个Dfs遍历过的图节点,这个标记可以提高算法的效率,因为如果在查找通过第一个接受状态的环时,没有找到环,那么说明再查找之后的接受状态时无需考虑上述已经被遍历过的节点
  • 关于查找通过某个接受状态环的算法,其实只需要利用Dfs算法,外加一个辅助栈,栈的作用是记录当前搜索中遍历的路径,如果进入深层的搜索就将状态压入栈,如果结束一次搜索就将状态弹出栈,如果查找到下一个状态就是搜索开始的状态,那么就说明找到了一个环,输出栈中的元素,就是一条环的路径(就如同边走迷宫边放线,走着走着发现从另外路径又回到了起点,那么就找到了一个环)

注意:这个算法只不过是普通查找环算法的小小变种,一般的查找环算法是判断下一个状态是否已经处于栈中,如果是就说明存在环,而这里找的是下一个状态是否是搜索开始的状态。所以该算法仍然只能判断是否存在通过接受状态的环,而且找到哪个环也要看遍历的循序,并不能找到所有的环。

import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Stack;/*** Created on 2020/12/5.* Description:嵌套dsf算法解决查找某可达节点是否存在通过该节点的环*/
public class NestedDfs {private Map marked;         // marked[v] 是否被标记,用于深度优先搜索private Map cycleMarked;         // cycleMarked[v] 是否被标记,用于查找环private Map edgeTo;            // edgeTo[v] 是否连通private Map onStack;       // onStack[v] 是否有顶点在栈里面private Stack<Integer> cycle;    // 递归调用的栈上的所有顶点public NestedDfs(Digraph graph, Integer[] initStates, Integer[] checkStates) {marked = new HashMap<Number, Boolean>();// 查看它是否能从初始状态开始可达for (Integer initState : initStates) {if (cycle != null) break;for (Integer checkState : checkStates) {dfs(graph, initState, checkState);if (cycle != null) break;}}}/*** 深度优搜索算法,判断两个点之间是否可达** @param G* @param initState* @param checkState*/private void dfs(Digraph G, Integer initState, Integer checkState) {marked.put(initState, true);// 如果有环节退出if (cycle != null)return;// 如果初始状态到检查状态是可达的,那么开始检查是否有通过检查状态的环if (initState.equals(checkState)) {onStack = new HashMap<Integer, Boolean>();edgeTo = new HashMap<Integer, Integer>();cycleMarked = new HashMap<Integer, Boolean>();dfsWithCheck(G, checkState, checkState);}for (Integer i : (List<Integer>) G.adj(initState)) {if (!(boolean) marked.getOrDefault(i, false))dfs(G, i, checkState);}}/*** 到某个顶点是否可达** @param id 顶点id* @return*/public boolean marked(Integer id) {return (boolean) marked.getOrDefault(id, false);}private void dfsWithCheck(Digraph G, Integer id, Integer checkId) {onStack.put(id, true);cycleMarked.put(id, true);for (Integer i : (List<Integer>) G.adj(id)) {if (this.hasCycle()) return;else if (!(Boolean) cycleMarked.getOrDefault(i, false)) {edgeTo.put(i, id);dfsWithCheck(G, i, checkId);} else if ((Boolean) onStack.getOrDefault(i, false) && i.equals(checkId)) {cycle = new Stack<>();for (Integer x = id; !x.equals(i); x = (Integer) edgeTo.get(x)) {cycle.push(x);}cycle.push(i);cycle.push(id);}}onStack.put(id, false);}/*** 是否有循环,有返回true,没有返回false** @return cycle is null?*/public boolean hasCycle() {return cycle != null;}/*** 返回环迭代序列** @return cycle*/public Iterable cycle() {return cycle;}
}

Main.java用于测试算法的准确性
测试用例就选取了红绿灯的例子,给每个TS中的状态表上序号,便于程序的处理,第一个不存在通过接受状态的环,输出为null,第二个存在,输出环的路径
在这里插入图片描述
在这里插入图片描述

public class Main {public static void main(String[] args) {// 构建图Digraph graph = new Digraph(6);graph.addEdge(0,1);graph.addEdge(1,0);graph.addEdge(1,2);graph.addEdge(2,4);graph.addEdge(3,2);graph.addEdge(4,5);graph.addEdge(5,4);// 输入初始节点和需要判断的点Integer[] initStates = new Integer[]{0,2};Integer[] checkStates = new Integer[]{2};NestedDfs nestedDfs = new NestedDfs(graph, initStates, checkStates);// 输出是否有通过代检查节点的环 以及 输出环的路径System.out.println(nestedDfs.hasCycle());System.out.println(nestedDfs.cycle());// 测试例子2Digraph graph2 = new Digraph(9);graph2.addEdge(0,1);graph2.addEdge(1,0);graph2.addEdge(1,2);graph2.addEdge(2,1);graph2.addEdge(0,4);graph2.addEdge(1,3);graph2.addEdge(3,4);graph2.addEdge(4,3);graph2.addEdge(2,4);graph2.addEdge(5,4);graph2.addEdge(4,8);graph2.addEdge(6,7);graph2.addEdge(7,6);graph2.addEdge(7,8);graph2.addEdge(8,7);Integer[] initStates2 = new Integer[]{1,4};Integer[] checkStates2 = new Integer[]{3,4,5};NestedDfs nestedDfs2 = new NestedDfs(graph2, initStates2, checkStates2);System.out.println(nestedDfs2.hasCycle());System.out.println(nestedDfs2.cycle());}
}

这篇关于Nested Dfs算法——判断Büchi自动机接受语言是否为空的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/478153

相关文章

不懂推荐算法也能设计推荐系统

本文以商业化应用推荐为例,告诉我们不懂推荐算法的产品,也能从产品侧出发, 设计出一款不错的推荐系统。 相信很多新手产品,看到算法二字,多是懵圈的。 什么排序算法、最短路径等都是相对传统的算法(注:传统是指科班出身的产品都会接触过)。但对于推荐算法,多数产品对着网上搜到的资源,都会无从下手。特别当某些推荐算法 和 “AI”扯上关系后,更是加大了理解的难度。 但,不了解推荐算法,就无法做推荐系

康拓展开(hash算法中会用到)

康拓展开是一个全排列到一个自然数的双射(也就是某个全排列与某个自然数一一对应) 公式: X=a[n]*(n-1)!+a[n-1]*(n-2)!+...+a[i]*(i-1)!+...+a[1]*0! 其中,a[i]为整数,并且0<=a[i]<i,1<=i<=n。(a[i]在不同应用中的含义不同); 典型应用: 计算当前排列在所有由小到大全排列中的顺序,也就是说求当前排列是第

csu 1446 Problem J Modified LCS (扩展欧几里得算法的简单应用)

这是一道扩展欧几里得算法的简单应用题,这题是在湖南多校训练赛中队友ac的一道题,在比赛之后请教了队友,然后自己把它a掉 这也是自己独自做扩展欧几里得算法的题目 题意:把题意转变下就变成了:求d1*x - d2*y = f2 - f1的解,很明显用exgcd来解 下面介绍一下exgcd的一些知识点:求ax + by = c的解 一、首先求ax + by = gcd(a,b)的解 这个

综合安防管理平台LntonAIServer视频监控汇聚抖动检测算法优势

LntonAIServer视频质量诊断功能中的抖动检测是一个专门针对视频稳定性进行分析的功能。抖动通常是指视频帧之间的不必要运动,这种运动可能是由于摄像机的移动、传输中的错误或编解码问题导致的。抖动检测对于确保视频内容的平滑性和观看体验至关重要。 优势 1. 提高图像质量 - 清晰度提升:减少抖动,提高图像的清晰度和细节表现力,使得监控画面更加真实可信。 - 细节增强:在低光条件下,抖

【数据结构】——原来排序算法搞懂这些就行,轻松拿捏

前言:快速排序的实现最重要的是找基准值,下面让我们来了解如何实现找基准值 基准值的注释:在快排的过程中,每一次我们要取一个元素作为枢纽值,以这个数字来将序列划分为两部分。 在此我们采用三数取中法,也就是取左端、中间、右端三个数,然后进行排序,将中间数作为枢纽值。 快速排序实现主框架: //快速排序 void QuickSort(int* arr, int left, int rig

poj 3974 and hdu 3068 最长回文串的O(n)解法(Manacher算法)

求一段字符串中的最长回文串。 因为数据量比较大,用原来的O(n^2)会爆。 小白上的O(n^2)解法代码:TLE啦~ #include<stdio.h>#include<string.h>const int Maxn = 1000000;char s[Maxn];int main(){char e[] = {"END"};while(scanf("%s", s) != EO

poj 3259 uva 558 Wormholes(bellman最短路负权回路判断)

poj 3259: 题意:John的农场里n块地,m条路连接两块地,w个虫洞,虫洞是一条单向路,不但会把你传送到目的地,而且时间会倒退Ts。 任务是求你会不会在从某块地出发后又回来,看到了离开之前的自己。 判断树中是否存在负权回路就ok了。 bellman代码: #include<stdio.h>const int MaxN = 501;//农场数const int

科研绘图系列:R语言扩展物种堆积图(Extended Stacked Barplot)

介绍 R语言的扩展物种堆积图是一种数据可视化工具,它不仅展示了物种的堆积结果,还整合了不同样本分组之间的差异性分析结果。这种图形表示方法能够直观地比较不同物种在各个分组中的显著性差异,为研究者提供了一种有效的数据解读方式。 加载R包 knitr::opts_chunk$set(warning = F, message = F)library(tidyverse)library(phyl

透彻!驯服大型语言模型(LLMs)的五种方法,及具体方法选择思路

引言 随着时间的发展,大型语言模型不再停留在演示阶段而是逐步面向生产系统的应用,随着人们期望的不断增加,目标也发生了巨大的变化。在短短的几个月的时间里,人们对大模型的认识已经从对其zero-shot能力感到惊讶,转变为考虑改进模型质量、提高模型可用性。 「大语言模型(LLMs)其实就是利用高容量的模型架构(例如Transformer)对海量的、多种多样的数据分布进行建模得到,它包含了大量的先验

秋招最新大模型算法面试,熬夜都要肝完它

💥大家在面试大模型LLM这个板块的时候,不知道面试完会不会复盘、总结,做笔记的习惯,这份大模型算法岗面试八股笔记也帮助不少人拿到过offer ✨对于面试大模型算法工程师会有一定的帮助,都附有完整答案,熬夜也要看完,祝大家一臂之力 这份《大模型算法工程师面试题》已经上传CSDN,还有完整版的大模型 AI 学习资料,朋友们如果需要可以微信扫描下方CSDN官方认证二维码免费领取【保证100%免费