原创 头文件重复引用

2010-2-2 19:07 2309 6 6 分类: MCU/ 嵌入式

#include文件的一个不利之处在于一个头文件可能会被多次包含,为了说明这种错误,考虑下面的代码:<?xml:namespace prefix = o ns = "urn:schemas-microsoft-com:office:office" />


#include "x.h"


#include "x.h"


 


显然,这里文件x.h被包含了两次,没有人会故意编写这样的代码。但是下面的代码:


#include "a.h"


#include "b.h"


 


看上去没什么问题。如果a.hb.h都包含了一个头文件x.h。那么x.h在此也同样被包含了两次,只不过它的形式不是那么明显而已。


 


多重包含在绝大多数情况下出现在大型程序中,它往往需要使用很多头文件,因此要发现重复包含并不容易。要解决这个问题,我们可以使用条件编译。如果所有的头文件都像下面这样编写:


#ifndef _HEADERNAME_H


#define _HEADERNAME_H


...


#endif


 


那么多重包含的危险就被消除了。当头文件第一次被包含时,它被正常处理,符号_HEADERNAME_H被定义为1。如果头文件被再次包含,通过条件编译,它的内容被忽略。符号_HEADERNAME_H按照被包含头文件的文件名进行取名,以避免由于其他头文件使用相同的符号而引起的冲突。


 


但是,你必须记住预处理器仍将整个头文件读入,即使这个头文件所有内容将被忽略。由于这种处理将托慢编译速度,所以如果可能,应该避免出现多重包含。


 


~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~我是分隔符~~~~~~~~~~~~~~~~~~~~~~~~~


 


使用#ifndef只是防止了头文件被重复包含(其实本例中只有一个头件,不会存在重复包含的问题),但是无法防止变量被重复定义。


 


# vi test.c


-------------------------------


#include <stdio.h>


#include "test.h"


 


extern i;


extern void test1();


extern void test2();


 


int main()


{


   test1();


   printf("ok\n");


   test2();


   printf("%d\n",i);


   return 0;


}


 


 


# vi test.h


-------------------------------


#ifndef _TEST_H_


#define _TEST_H_


 


char add1[] = "char1";


char add2[] = "char2";


int i = 10;


void test1();


void test2();


 


#endif


 


 


# vi test1.c


-------------------------------


#include <stdio.h>


#include "test.h"


 


extern char add1[];


 


void test1()


{


   printf(add1);


}


 


 


# vi test2.c


-------------------------------


#include <stdio.h>


#include "test.h"


 


extern char add2[];


extern i;


 


void test2()


{


   printf(add2);


   for (; i > 0; i--)


       printf("%d-", i);


}


 


# Makefile


-------------------------------


test:    test.o test1.o test2.o


test1.o: test1.c


test2.o: test2.c


clean:


   rm test test.o test1.o test2.o


 


错误:


test-1.0编译后会出现"multiple definition of"错误。


 


错误分析:


由于工程中的每个.c文件都是独立的解释的,即使头文件有


#ifndef _TEST_H_


#define _TEST_H_


....


#enfif


在其他文件中只要包含了global.h就会独立的解释,然后每个.c文件生成独立的标示符。在编译器链接时,就会将工程中所有的符号整合在一起,由于文件中有重名变量,于是就出现了重复定义的错误。


 


解决方法


.c文件中声明变量,然后建一个头文件(.h文件)在所有的变量声明前加上extern,注意这里不要对变量进行的初始化。然后在其他需要使用全局变量的.c文件中包含.h文件。编译器会为.c生成目标文件,然后链接时,如果该.c文件使用了全局变量,链接器就会链接到此.c文件


 


test-2.0


 


# vi test.c


-------------------------------


#include <stdio.h>


#include "test.h"


 


int i = 10;


char add1[] = "char1";


char add2[] = "char2";


extern void test1();


extern void test2();


 


int main()


{


   test1();


   printf("ok\n");


   test2();


   printf("%d\n",i);


   return 0;


}


 


 


# vi test.h


-------------------------------


#ifndef _TEST_H_


#define _TEST_H_


 


extern i;


extern char add1[];


extern char add2[];


 


void test1();


void test2();


 


#endif


 


 


 


# vi test1.c


-------------------------------


#include <stdio.h>


#include "test.h"


 


void test1()


{


   printf(add1);


}


 


 


# vi test2.c


-------------------------------


#include <stdio.h>


#include "test.h"


 


void test2()


{


   printf(add2);


   for (; i > 0; i--)


       printf("%d-", i);


}


 


本文来自CSDN博客,转载请标明出处:


http://blog.csdn.net/beiguoyouzi/archive/<?xml:namespace prefix = st1 ns = "urn:schemas-microsoft-com:office:smarttags" />2009/05/20/4204445.aspx

PARTNER CONTENT

文章评论0条评论)

登录后参与讨论
EE直播间
更多
我要评论
0
6
关闭 站长推荐上一条 /3 下一条