建议直接看总结,如果有地方不懂在回头看细节

头文件怎么起作用

当一个test.cpp#include "Sal_Item.h"时,实际上编译器会把Sal_Item.h文件内的内容全部复制test.cpp

//Sale_Item.h  【未完全版】
#include<string>
struct Sale_Item{
std::string bookname;
double renenue;
int copies;
};
//test.cpp
#include<iostream>
#include<string>
#include"Sale_Item.h" //Sale_Item.h的内容会被复制到这里
using namespace std;
int main(){
Sale_Item data;
return 0;
}

避免头文件被重复引用

  • 因为头文件Sale_Item.h内容会被复制到引用它的文件test.cpp

  • 所以如果头文件Sale_Item.h被重复引用,就会导致在Sale_Item.h中定义的变量在test.cpp中被定义了多次,造成编译错误。

  • 因此,需要避免头文件被重复引用以及不同头文件中定义了同名的全局变量【当这些定义了同名变量的文件被#include到同一个程序时,同样会造成编译错误】

  • 必须对上面的Sale_Item.h进行改进,否则一旦被重复引用就会报错

    执行下面的代码

    #include<iostream>
    #include<string>
    #include"Sale_Item.h"
    #include"Sale_Item.h" //重复调用了Sale_Item.h,对于此时的头文件Sale_Item.h是不允许的
    using namespace std;
    int main(){
    Sale_Item data;
    std::string name = "xxx";
    data.bookname = name;
    return 0;
    }

    结果

避免头文件被重复引用的方法:条件编译

1. 给每个头文件添加一个预编译变量(preprocessor variable)作为标记(Label)

#define SALE_ITEM_H

  • 此时define的作用是定义预编译变量SALE_ITEM_H,而不是定义宏
  • 为什么预编译变量如此命名?
    • 头文件内容会被复制到test.cpp中,而且预编译变量无视作用域规则,也就是说整个test.cpp中不能出现与预编译变量SALE_ITEM_H同名的变量
    • 一般约定俗成把预编译变量写成头文件名的大写,这样既直观,又不容易和test.cpp和其他头文件的预编译变量产生冲突

2. 使用头文件保护符:ifdef/ifndef

ifdef XXX:如果预编译变量XXX已经被定义,则执行该指令与endif之间的代码块

ifndef XXX:如果预编译变量XXX还没有被定义,则执行该指令与endif之间的代码块

常规写法

//Sale_Item.h
#ifndef SAL_ITEM_H //如果SAL_ITEM_H未被定义
#define SAL_ITEM_H //定义SAL_ITEM_H预编译变量作为标记(Lable)。
//如果`test.cpp`在之前已经引用过该头文件,那么SAL_ITEM_H就已经被定义过
//在ifndef的作用下,之后的Sale_Item.h文件都不会执行
#include<string>
int i;
struct Sal_Item{
std::string bookname;
double renenue;
int copies;
};
#endif //结束符

3. 关于使用条件编译的必要性的探讨

  • 或许读者会有一个疑问:既然重复引用Sale_Item.h会导致test.cpp出错,那么不重复引用不就行了,何必那么麻烦写条件编译呢
  • 在上述例子中,Sale_Item.h不写条件编译确实可以,但在很多情况下,我们不得不重复引用同一个头文件
  • 还是以上述例子为例,对于头文件string
    • Sale_Item.h中,我们#include<string>来定义一个变量std::string bookname
    • test.cpp中,我们#include<string>来定义一个变量std::string name来给对象赋值
    • 所以test.cpp实际上就引用了头文件string两次,一次是显式地引用,一次是在#include"Sale_Item.h"时隐式地引用了string【Sale_Item.h中也includestring
  • 所以,无论是否有必要,我们建议在书写头文件时,都习惯性地使用条件编译

总结:创建自己的头文件

//Sale_Item.h
#ifndef SAL_ITEM_H //如果SAL_ITEM_H未被定义
#define SAL_ITEM_H //定义SAL_ITEM_H预编译变量作为标记(Lable)。
//如果`test.cpp`在之前已经引用过该头文件,那么SAL_ITEM_H就已经被定义过
//在ifndef的作用下,之后的Sale_Item.h文件都不会执行
#include<string>
int i;
struct Sal_Item{
std::string bookname;
double renenue;
int copies;
};
#endif //结束符
//test.cpp
#include<iostream>
#include<string>
#include"Sale_Item.h"
// #include"Sale_Item.h" //加上不要紧,因为进行了条件编译
using namespace std;
int main(){
Sale_Item data;
std::string name = "xxx";
data.bookname = name;
return 0;
}

