ser Generator使用说明

2024-03-10 19:30
文章标签 使用 说明 generator ser

本文主要是介绍ser Generator使用说明,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

分享到
  • 一键分享
  • QQ空间
  • 新浪微博
  • 百度云收藏
  • 人人网
  • 腾讯微博
  • 百度相册
  • 开心网
  • 腾讯朋友
  • 百度贴吧
  • 豆瓣网
  • 搜狐微博
  • 百度新首页
  • QQ好友
  • 和讯微博
  • 更多...

百度分享

“大声说出你的爱”情人节特别活动获奖名单发布      专访小恩爱CTO钟超:让数据决定创意功能的去留      电子版《程序员》杂志免费领

Parser Generator使用说明

5913人阅读 评论(7) 收藏 举报
generator library include yacc 多线程 token

Parser Generator的使用说明

最近1个星期,大致学习了一下lex,虽然在windows系统上它并没有我所期望的强大,在调试和编写代码都遇到了不少困难,但是总体来说Parser Generator还是让我体会到了lex编程的快捷, 为了自己加深印象把一些参考资料和认识总结了一下.

<!--[if !supportLists]-->1.     <!--[endif]-->parser generator是什么?

它是一款在windows下编写lex/yacc程序的工具.可以到

http://www.bumblebeesoftware.com/downloads.htm

下载

<!--[if !supportLists]-->2.     <!--[endif]-->parser generator的使用方法

这里只以vc6.0为例,首先打开parser generator编辑器,选择Project->LibBuilder

LibBuilder对话框中选中Visual C++(32-bit),按属性键Properties后确以下设置

Script file name                        ./Cpp/Script/msvc32.lbs

Name                                    Visual C++(32-bit)

Directory                               msvc32

Compiler Version                         Version 6

Unicode                                  True

Treat wchar_t as Built-in Type            False

Compiler Bin Directory             安装路径/Microsoft Visual Studio/Vc98/bin

Compiler Bin Directory(2) 安装路径/Microsoft Visual Studio/Common/MSDev98/bin

Compiler Include Directory      安装路径/Microsoft Visual Studio/Vc98/include

Compiler Include Directory(2)               

Compiler Library Directory          安装路径/Microsoft Visual Studio/Vc98/lib

Compiler Library Directory(2)               

Libraries下的库文件全部选中后ok

LibBuilder对话框->Build(编译过程可能几分钟)

编译完成后我们就可以使用parser generator编写lex或是yacc程序了

Project->ParserWizard

Step1 工程设定(一点需要注意语言可以选择c或是c++java)

Step2 工程设定(默认创建带main函数的yacc文件和lex文件)

Step3 yacc文件设定

Step4 lex文件设定

Lexyacc的语法参考http://www.ibm.com/developerworks/cn/linux/sdk/lex/

编辑好代码后Project->RebBuild All在你创建好的工程下自动生成Step1选定语言的文件(.h/..c/.cpp/.java)

之后在vc6.0加入如下设置

Tool->Option-> directory

Bin file :

安装目录/PARSER GENERATOR 2/BIN

Include file:

安装目录/PARSER GENERATOR 2/CPP/INCLUDE

Library file

安装目录/PARSER GENERATOR 2/CPP/LIB/MSVC32

Soure file

安装目录/PARSER GENERATOR 2/CPP/SOURCE

创建vc6.0工程

将生成文件复制到vc6.0创建工程下

Source filesHeader Files中加入生成文件(.h/.c/.cpp)

在工程设定中Project->Settings For box选中win32 debug

c/c++ ->Category选中General ->Preprocessor Definitions加入YYDEBUG

在工程Project设定Project->Settings For box中选中all

link -> Category选中General->Object/Library Modules中加入yld.lib

这里需要注意的是yld.libparser generatorDUBUG单线程版本,对于vc的控制台程序是可以的,如果使用了MFC或是Windows applications程序则需要对应下表追加

Library(DEBUG)

Run-time Library

Description

yld.lib

Debug Single-Threaded

单线程静态链接库(DEBUG版本)

ylmtd.lib

Debug Multithreaded

多线程静态链接库(DEBUG版本)

