寫這一章的目的就是為了能夠提供一個能夠使用C++特色的Lex和Yacc框架,這個框架 同前一章的目的一樣,也是僅僅為了能夠提供一個什么也不作的框架程序,當時有點 不同的就是:這個新的框架使用了C++語法,能夠使用所有的C++特色,包括STL的強 勁算法和容器,還有各式各樣的C++庫。采用C++的目的就是為了能夠快速的編碼來完 成自己需要完成的任務。
在編碼詞法分析和語法分析程序的時候經常會編寫一些鏈表和容器,實際上這些鏈表和容 器在C++語言函數庫里面都已經有了,就需要好好的利用這些C++的特色,因此有必要認真 考慮一下這些生成的詞法分析程序和語法分析程序如何和C++聯系起來了。本章就是給出 了一個能夠使用C++語法要素的程序框架。希望對那些使用C++又希望使用C語法的Lex和Yacc 程序的人有所幫助。
1. lex文件
例 4.1. frame.l
%{
extern "C"{// 如果是采用C++語言環境,就必須設置C鏈接類型
int yywrap(void);
int yylex(void);// 這個是lex產生的詞法分析函數,必須在這里進行聲明
}
%}
%%
%%
int yywrap(void)
{
// 返回1表示讀取全部結束,返回0表示還有輸入需要讀取,這一點
// 會在后面的文檔的讀取多個輸入文件的時候進行討論
return 1;
}
2. yacc文件
例 4.2. frame.y
%{
#include <iostream>// 這里引用了C++語言特有的流庫
extern "C"{// 如果是采用C++語言環境,就必須設置C鏈接類型
void yyerror(const char *s);
extern int yylex(void);// 為了能夠在語法文件里面找到詞法分析函數,必須聲明
}
%}
%%
program:// 仍然是一個什么也不干的程序
;
%%
void yyerror(const char *s)
{// 通常的語法錯誤就是直接打印錯誤信息
std::cerr<< s << std::endl;// 在這里使用了C++標準錯誤流
}
int main()
{
// 直接調用yacc生成的語法分析函數從標準輸入讀取
// 向標準輸出寫入
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. 結論
為了能夠在C++程序里面使用C函數,就必須把所有的C函數用extern "C"{}包括起來 ,而且必須把每一個需要使用的C函數都包括在extern "C"塊里面。因此,上面的框 架程序里面的yylex()函數就必須聲明在 frame.l
的頭部,而在 frame.y
里面為 了引用frame.l里面的yylex()函數,則必須在extern "C"塊里面在聲明一次成為 extern的,這樣就可以使用外部C鏈接函數yylex()了。因此所有的C語法元素都必須 用extern "C"進行描述。