问题:
在一个源文件里定义了一个数组:char a[6];
在另外一个文件里用下列语句进行了声明:extern char *a;
请问,这样可以吗?
我当时以为*a只是声明了一个a变量,并不分配内存,其等价于与数组首地址的a,所以答的是可以。没想到出来后网上查了才知道是不可以的。需要类型完全匹配才行。
网上答案:
不可以,程序运行时会告诉你非法访问。原因在于,指向类型T的指针并不等价于类型T的数组。extern char *a声明的是一个指针变量而不是字符数组,因此与实际的定义不同,从而造成运行时非法访问。应该将声明改为extern char a[ ]。
继而继续搜寻关于extern的介绍,其用法总结如下:
1. extern 是告诉编译器,他修饰的变量已经在另外的源文件编译生成的obj中定义了。比如 extern int a。
由于每个程序都会经历每个源文件编译成obj然后再链接起来的过程,所以在不可能在多个源文件中同时定义多个全局的同名的变量。
2. 在h文件中尽量不使用全局变量定义。因为头文件要被其他源文件所引用,每个源文件在扩展头文件时都会再次定义这个变量,这样,又会导致重复定义。
比如一个头文件:
#ifndef __H
#define __H
int a = 6;
#endif;
此时某一个源文件include这个头文件时是没有问题的,但是如果多个源文件都在引用这个头文件时,就会报如下错误:
two.obj : error LNK2005: "char * a" (?a@@3PADA) already defined in main.obj
Debug/onetest.exe : fatal error LNK1169: one or more multiply defined symbols found
Error executing link.exe.
原因在于链接时发现第二个源文件又重复定义了a。所以可以看出头文件的防止重复引用只是在本源文件中才起作用。
所以基于以上两点,可以看出,定义尽量放在源文件中,需要用到其他源文件中的全局变量,可以使用extern。