ylmtrd.lib

Debug Multithreaded DLL

多线程静态链接库当run time library 使用动态库(DEBUG版本)

ylmtrid.lib    

Debug Multithreaded DLL

多线程动态链接库当run time library 使用动态库(DEBUG版本)

Library(RELEASE)

Run-time Library

Description

yl.lib

Single-Threaded

单线程静态链接库(RELEASE版本)

ylmt.lib

Multithreaded

多线程静态链接库(RELEASE版本)

ylmtr.lib

Multithreaded DLL      

多线程静态链接库当run time library 使用动态库(RELEASE版本)

ylmtri.lib

Multithreaded DLL      

多线程动态链接库当run time library 使用动态库(RELEASE版本)

如果使用了动态库版本需要在程序运行环境中追加DLL的地址

安装目录/PARSER GENERATOR 2/CPP/LIB/MSVC32

如果需要链接yacc或是lexdll.Preprocessor Definitions下加入YYDLL.

这样就可以使用vc6.0lex生成文件进行编译了

3PG2的默认输FILE *,怎样转char *?
PG2
提供了两方法,一是重定yyinput,二是重定yygetchar实际使用中后者方便,因yyinput除了yygetchar之外需要负责lineno量的增加。
yygetchar通常这样
int yygetchar(void)
{
     if (* inputstring=='/0')
     {
           return -1;
     }
     return (unsigned char) * inputstring++;
}
唯一要注意的就是char *的末尾要返回一个-1代表EOF,使得yyinput停止。(感luocong的提醒)。

4
、同理,yyoutput
FILE *char *
直接
yytext操作既可,yytext就是char[]

3,4说明转自

http://blog.csdn.net/tankaiha/archive/2005/12/13/551457.aspx

总体来说lex/yacc比起自己编写分析程序要快的多,但是不足点就是错误的调试会非常困难,需要把.lex/.l.yacc/.y文件加入工程调试,每次变更都需要重新编译parser generator工程,然后再次粘贴,lex的语法熟练度可能决定了效率,新手基于模型为基础进行研究逐步认识parser generator是最佳的学习方法,

Lex的自带例程在/Parser Generator 2/Cpp/Examples ,在这对其中的class(四则计算器)做下简单说明

lexer.l文件

%{              //%{ %}中的c代码将被复制到.cpp文件中

#include <stdlib.h>

#include <math.h>

#include <assert.h>

#include "parser.h"//自动生成头文件必须包含

%}

// include file

%include {              //%include {}里的代码复制到对应的.h文件中,这里声明calc_parser

class calc_parser;   // symboltable是为在下面calc_lexer::create函数能够找到使用类型的提前

class symboltable;  //声明

}

// lexical analyser name

%name calc_lexer         //%name指定继承自lexer的类名{}中为类成员声明,类的声明将被解//释到.h文件中

// class definition

{

// Attributes

protected:

        symboltable* m_symboltable;             // the symbol table

// Operations

public:

        int create(calc_parser* parser, symboltable* symboltable);

       

        // attribute commands

        double number() const;

        symbol* id() const;

}

// constructor

{

        // do nothing

}

// macros

exponent        ([Ee][+-]?[0-9]+)  //科学计算方定义为exponent

%%

%{

// extract yylval for use later on in actions

YYSTYPE& yylval = *(YYSTYPE*)yyparserptr->yylvalptr;//指定yylval值的类型

%}

// 对于数字的规则处理 |表示或,|下一个算式使用的规则与第一个相同{}中是规则执行代码

[0-9]+"."[0-9]*{exponent}?      |

"."[0-9]+{exponent}?            |

[0-9]+{exponent}?                       { yylval.value = number(); return NUMBER; }

//对于3角函数的规则处理

"sin"                                           { return SIN; }

"cos"                                           { return COS; }

"tan"                                           { return TAN; }

// 对于变量的规则处理

[a-zA-Z_][a-zA-Z0-9_]*          { yylval.symbol = id(); return ID; }

//对于符号的规则处理

"="                                                     { return '='; }

"+"                                                     { return '+'; }

"-"                                                     { return '-'; }

"*"                                                     { return '*'; }

