此次发表的是一个不确定的自动机(NFA),它可以根据输入的正规式输出由函数映像表示的结果。

  此版本可以输入括号’(‘,‘)’,但是,实现的过程理解起来有点吃力,所以,在时间允许的情况下,我还将写新文章,使用单纯递归方法实现该程序。

#include"stdio.h"
#include"stdlib.h"
#define MAX 200 struct Stack
{
int Stack[MAX];
int top;
}St; struct Queue
{
int front;
int rear;
int elem[MAX];
}Q; void Input(char String[]);
void initStack(); //初始化栈
void PushStack(int x); //进栈
int PopStack(); //出栈
void initQueue();
void InserQ(int x);
int Gethead();
void decompose(char String[]);
int deal_1(char String[],int front,int rear);//处理a.b.a...
int deal_2(char String[],int front,int rear);//处理a|b|a....
int deal_3(char String[],int front,int rear);//处理a*
void cut_string(int start, int end, char String[], char temp[]); int num = ; main()
{
char String[MAX];
int i;
while(){initStack();
Input(String);
//printf("%s",String);
decompose(String);} } void Input(char String[])
{
char temp;
int i = ;
printf("请输入正规式:");
do{
temp = getchar();
//printf("%c",temp);
String[i] = temp;
i++;
}while(temp != '\n'); } void initStack()
{
St.top = -;
} void PushStack(int x)
{
St.top++;
St.Stack[St.top] = x;
}
int PopStack()
{
int temp;
temp = St.Stack[St.top];
St.top--;
return temp;
} void initQueue()
{
Q.front = Q.rear = ;
} void InserQ(int x)
{
if((Q.rear + )%MAX == Q.front)
return;
else
{
Q.elem[Q.rear] = x;
Q.rear = (Q.rear + ) % MAX;
}
} int Gethead()
{
int x;
if(Q.rear == Q.front)
printf("null\n");
else
{
x = Q.elem[Q.front];
Q.front = (Q.front+)%MAX;
}
return x;
} void decompose(char String[])
{
int flag_index = ;
char x[MAX];
int temp;
int i;
while(String[flag_index] != '\n')
{
if(String[flag_index] == '(')
{
PushStack(flag_index+);
flag_index++;
while(String[flag_index] != '\n' && St.top != -)
{
if(String[flag_index] == '(')
{
PushStack(flag_index+);
flag_index++;
}
else if(String[flag_index] == ')')
{
temp = PopStack();
flag_index++;
if(String[flag_index] == '*')
{
cut_string(temp-,flag_index,String,x);
deal_3(x,num,num);
flag_index++;
}
else
{
cut_string(temp,flag_index-,String,x);
decompose(x);
} }
else flag_index++;
}
}
for(i = flag_index ; String[i] != '\n' && String[i] != '('; i++);
if(i != flag_index)
{
// printf("1");
cut_string(flag_index,i-,String,x);
num++;
deal_2(x,num-, num); } //printf("%s",cut_string(flag_index,i - 1,String));
flag_index = i; } } int deal_1(char String[],int front,int rear)
{
int flag_index = ;
//printf("%s",String);
if(String[flag_index+] != '\n')
{
printf("f(%d, %c) = %d\n",front,String[flag_index],num);
flag_index++;
while(String[flag_index+] != '\n')
{
//printf("%c",String[flag_index]);
if(String[flag_index]>='a' && String[flag_index]<='z')
{
printf("f(%d, %c) = %d\n",num,String[flag_index],num+);
flag_index++;
num++;
}
else
flag_index++;
}
// printf("%c",String[6]);
printf("f(%d, %c) = %d\n",num,String[flag_index+],rear);
}
else
printf("f(%d, %c) = %d\n",front,String[flag_index],rear); } int deal_2(char String[],int front,int rear)
{
int flag_index = ;
int i = ;
int count = ;
char x[MAX];
int temp;
while(String[i] != '\n')
{
//printf("%c",String[flag_index]);
if(String[i] == '|')
InserQ(i++);
else i++;
}
if(Q.rear == Q.front)
{
for(i = ; String[i] != '\n'; i++)
{
if(String[i]>='a' && String[i]<='z')
count++;
}
deal_1(String,front,num + count - );
}
else
{
while(Q.rear != Q.front)
{
temp = Gethead();
cut_string(flag_index,temp-, String,x);
// printf("%d",temp);
num++;
deal_1(x, front, rear);
flag_index = temp+;
}
cut_string(flag_index,i, String,x);
deal_1(x, front, rear); }
} int deal_3(char String[],int front,int rear)
{
int flag_index = ;
int temp;
printf("f(%d, ~) = %d\n",front,++num);
temp = num;
if(String[flag_index+]==')')
{
printf("f(%d, %c) = %d\n",temp,String[],temp);
}
else
{
for(flag_index = ; String[flag_index] != '*'; flag_index++);
String[flag_index] = '\n';
decompose(String);
}
printf("f(%d, ~) = %d\n",temp,rear);
return front;
} void cut_string(int start, int end, char String[], char temp[])
{
int i;
end -= start;
for(i = ; i <= end ; i++)
{
temp[i] = String[start];
start++;
// printf("%c",temp[i]);
} temp[i] = '\n';
//printf("%d ",i);
// printf("%c",temp[i]);
}

  因为时间的关系,所以代码的注释也就没有太在意,望见谅。

