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

    相关文章

    C++变换迭代器使用方法小结

    《C++变换迭代器使用方法小结》本文主要介绍了C++变换迭代器使用方法小结,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录1、源码2、代码解析代码解析:transform_iterator1. transform_iterat

    C++中std::distance使用方法示例

    《C++中std::distance使用方法示例》std::distance是C++标准库中的一个函数,用于计算两个迭代器之间的距离,本文主要介绍了C++中std::distance使用方法示例,具... 目录语法使用方式解释示例输出:其他说明:总结std::distance&n编程bsp;是 C++ 标准

    vue使用docxtemplater导出word

    《vue使用docxtemplater导出word》docxtemplater是一种邮件合并工具,以编程方式使用并处理条件、循环,并且可以扩展以插入任何内容,下面我们来看看如何使用docxtempl... 目录docxtemplatervue使用docxtemplater导出word安装常用语法 封装导出方

    Linux换行符的使用方法详解

    《Linux换行符的使用方法详解》本文介绍了Linux中常用的换行符LF及其在文件中的表示,展示了如何使用sed命令替换换行符,并列举了与换行符处理相关的Linux命令,通过代码讲解的非常详细,需要的... 目录简介检测文件中的换行符使用 cat -A 查看换行符使用 od -c 检查字符换行符格式转换将

    使用Jackson进行JSON生成与解析的新手指南

    《使用Jackson进行JSON生成与解析的新手指南》这篇文章主要为大家详细介绍了如何使用Jackson进行JSON生成与解析处理,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录1. 核心依赖2. 基础用法2.1 对象转 jsON(序列化)2.2 JSON 转对象(反序列化)3.

    使用Python实现快速搭建本地HTTP服务器

    《使用Python实现快速搭建本地HTTP服务器》:本文主要介绍如何使用Python快速搭建本地HTTP服务器,轻松实现一键HTTP文件共享,同时结合二维码技术,让访问更简单,感兴趣的小伙伴可以了... 目录1. 概述2. 快速搭建 HTTP 文件共享服务2.1 核心思路2.2 代码实现2.3 代码解读3.

    Elasticsearch 在 Java 中的使用教程

    《Elasticsearch在Java中的使用教程》Elasticsearch是一个分布式搜索和分析引擎,基于ApacheLucene构建,能够实现实时数据的存储、搜索、和分析,它广泛应用于全文... 目录1. Elasticsearch 简介2. 环境准备2.1 安装 Elasticsearch2.2 J

    使用C#代码在PDF文档中添加、删除和替换图片

    《使用C#代码在PDF文档中添加、删除和替换图片》在当今数字化文档处理场景中,动态操作PDF文档中的图像已成为企业级应用开发的核心需求之一,本文将介绍如何在.NET平台使用C#代码在PDF文档中添加、... 目录引言用C#添加图片到PDF文档用C#删除PDF文档中的图片用C#替换PDF文档中的图片引言在当

    Java中List的contains()方法的使用小结

    《Java中List的contains()方法的使用小结》List的contains()方法用于检查列表中是否包含指定的元素,借助equals()方法进行判断,下面就来介绍Java中List的c... 目录详细展开1. 方法签名2. 工作原理3. 使用示例4. 注意事项总结结论:List 的 contain

    C#使用SQLite进行大数据量高效处理的代码示例

    《C#使用SQLite进行大数据量高效处理的代码示例》在软件开发中,高效处理大数据量是一个常见且具有挑战性的任务,SQLite因其零配置、嵌入式、跨平台的特性,成为许多开发者的首选数据库,本文将深入探... 目录前言准备工作数据实体核心技术批量插入:从乌龟到猎豹的蜕变分页查询:加载百万数据异步处理:拒绝界面