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

    相关文章

    RabbitMQ 延时队列插件安装与使用示例详解(基于 Delayed Message Plugin)

    《RabbitMQ延时队列插件安装与使用示例详解(基于DelayedMessagePlugin)》本文详解RabbitMQ通过安装rabbitmq_delayed_message_exchan... 目录 一、什么是 RabbitMQ 延时队列? 二、安装前准备✅ RabbitMQ 环境要求 三、安装延时队

    Python ORM神器之SQLAlchemy基本使用完全指南

    《PythonORM神器之SQLAlchemy基本使用完全指南》SQLAlchemy是Python主流ORM框架,通过对象化方式简化数据库操作,支持多数据库,提供引擎、会话、模型等核心组件,实现事务... 目录一、什么是SQLAlchemy?二、安装SQLAlchemy三、核心概念1. Engine(引擎)

    Java Stream 并行流简介、使用与注意事项小结

    《JavaStream并行流简介、使用与注意事项小结》Java8并行流基于StreamAPI,利用多核CPU提升计算密集型任务效率,但需注意线程安全、顺序不确定及线程池管理,可通过自定义线程池与C... 目录1. 并行流简介​特点:​2. 并行流的简单使用​示例:并行流的基本使用​3. 配合自定义线程池​示

    GO语言中函数命名返回值的使用

    《GO语言中函数命名返回值的使用》在Go语言中,函数可以为其返回值指定名称,这被称为命名返回值或命名返回参数,这种特性可以使代码更清晰,特别是在返回多个值时,感兴趣的可以了解一下... 目录基本语法函数命名返回特点代码示例命名特点基本语法func functionName(parameters) (nam

    使用shardingsphere实现mysql数据库分片方式

    《使用shardingsphere实现mysql数据库分片方式》本文介绍如何使用ShardingSphere-JDBC在SpringBoot中实现MySQL水平分库,涵盖分片策略、路由算法及零侵入配置... 目录一、ShardingSphere 简介1.1 对比1.2 核心概念1.3 Sharding-Sp

    Java 正则表达式的使用实战案例

    《Java正则表达式的使用实战案例》本文详细介绍了Java正则表达式的使用方法,涵盖语法细节、核心类方法、高级特性及实战案例,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要... 目录一、正则表达式语法详解1. 基础字符匹配2. 字符类([]定义)3. 量词(控制匹配次数)4. 边

    Python Counter 函数使用案例

    《PythonCounter函数使用案例》Counter是collections模块中的一个类,专门用于对可迭代对象中的元素进行计数,接下来通过本文给大家介绍PythonCounter函数使用案例... 目录一、Counter函数概述二、基本使用案例(一)列表元素计数(二)字符串字符计数(三)元组计数三、C

    使用Spring Cache本地缓存示例代码

    《使用SpringCache本地缓存示例代码》缓存是提高应用程序性能的重要手段,通过将频繁访问的数据存储在内存中,可以减少数据库访问次数,从而加速数据读取,:本文主要介绍使用SpringCac... 目录一、Spring Cache简介核心特点:二、基础配置1. 添加依赖2. 启用缓存3. 缓存配置方案方案

    使用Python的requests库来发送HTTP请求的操作指南

    《使用Python的requests库来发送HTTP请求的操作指南》使用Python的requests库发送HTTP请求是非常简单和直观的,requests库提供了丰富的API,可以发送各种类型的HT... 目录前言1. 安装 requests 库2. 发送 GET 请求3. 发送 POST 请求4. 发送

    Nginx中配置使用非默认80端口进行服务的完整指南

    《Nginx中配置使用非默认80端口进行服务的完整指南》在实际生产环境中,我们经常需要将Nginx配置在其他端口上运行,本文将详细介绍如何在Nginx中配置使用非默认端口进行服务,希望对大家有所帮助... 目录一、为什么需要使用非默认端口二、配置Nginx使用非默认端口的基本方法2.1 修改listen指令