寫(xiě)這一章的目的就是為了能夠提供一個(gè)能夠使用C++特色的Lex和Yacc框架,這個(gè)框架 同前一章的目的一樣,也是僅僅為了能夠提供一個(gè)什么也不作的框架程序,當(dāng)時(shí)有點(diǎn) 不同的就是:這個(gè)新的框架使用了C++語(yǔ)法,能夠使用所有的C++特色,包括STL的強(qiáng) 勁算法和容器,還有各式各樣的C++庫(kù)。采用C++的目的就是為了能夠快速的編碼來(lái)完 成自己需要完成的任務(wù)。
在編碼詞法分析和語(yǔ)法分析程序的時(shí)候經(jīng)常會(huì)編寫(xiě)一些鏈表和容器,實(shí)際上這些鏈表和容 器在C++語(yǔ)言函數(shù)庫(kù)里面都已經(jīng)有了,就需要好好的利用這些C++的特色,因此有必要認(rèn)真 考慮一下這些生成的詞法分析程序和語(yǔ)法分析程序如何和C++聯(lián)系起來(lái)了。本章就是給出 了一個(gè)能夠使用C++語(yǔ)法要素的程序框架。希望對(duì)那些使用C++又希望使用C語(yǔ)法的Lex和Yacc 程序的人有所幫助。
1. lex文件
例 4.1. frame.l
%{
extern "C"{// 如果是采用C++語(yǔ)言環(huán)境,就必須設(shè)置C鏈接類型
int yywrap(void);
int yylex(void);// 這個(gè)是lex產(chǎn)生的詞法分析函數(shù),必須在這里進(jìn)行聲明
}
%}
%%
%%
int yywrap(void)
{
// 返回1表示讀取全部結(jié)束,返回0表示還有輸入需要讀取,這一點(diǎn)
// 會(huì)在后面的文檔的讀取多個(gè)輸入文件的時(shí)候進(jìn)行討論
return 1;
}
2. yacc文件
例 4.2. frame.y
%{
#include <iostream>// 這里引用了C++語(yǔ)言特有的流庫(kù)
extern "C"{// 如果是采用C++語(yǔ)言環(huán)境,就必須設(shè)置C鏈接類型
void yyerror(const char *s);
extern int yylex(void);// 為了能夠在語(yǔ)法文件里面找到詞法分析函數(shù),必須聲明
}
%}
%%
program:// 仍然是一個(gè)什么也不干的程序
;
%%
void yyerror(const char *s)
{// 通常的語(yǔ)法錯(cuò)誤就是直接打印錯(cuò)誤信息
std::cerr<< s << std::endl;// 在這里使用了C++標(biāo)準(zhǔn)錯(cuò)誤流
}
int main()
{
// 直接調(diào)用yacc生成的語(yǔ)法分析函數(shù)從標(biāo)準(zhǔn)輸入讀取
// 向標(biāo)準(zhǔn)輸出寫(xiě)入
yyparse();
return 0;
}
3. Makefile文件
例 4.3. Makefile
LEX=flex
YACC=bison
CC=g++
a.exe:lex.yy.o frame.tab.o
$(CC) lex.yy.o frame.tab.o -o a.exe
lex.yy.o:lex.yy.c frame.tab.h
$(CC) -c lex.yy.c
frame.tab.o:frame.tab.c
$(CC) -c frame.tab.c
frame.tab.c frame.tab.h:frame.y
$(YACC) -d frame.y
lex.yy.c:frame.l
$(LEX) frame.l
clean:
rm -f *.o *.c *.h
4. 結(jié)論
為了能夠在C++程序里面使用C函數(shù),就必須把所有的C函數(shù)用extern "C"{}包括起來(lái) ,而且必須把每一個(gè)需要使用的C函數(shù)都包括在extern "C"塊里面。因此,上面的框 架程序里面的yylex()函數(shù)就必須聲明在 frame.l
的頭部,而在 frame.y
里面為 了引用frame.l里面的yylex()函數(shù),則必須在extern "C"塊里面在聲明一次成為 extern的,這樣就可以使用外部C鏈接函數(shù)yylex()了。因此所有的C語(yǔ)法元素都必須 用extern "C"進(jìn)行描述。