全局变量定义在头文件的相关试验

来源:互联网 发布:知乎加载不出图片 编辑:程序博客网 时间:2024/04/27 23:42

想在头文件中定义全局变量,测试程序如下:

a.h

int a = 3;


funca.h

void funca();
funca.c
#include <stdio.h>#include "a.h"void func_a(){printf("%p,%d\n",&a,a);}

funcb.h

void funcb();
funcb.c

#include <stdio.h>#include "a.h"void func_b(){printf("%p,%d\n",&a,a);}

main.c

#include <stdio.h>#include "a.h"void func_b(){printf("%p,%d\n",&a,a);}

编译报错:

gcc main.c funca.c funca.h funcb.c funcb.h a.h
/tmp/ccpq1WuB.o:(.data+0x0): multiple definition of `a'
/tmp/cccL6SXK.o:(.data+0x0): first defined here
collect2: ld returned 1 exit status

因为头文件的内容在预处理时完整地复制到被包含的文件中,所以在编译时不会出现问题,但是在链接时就会报错重定义。


看到网上有人说可以使用define来避免重定义,所以修改a.h:

#ifndef G_A#define G_Aint a = 3;#elseextern int a;#endif
编译时还是报错:

gcc main.c funca.c funca.h funcb.c funcb.h a.h
/tmp/ccqS3ryZ.o:(.data+0x0): multiple definition of `a'
/tmp/ccDDXBns.o:(.data+0x0): first defined here
collect2: ld returned 1 exit status


如果将a.h修改为:

int a;

成功编译,运行结果为:

0x601030,0
0x601030,0

int a;这样的形式(没有初始化)的全局变量是C语言中的一个特殊情况,叫做 weak symbol,链接的时候会把weak symbol合并成

一个,而不会报重复定义。

如果将a.h修改为:

static int a = 4;
成功编译,运行结果为:

0x601020,4
0x601024,4

但是结果不是我们想要的,static a被包含在不同的文件中,成为了该文件的static变量,而不再是同一个(地址不同)


如果将a.h修改为

extern int a;
增加a.c文件:

int a=4;
成功编译,运行结果为:

0x601020,4
0x601020,4

所以如果要定义公共全局变量最好使用这一种方法。

0 0
原创粉丝点击