• 头文件重复引用


    #include文件的一个不利之处在于一个头文件可能会被多次包含,为了说明这种错误,考虑下面的代码:
    #include "x.h"
    #include "x.h"

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

    看上去没什么问题。如果a.h和b.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文件 。

  • 相关阅读:
    罗美琪和春波特的故事...
    欢迎参与 KubeVela 官方文档翻译活动
    开源 1 年半 star 破 1.2 万的 Dapr 是如何在阿里落地的?
    Service Mesh 从“趋势”走向“无聊”
    Nacos 2.0 性能提升十倍,贡献者 80% 以上来自阿里之外
    阿里巴巴云原生 etcd 服务集群管控优化实践
    KubeVela 1.0 :开启可编程式应用平台的未来
    知行动手实验室可以用来做什么?
    7. Jackson用树模型处理JSON是必备技能,不信你看
    Linux:sudo,没有找到有效的 sudoers 资源
  • 原文地址:https://www.cnblogs.com/shenlanzifa/p/5288784.html
Copyright © 2020-2023  润新知