多种定义

Multiple definitions?

本文关键字:定义      更新时间:2023-10-16

我在编译flex和bison代码时遇到了问题。更具体地说是我的parser.yy文件。在这个文件中,我包含了MathCalc.h和BaseProg.h,它们都是我创建的类。问题是,当我实例化类时,它会在编译时给我一个"多重定义"错误。任何帮助都将不胜感激!非常感谢。

Parser.yy(代码段):

%code requires {
#include <iostream>
#include <cmath>
#include "MathCalc.h"
#include "BaseProg.h"
/* Parser error reporting routine */
void yyerror(const char *msg);
/* Scannar routine defined by Flex */
int yylex();
using namespace std;
BaseProg bprog;
MathCalc calc;
enum Type { INT, FLT};
}
/* yylval union type */
%union {
        double dval;
        int ival;
        char* name;
        Type type;
}

错误:

bison -d parser.yy
g++    -c -o scanner.o scanner.cc
g++    -c -o parser.tab.o parser.tab.cc
g++ scanner.o parser.tab.o BaseProg.o MathCalc.o -lfl -o ../Proj2
parser.tab.o:(.bss+0x0): multiple definition of `bprog'
scanner.o:(.bss+0x28): first defined here
parser.tab.o:(.bss+0x1): multiple definition of `calc'
scanner.o:(.bss+0x29): first defined here
collect2: ld returned 1 exit status

%code requires块中的任何代码都将被放置在解析器源文件和解析器头文件中。#include是扫描器源中的解析器头文件,这是正常的(毕竟,这就是bison生成头文件的原因),因此将全局变量定义放在%code requires块中是不明智的。

事实上,将全局变量定义放在头文件中总是不明智的,正是因为头文件可能包含在多个源文件中,因此任何全局定义(与声明相反)都将插入多个翻译单元,从而违反ODR。

对于头文件,应该将这些对象(BaseProg bprog;MathCalc calc;)标记为extern,然后确保在某个源文件中实际定义了它们。或者,更好的是,您应该首先避免使用全局变量。