"/"                                                     { return '/'; }

"("                                                     { return '('; }

")"                                                     { return ')'; }

// 对于特殊符号的处理

[ /t]                                           { /* do nothing */ }

/n                                                      { return '/n'; }

.                                                       { printf("invalid character '0x%02x'/n", (unsigned int)yytext[0]); }

%%

/

// 以下都是对成员函数的实现,将被复制到.cpp文件中

int calc_lexer::create(calc_parser* parser, symboltable* symboltable)

{

        assert(parser != NULL);

        assert(symboltable != NULL);

       

        m_symboltable = symboltable;

        return yycreate(parser);

}

/

// calc_lexer attribute commands

double calc_lexer::number() const

{

        errno = 0;              // clear error flag

        char* endp;

        double d = strtod(yytext, &endp);

        if ((d == +HUGE_VAL || d == -HUGE_VAL) && errno == ERANGE) {

                printf("number too big/n");

        }

        return d;

}

symbol* calc_lexer::id() const

{

        symbol* p = m_symboltable->install(yytext);

        return p;

}//lexer.l end

parser.y文件

%{                      //%{ %}中的c代码将被复制到.cpp文件中

#include <math.h>

%}

// include file

%include {              //%include {}里的代码复制到对应的.h文件中,这里声明symbol

 // 是为在下面calc_parser:: assign函数能够找到使用类型的提前

 //声明

// forward references

class symbol;

}

//%union表示使用联合体声明yytest的类型可能有2,也就是lex返回的标记的值的类型(//一个标记都会有一个值)

%union {

        symbol* symbol;

        double value;

}

// nonterminals

%type <value> expr  //%type是声明变量exprdouble类型,这里要注意的是<>声明的类型必//须是在%union中定义的变量

//这里要注意下越是后声明的越高

%right '='          //%right声明=为右结合

%left '+', '-'  //%left 声明+ -为左结合

%left '*', '/'        //%left 声明* /为左结合

%right UMINUS     //%right声明UMINUS为右结合,此优先级最高

%token <value> NUMBER //%token声明标记NUMBER和其类型,%type注意点一样

%token <symbol> ID     //声明标记ID的类型,%type注意点一样

// keywords

%token SIN           //声明标记SIN,因为本身的值没有用途所以不对其类型特殊声明

%token COS           //声明标记COS,因为本身的值没有用途所以不对其类型特殊声明

%token TAN           //声明标记TAN,因为本身的值没有用途所以不对其类型特殊声明

// include file

%include {

#include "symbol.h"  //%{ %}中的c代码将被复制到.h文件中

#include "lexer.h"

}

// parser name

%name calc_parser   //%name指定继承自lexer的类名{}中为类成员声明,类的声明将被解

//释到.h文件中

// class definition

{

// 类成员声明

protected:

        symboltable m_symboltable;              // the symbol table

        calc_lexer m_lexer;                             // the lexical analyser

       

// Operations

public:

        int create();

       

        // attribute commands

        double assign(symbol* id, double value);

        double divide(double dividend, double divisor);

}

// constructor

{

        // do nothing

}

%%

lines

        : lines line

        | /* empty */

        ;

line

        : expr '/n'                                     { printf("%g/n", (double)$1); }

        | error '/n'                            { yyerrok(); }

        ;

//这里只对expr进行说明$$是冒号左边表达式的值   $1为右边第一个表达式的值

$2为右边第二个表达式的值  $3为右边第三个表达式的值,以此类推.

expr

        : ID '=' expr                           { $$ = assign($1, $3); }//变量保存

        | expr '+' expr                         { $$ = $1 + $3; }

        | expr '-' expr                         { $$ = $1 - $3; }

        | expr '*' expr                         { $$ = $1 * $3; }

        | expr '/' expr                         { $$ = divide($1, $3); }//除法判断

        | '(' expr ')'                          { $$ = $2; }

        | '-' expr %prec UMINUS         { $$ = -$2; }