NFA的实现的更多相关文章

  1. NFA转DFA - json数字识别

    json的主页上,提供了number类型的符号识别过程,如下: 图片引用:http://www.json.org/json-zh.html 实际上这张图片表示的是一个状态机,只是状态没有标出来.因为这 ...

  2. 求子串-KPM模式匹配-NFA/DFA

    求子串 数据结构中对串的5种最小操作子集:串赋值,串比较,求串长,串连接,求子串,其他操作均可在该子集上实现 数据结构中串的模式匹配 KPM模式匹配算法 基本的模式匹配算法 //求字串subStrin ...

  3. NFA引擎匹配原理

    1       为什么要了解引擎匹配原理 一个个音符杂乱无章的组合在一起,弹奏出的或许就是噪音,同样的音符经过作曲家的手,就可以谱出非常动听的乐曲,一个演奏者同样可以照着乐谱奏出动听的乐曲,但他/她或 ...

  4. 编译系统中的 NFA/DFA算法理解

    1.问题概述 NFA 和 DFA浅析---要深入了解正则表达式,必须首先理解有穷自动机. 有穷自动机(Finite Automate)是用来模拟实物系统的数学模型,它包括如下五个部分: 有穷状态集St ...

  5. C# 词法分析器(四)构造 NFA

    系列导航 (一)词法分析介绍 (二)输入缓冲和代码定位 (三)正则表达式 (四)构造 NFA (五)转换 DFA (六)构造词法分析器 (七)总结 有了上一节中得到的正则表达式,那么就可以用来构造 N ...

  6. nfa转dfa,正式完成

    为了加速转换的处理,我压缩了符号表.具体算法参考任何一本与编译或者自动机相关的书籍. 这里的核心问题是处理传递性闭包,transitive closure,这个我目前采取的是最简单的warshall算 ...

  7. 正则转nfa:完成

    太累了,感觉不会再爱了.问题已经解决,具体的懒得说了. #include "regular_preprocess.h" //这个版本终于要上nfa了,好兴奋啊 //由于连个节点之间 ...

  8. 正则转nfa:bug消除

    正则到nfabug的解决方法 前面提到了这个bug,为了解决这个bug,我们必须在每次引用到一个假名的时候,都构建一个拷贝.现在假设我们遇到了一个假名,并得到了他的开始节点和结束节点,当前的难题就是构 ...

  9. 正则转nfa:bug出现。

    本人写的一个正则到nfa的bug 刚写完前面的那篇,自己用脑子过了一下,发现了一个bug.具体情况如下. 这个bug的产生条件是多次调用假名的时候,每次调用都会修改假名的nfa图.直接这么说不好理解, ...

  10. 最初步的正则表达式引擎:nfa的转换规则。

    [在此处输入文章标题] 正则到nfa 前言 在写代码的过程中,本来还想根据龙书上的说明来实现re到nfa的转换.可是写代码的时候发现,根据课本来会生成很多的无用过渡节点和空转换边,需要许多的代码.为了 ...

随机推荐

  1. 设置双击直接打开.ipynb文件

    本文环境:win10(64)+anaconda3(64) anaconda3安装好后,可在开始菜单下查看 背景:jupyter notebook打开后无法更改路径,只能在默认路径下upload文件(如 ...

  2. CocoaPods的一些略为高级一丁点的使用【转】

    记得我刚开始做iOS开发的时候,是没有项目依赖管理工具.当需要引入第三方库的时候是相当麻烦的,不是直接拷贝库近来,就是添加依赖工程,直到CocoaPods出来才改变这个状况.项目依赖管理不是Cocoa ...

  3. VIM的字符编码设置

    vim 编码方式的设置和所有的流行文本编辑器一样,Vim 可以很好的编辑各种字符编码的文件,这当然包括UCS-2.UTF-8 等流行的 Unicode 编码方式.然而不幸的是,和很多来自 Linux ...

  4. Orcle导入导出dmp文件

    --Orcle导入导出dmp文件 --------------------------2013/12/06 导出表:   exp scott/tiger@mycon tables=(dept,emp) ...

  5. JavaScript获取和创建元素

    1.JavaScript中获取元素 常用的获取document中元素的方法: 1) document.getElementById()  =>通过元素ID获取文档中特定的元素,如获取 id = ...

  6. linux下利用mentohust校园拨号上网小记

    mentohust下载地址 :http://code.google.com/p/mentohust/downloads/list 安装就不多说了,我是在kali系统下使用的,经测试只有32为的可以使用 ...

  7. How to customize the console applicaton

    下面是如何最大化console和改变其显示的字体颜色的代码,顺便包含了计时代码(帮助做性能分析): class Program { [DllImport("kernel32.dll" ...

  8. Tomcat去除端口号和项目名(转)

    方法一: 在一个项目结束的时候,往往根据需要直接映射到一个IP地址上.此时就需要去除端口和项目名了,在访问的时候.下面根据具体的例子提供一种实现方法. 一.去掉项目名称 把<Context do ...

  9. 对于查询调优,你需要的不止STATISTICS IO

    在我查询调优期间,STATISTICS IO会话选项是我的朋友,因为对于指定的查询,它准确告诉你有多少页已读取.每次,SQL Server从缓存池骑牛一个8K的页,它通过STATISTICS IO的输 ...

  10. Words used when reading Redis documents

    Redis-----------------First pageevolution n.演变,进化,发展 closely adv.紧密地trade off 交换物品,权衡achieved adj.高度 ...