2-6 C/C++ 编写头文件的更多相关文章

  1. 编写自己的C头文件

    1.       头文件用于声明而不是用于定义 当设计头文件时,记住定义和声明的区别是很重要的.定义只可以出现一次,而声明则可以出现多次. 下列语句是一些定义,所以不应该放在头文件里: extern ...

  2. 头文件里面的ifndef /define/endif的作用

    c,c++里面,头文件里面的ifndef /define/endif的作用 今天和宿舍同学讨论一个小程序,发现有点地方不大懂······ 是关于头文件里面的一些地方: 例如:要编写头文件test.h ...

  3. C++头文件为什么要加#ifndef #define #endif

    #ifndef 在头文件中的作用 在一个大的软件工程里面,可能会有多个文件同时包含一个头文件,当这些文件编译链接成一个可执行文件时 ,就会出现大量“重定义”的错误.在头文件中实用#ifndef #de ...

  4. 头文件为什么要加#ifndef #define #endif

    #ifndef 在头文件中的作用 在一个大的软件工程里面,可能会有多个文件同时包含一个头文件,当这些文件编译链接成一个可执行文件时 ,就会出现大量“重定义”的错误.在头文件中实用#ifndef #de ...

  5. 头文件中的#ifndef/#define/#endif 的作用

    在一个大的软件工程里面,可能会有多个文件同时包含一个头文件,当这些文件编译链接成一个可执行文件时,就会出现大量重定义的错误.在头文件中实用#ifndef #define #endif能避免头文件的重定 ...

  6. c++中,保证头文件只被编译一次,避免多重包含的方法

    保证头文件只被编译一次 #pragma once这是一个比较常用的C/C++杂注,只要在头文件的最开始加入这条杂注,就能够保证头文件只被编译一次. #pragma once是编译器相关的,有的编译器支 ...

  7. c语言头文件的认识

    c头文件的作用是什么,和.c文件是怎么联系的,该怎么样编写头文件呢?这些问题我一直没搞明白,在阅读uCOS-II(邵贝贝)“全局变量”部分有些疑惑,今天终于搞清楚了头文件的一些基础知识,特地分享一下. ...

  8. 头文件中ifndef/define/endif的作用以及#pragma once使用

    例如:要编写头文件test.h 在头文件开头写上两行: #ifndef _TEST_H #define _TEST_H//一般是文件名的大写 ············ ············ 头文件 ...

  9. C++学习 之 初识头文件

    声明:            本人自学C++, 没有计算机基础,在学习的过程难免会出现理解错误,出现风马牛不相及的现象,甚至有可能会贻笑大方. 如果有幸C++大牛能够扫到本人的博客,诚心希望大牛能给予 ...

  10. C/C++ 引入头文件时 #include<***.h> 与 #include"***.h" 区别

    两种情况区分: 1.#include <> 编译器只会去系统文件目录中查找,找不到就报错. 2.#include " "  编译器会先在用户目录中查找,再到编译器设定的 ...

随机推荐

  1. Sql语句的两表联合查询

    string sql = "select mID,mName,mSex,mAge,(select fzName from TxlFenZu where ID=mFenZu) as mFenZ ...

  2. WIN32下的模拟时钟

    #include <Windows.h> #include <math.h> #include <tchar.h> #include "resource. ...

  3. 如何阅读 diff 命令的输出

    diff 命令有三种模式:上下文模式(context),合并模式(unified)和普通模式(normal).其中最常用的是合并模式. 合并模式 diff -u f1 f2 --- f1 2024-0 ...

  4. 【Mac + Appium + Java1.8(二)】之Android模拟器自动化测试脚本开发以及简易例子

    直接上代码: import io.appium.java_client.AppiumDriver; import org.junit.After; import org.junit.Before; i ...

  5. sql server 将数据库表里面的数据,转为insert语句,方便小批量转移数据

    create proc [dbo].[proc_insert] (@tablename varchar(256)) as begin set nocount on declare @sqlstr va ...

  6. 五子棋AI:实现逻辑与相关背景探讨(上)

    绪论 本合集将详细讲述如何实现基于群只能遗传算法的五子棋AI,采用C++作为底层编程语言 本篇将简要讨论实现思路,并在后续的文中逐一展开 了解五子棋 五子棋规则 五子棋是一种经典的棋类游戏,规则简单却 ...

  7. 【SpringBoot Demo】MySQL + JPA + Hibernate + Springboot + Maven Demo

    主要包含:springboot+jpa+hibernate+mysql+lombok (两年前写过一个,现在重新记录一个) 1. 目录结构: 2. pom 文件 1 <?xml version= ...

  8. 原生JavaScript实现一个简单的Promise构造函数示例

    下面demo示例,只支持实例的then和catch,代码如下: function PromiseDiffer(fn){ var self = this; this.status = 'pendding ...

  9. MDC – Checkbox

    前言 Checkbox 不是搭配 TextField 使用, 而是搭配 FormField. 所以独立一篇来写. 参考 Docs – Selection controls: checkboxes 效果 ...

  10. BOOST 定时器 stop探究

    Asio是一个建立在Boost所提供的相关组件之上的异步的网络库,可以运行在Win/Linux/Unix等各种平台之上. 不过随着C++11的发布,其对Boost的依赖也越来越少,作者又做了一个不依赖 ...