// %prec说明'-' expr表达式的优先级和UMINUS一样.

        | NUMBER                                        { $$ = $1; }

        | ID                                            { $$ = $1->m_value; }

        | SIN '(' expr ')'                      { $$ = sin($3); }

        | COS '(' expr ')'                      { $$ = cos($3); }

        | TAN '(' expr ')'                      { $$ = tan($3); }

        ;

%%

/

//以下都是对成员函数的实现,将被复制到.cpp文件中

int main(void)

{

        int n = YYEXIT_FAILURE;

       

        calc_parser parser;

        if (parser.create()) {

                n = parser.yyparse();

        }

        return n;

}

/

// calc_parser commands

int calc_parser::create()

{

        if (!yycreate(&m_lexer)) {

                return 0;

        }

        if (!m_lexer.create(this, &m_symboltable)) {

                return 0;

        }

        return 1;       // success

}

/

// calc_parser attribute commands

double calc_parser::assign(symbol* id, double value)

{

        assert(id != NULL);

        id->m_value = value;

        return id->m_value;

}

double calc_parser::divide(double a, double b)

{

        if (b == 0) {

                printf("division by zero/n");

                yythrowerror();         // causes a syntax error

                return 0;

        }

        else {

                return a / b;

        }

}

更多详细参考可以查看Parser Generator Help帮助文档

以上都是作者的辛苦劳动,转载请注明出处.

更多 0
相关主题推荐
visual studio 多线程 visual c++ 对话框 character
相关博文推荐
【Java核心技术——多线程】
操作系统(3)进程及其实现
CString头文件
mfc深入浅出 系列 深入浅出mfc 第...
Java 线程同步异步学习
黑马程序员01_多线程
win7中VC6.0 visual C+...
关于JAVA多线程Runnable和Th...
id="google_ads_frame1" height="90" marginheight="0" src="http://googleads.g.doubleclick.net/pagead/ads?client=ca-pub-1076724771190722&output=html&h=90&slotname=4497271841&adk=1868520242&w=728&lmt=1393734404&ea=0&flash=12.0.0.70&url=http%3A%2F%2Fblog.csdn.net%2Fxujianlane%2Farticle%2Fdetails%2F2801204&dt=1393734404343&shv=r20140220&cbv=r20140226&saldr=sb&correlator=1393734404358&frm=20&ga_vid=1724069759.1393734404&ga_sid=1393734404&ga_hid=442748419&ga_fc=0&u_tz=480&u_his=1&u_java=1&u_h=670&u_w=1192&u_ah=670&u_aw=1138&u_cd=32&u_nplug=0&u_nmime=0&dff=arial&dfs=12&adx=298&ady=16062&biw=1120&bih=490&eid=317150313&oid=3&ref=http%3A%2F%2Fblog.csdn.net%2Fxujianlane%2Farticle%2Fdetails%2F2801204&rx=0&eae=4&docm=7&brdim=54%2C143%2C%2C%2C1138%2C%2C%2C%2C%2C&vis=0&fu=0&ifi=1&dtd=93" frameborder="0" width="728" allowtransparency="" name="google_ads_frame1" marginwidth="0" scrolling="no">
查看评论
7楼 fengjuyixuan 2012-06-10 18:11发表 [回复]
不错 刚好正在学习
6楼 hufeikong 2011-06-06 00:01发表 [回复]
文章不错,就是对照不能成功!
关于&quot;YYLEXERNAME&quot;这个错误相信所有按照这步骤操作的人都遇到了.
5楼 chouchou2007 2010-03-09 13:59发表 [回复]
你好,请教下如果我想在64位Windows7下使用该词法分析工具,应该怎么配置,能在64位环境下使用吗
4楼 everyhook 2010-01-07 19:43发表 [回复]
但貌似你已经不在了,你的最后文章是在08年。
3楼 everyhook 2010-01-07 19:42发表 [回复]
你好,可以回答一下吗???
太可惜了,你是两年前写的文章。
我的编译器不能编译成功。
单独编译 *.l 或 *.y都可以 ,但是合起来就不可以。
很希望你还在csdn上。
2楼 霸天Dana 2009-08-17 09:58发表 [回复]
呵呵,我自己已经明白了.多谢
1楼 霸天Dana 2009-08-17 09:55发表 [回复]
之后在vc6.0加入如下设置

