全局变量定义在头文件的相干试验
全局变量定义在头文件的相关试验
funcb.h
main.c
编译报错:
成功编译,运行结果为:
想在头文件中定义全局变量,测试程序如下:
a.h
int a = 3;
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_A int a = 3; #else extern 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
所以如果要定义公共全局变量最好使用这一种方法。