Tool-&gt;Option-&gt; directory

Bin file :???

你好,请问为什么我找不到这个项目栏呢?
我本地显示的是
Tools-&gt;Options-&gt; directories,其中显示了以下三个属性
Platform;Show directories for;directories
发表评论
  • 用 户 名:
  • xizmi
  • 评论内容:
  • 插入代码
  •   
* 以上用户言论只代表其个人观点,不代表CSDN网站的观点或立场
TOP
核心技术类目
全部主题 Java VPN Android iOS ERP IE10 Eclipse CRM JavaScript Ubuntu NFC WAP jQuery 数据库 BI HTML5 Spring Apache Hadoop .NET API HTML SDK IIS Fedora XML LBS Unity Splashtop UML components Windows Mobile Rails QEMU KDE Cassandra CloudStack FTC coremail OPhone CouchBase 云计算 iOS6 Rackspace Web App SpringSide Maemo Compuware 大数据 aptech Perl Tornado Ruby Hibernate ThinkPHP Spark HBase Pure Solr Angular Cloud Foundry Redis Scala Django Bootstrap
  • 个人资料

  • xujianlane
    • 访问:28104次
    • 积分:416分
    • 排名:千里之外
    • 原创:11篇
    • 转载:3篇
    • 译文:1篇
    • 评论:13条
  • 文章存档
    • 2008年08月(1)
    • 2008年01月(2)
    • 2007年12月(1)
    • 2007年11月(4)
    • 2007年07月(6)
    • 2007年05月(1)
      展开
  • 阅读排行
  • vc中使用nmake命令编译方法(8806)
  • Parser Generator使用说明(5912)
  • 初识IPv6(四)(2586)
  • ControlSet001、ControlSet002以及CurrentControlSet(1523)
  • 如何判断当前系统是ipv4还是ipv6(1250)
  • IBM-T60 冻死问题解决(1179)
  • 初识IPv6(二)(943)
  • 初识IPv6(三)(886)
  • xp,windows2000/2003/2008注册表里ipv6地址的位置 (753)
  • RFC2428--FTP对IPv6和NAT的扩展(615)
  • 评论排行
  • Parser Generator使用说明(7)
  • ControlSet001、ControlSet002以及CurrentControlSet(3)
  • 初识IPv6(二)(1)
  • vc中使用nmake命令编译方法(1)
  • 如何判断当前系统是ipv4还是ipv6(1)
  • 初识IPv6(一)(0)
  • IBM-T60 冻死问题解决(0)
  • RFC2428--FTP对IPv6和NAT的扩展(0)
  • unicode,ansi编码转换(0)
  • windows常用api(0)
  • 推荐文章
    • 最新评论
    • vc中使用nmake命令编译方法

      xujiali5172923: 第二条很重要啊

    • Parser Generator使用说明

      fengjuyixuan: 不错 刚好正在学习

    • 如何判断当前系统是ipv4还是ipv6

      hwwill: 怎么判断,我电脑上网卡绑定的是DNE

    • Parser Generator使用说明

      hufeikong: 文章不错,就是对照不能成功!关于&quot;YYLEXERNAME&quot;这个错误相信所有按照这...

    • Parser Generator使用说明

      chouchou2007: 你好,请教下如果我想在64位Windows7下使用该词法分析工具,应该怎么配置,能在64位环境下使用...

    • Parser Generator使用说明

      everyhook: 但貌似你已经不在了,你的最后文章是在08年。

    • Parser Generator使用说明

      everyhook: 你好,可以回答一下吗???太可惜了,你是两年前写的文章。我的编译器不能编译成功。单独编译 *.l 或...

    • 初识IPv6(二)

      insulted: 感谢博主这篇文章!我想问一个问题:// IPv6 fragment header typedef s...

    • Parser Generator使用说明

      霸天Dana: 呵呵,我自己已经明白了.多谢

    • Parser Generator使用说明

      霸天Dana: 之后在vc6.0加入如下设置Tool-&gt;Option-&gt; directoryBin fi...

    公司简介| 招贤纳士| 广告服务| 银行汇款帐号| 联系方式| 版权声明| 法律顾问| 问题报告
    QQ客服 微博客服 论坛反馈 联系邮箱:webmaster@csdn.net 服务热线:400-600-2320
    京 ICP 证 070598 号
    北京创新乐知信息技术有限公司 版权所有
    江苏乐知网络技术有限公司 提供商务支持
    Copyright © 1999-2014, CSDN.NET, All Rights Reserved  GongshangLogo

    这篇关于ser Generator使用说明的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

    相关文章

    Conda与Python venv虚拟环境的区别与使用方法详解

    《Conda与Pythonvenv虚拟环境的区别与使用方法详解》随着Python社区的成长,虚拟环境的概念和技术也在不断发展,:本文主要介绍Conda与Pythonvenv虚拟环境的区别与使用... 目录前言一、Conda 与 python venv 的核心区别1. Conda 的特点2. Python v

    Spring Boot中WebSocket常用使用方法详解

    《SpringBoot中WebSocket常用使用方法详解》本文从WebSocket的基础概念出发,详细介绍了SpringBoot集成WebSocket的步骤,并重点讲解了常用的使用方法,包括简单消... 目录一、WebSocket基础概念1.1 什么是WebSocket1.2 WebSocket与HTTP

    C#中Guid类使用小结

    《C#中Guid类使用小结》本文主要介绍了C#中Guid类用于生成和操作128位的唯一标识符,用于数据库主键及分布式系统,支持通过NewGuid、Parse等方法生成,感兴趣的可以了解一下... 目录前言一、什么是 Guid二、生成 Guid1. 使用 Guid.NewGuid() 方法2. 从字符串创建

    Python使用python-can实现合并BLF文件

    《Python使用python-can实现合并BLF文件》python-can库是Python生态中专注于CAN总线通信与数据处理的强大工具,本文将使用python-can为BLF文件合并提供高效灵活... 目录一、python-can 库:CAN 数据处理的利器二、BLF 文件合并核心代码解析1. 基础合

    Python使用OpenCV实现获取视频时长的小工具

    《Python使用OpenCV实现获取视频时长的小工具》在处理视频数据时,获取视频的时长是一项常见且基础的需求,本文将详细介绍如何使用Python和OpenCV获取视频时长,并对每一行代码进行深入解析... 目录一、代码实现二、代码解析1. 导入 OpenCV 库2. 定义获取视频时长的函数3. 打开视频文

    Go语言中make和new的区别及说明

    《Go语言中make和new的区别及说明》:本文主要介绍Go语言中make和new的区别及说明,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1 概述2 new 函数2.1 功能2.2 语法2.3 初始化案例3 make 函数3.1 功能3.2 语法3.3 初始化

    Spring IoC 容器的使用详解(最新整理)

    《SpringIoC容器的使用详解(最新整理)》文章介绍了Spring框架中的应用分层思想与IoC容器原理,通过分层解耦业务逻辑、数据访问等模块,IoC容器利用@Component注解管理Bean... 目录1. 应用分层2. IoC 的介绍3. IoC 容器的使用3.1. bean 的存储3.2. 方法注

    Python内置函数之classmethod函数使用详解

    《Python内置函数之classmethod函数使用详解》:本文主要介绍Python内置函数之classmethod函数使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录1. 类方法定义与基本语法2. 类方法 vs 实例方法 vs 静态方法3. 核心特性与用法(1编程客

    Linux中压缩、网络传输与系统监控工具的使用完整指南

    《Linux中压缩、网络传输与系统监控工具的使用完整指南》在Linux系统管理中,压缩与传输工具是数据备份和远程协作的桥梁,而系统监控工具则是保障服务器稳定运行的眼睛,下面小编就来和大家详细介绍一下它... 目录引言一、压缩与解压:数据存储与传输的优化核心1. zip/unzip:通用压缩格式的便捷操作2.

    java中新生代和老生代的关系说明

    《java中新生代和老生代的关系说明》:本文主要介绍java中新生代和老生代的关系说明,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、内存区域划分新生代老年代二、对象生命周期与晋升流程三、新生代与老年代的协作机制1. 跨代引用处理2. 动态年龄判定3. 空间分