C/C++:多个.cpp文件包括同一个.h头文件定义方法

时间:2022-09-08 18:43:34

本文解决multiple definition of `XX'的错误。【出于反爬虫的目的,你不是在http://blog.csdn.net/zhanh1218上看到的,肯定不是最新最全的。】

关于头文件的定义中,请一定加上下面代码(此为头文件保护符):

<span style="font-size:14px;"><span style="font-size:12px;">#ifndef PERSON_H_
#define PERSON_H_
// 你的代码块
#endif /* PERSON_H_ */</span></span>

当中PERSON_H_为保护符的名字,一般建议与类名保持一致!样例中我的类名为Person.h。

每当编译器遇到#include时,都要去编译相关代码块,可是编译器不知道是不是已经编译过了,假设编译过了还去编译,那是不是等于代码块写了两次呢?所以,须要有不反复编译的机制,而这个机制正式以上代码。

详细实现为:#ifdef当且仅当变量已定义时为真,#ifndef当且仅当变量没有定义时为真。一旦检測结果为真,则运行兴许操作直至遇到#endif。

也就是说:假设首次include "Person.h",PERSON_H_是未定义,此时,编译器会define这个保护符,并运行代码块的编译!直到遇到#endif。下次遇到这个保护符,就不会运行代码块的编译了。这种机制保证了不会反复编译。

实际使用中,我发现,单个cpp文件里多次include 同一个.h头文件或者头文件里多次include某个头文件,不会有问题。可是,多个cpp文件都include 同一个.h头文件时,这样会出问题。问题是类外定义的非static及非inline函数还是会报multiple
definition of `XX'的错误。【也就是说:#define的作用域仅仅是单个.cpp,而不是全局全部的.cpp文件】

终于解决方法是:仅仅在头文件定义类的申明和类的主体定义(也就是{}内的内容),在一个同名的.cpp文件中定义类外函数的实现!问题完美解决。所以,就算是大神写的书,书上也不全然是对的,或者表述的所有都清楚。

那么为什么头文件能够定义成下面形式呢?而不是仅仅申明,不定义类体呢?

<span style="font-size:14px;"><span style="font-size:12px;">class A
{
// 类定义
};</span></span>

类的定义,仅仅是告诉编译器,类的数据格式是怎样的,实例话后对象该占多大空间。  类的定义也不产生目标代码。因此它和普通变量的声明唯一的差别是不能在同一编译单元内出现多次。

还有一个原因就是,类能够在多个.cpp文件中重定义,变量却不行,除非用extern或者staic修饰的变量。

至于普通变量:同意static型的变量的定义;同意extern申明(不能定义!);直接申明比如int
a; 是不行的,也是多次又一次定义。

extern表明该变量在别的地方已经定义过了,在这里要使用那个变量;static 表示静态的变量,分配内存的时候,存储在静态区,不存储在栈上面。【一篇不错的Blog:点击打开链接

以下是代码演示样例,此实例部分为C++ Primer练习题。【反爬虫,第二天更新代码!】

/*********************************************************************
* file_name: vector_test.cpp
*
* Created on: 2014年6月28日 下午3:34:23
* Author: The_Third_Wave, Blog: http://blog.csdn.net/zhanh1218
* Email: zhanh121823@sina.com
* Last modified: 2014年6月28日 下午3:34:23
*********************************************************************/ #include <iostream>
#include <vector>
#include <string>
#include "Headers/Person.h" extern int k;
int main()
{
std::vector<Person> per = {{"The_Third_Wave", 100, "Blog: http://blog.csdn.net/zhanh1218"}}; // 类初始化+vector初始化,所以{{}, {}}必须的
for (auto &p: per)
{
print(std::cout, p);
}
}
/*********************************************************************
* file_name: ddqdq.cpp
*
* Created on: 2014年6月28日 下午10:28:42
* Author: The_Third_Wave, Blog: http://blog.csdn.net/zhanh1218
* Email: zhanh121823@sina.com
* Last modified: 2014年6月28日 下午10:28:42
*********************************************************************/
#include <iostream>
#include <vector>
#include "Headers/Person.h" extern int k;
/*********************************************************************
* file_name: Person.h
*
* Created on: 2014年6月28日 下午11:47:08
* Author: The_Third_Wave, Blog: http://blog.csdn.net/zhanh1218
* Email: zhanh121823@sina.com
* Last modified: 2014年6月28日 下午11:47:08
*********************************************************************/ #ifndef PERSON_H_
#define PERSON_H_
/*****************************BEGIN***********************************/
#include <iostream>
#include <string>
using namespace std;
extern int a;
class Person
{
friend istream &read(istream &is, Person &item);
friend ostream &print(ostream &os, const Person &item);
public: Person() = default;
Person(const string &n, unsigned int a, string add):
name(n), age(a), address(add) { }
Person(istream &); string Name() const {return name;}
unsigned int Age() const {return age;}
string Address() const {return address;} private:
string name = "";
unsigned int age = 1;
string address = "";
}; inline
Person::Person(istream &is)
{
read(is, *this);
}
/******************************END************************************/
#endif /* PERSON_H_ */
/*********************************************************************
* file_name: Person.cpp
*
* Created on: 2014年6月28日 下午10:55:32
* Author: The_Third_Wave, Blog: http://blog.csdn.net/zhanh1218
* Email: zhanh121823@sina.com
* Last modified: 2014年6月28日 下午10:55:32
*********************************************************************/
#include <iostream>
#include <string>
#include "Person.h"
using namespace std; istream &read(istream &is, Person &item)
{
is >> item.name >> item.age >> item.address;
return is;
}
ostream &print(ostream &os, const Person &item)
{
os << item.name << " " << item.age << " " << item.address << endl;
return os;
}

还有不懂的请留言。

本文由@The_Third_Wave(Blog地址:http://blog.csdn.net/zhanh1218)原创。还有未涉及的,会不定期更新,有错误请指正。

假设你看到这篇博文时发现不完整,那是我为防止爬虫先公布一半的原因,请看原作者Blog。

假设这篇博文对您有帮助,为了好的网络环境,不建议转载,建议收藏!假设您一定要转载,请带上后缀和本文地址。

C/C++:多个.cpp文件包括同一个.h头文件定义方法的更多相关文章

  1. Cocos2dx3&period;4 VS2013无法打开包括文件extensions&sol;ExtensionExport&period;h解决的方法

    本来打算看白鹭引擎的.可是又被叫回来研究新引擎呢,不搞不知道,一搞发现cocos2dx新版本号3.4又有了一些变化. 我查了网上的资源,都没有解决的方法.我如今应该是第一个出这个问题的解决的方法哦.. ...

  2. C语言中头文件和cpp文件解析

    务必提前预读这里的内容:http://www.cnblogs.com/stemon/p/3999844.html 回到cpp文件与头文件各写什么内容的话题上: 理论上来说cpp文件与头文件里的内容,只 ...

  3. c中头文件在cpp文件里引用和&period;h文件引用的思考

    我们在编敲代码中头文件是常常使用的. 可是头文件是应该包括在.H文件里还是在.cpp文件里.在这个其中有什么样去差别呢. 假如说我们编写了一个a.cpp  .我们将a.cpp文件的变量和函数申明在a. ...

  4. C&plus;&plus;中template的&period;h文件和&period;cpp文件的问题

    在C++中,用到类模板时,如果类似一般的类声明定义一样,把类声明放在.h文件中,而具体的函数定义放在.cpp文件中的话,会发现编译器会报错.如类似下面代码: //test.h文件 #ifndef TE ...

  5. qt 在ui界面添加控件后在cpp文件中无法调用?

    问题:qt 在ui界面添加控件后在cpp文件中无法调用? 解决方法:在build选项中选择“重新build项目”,再次在cpp中调用添加的控件发现可以调用了. 还有一种情况导致添加控件后无法调用,就是 ...

  6. VS 2010不显示头文件源文件和所有以前分类的文件夹,&ast;&period;h 和&ast;&period;cpp都显示在同一个文件

    打开VS后不显示头文件源文件和所有以前分类的文件夹,*.h 和*.cpp都显示在同一个文件 点击右图红色指示显示所有文件夹按钮,就能恢复.

  7. fatal error LNK1169&colon; 找到一个或多个多重定义的符号或多个&period;c&sol;&period;cpp文件想同时调用定义在&period;h文件里面的全局变量,防止重定义变量问题。

    为什么.h文件中不能定义全局变量? 原因: 存在多次创建变量.如果头文件中可以定义全局变量,那么每个包含该头文件的文件里都会有该全局变量的定义.因为C语言的include是直接将文件嵌入到includ ...

  8. h和&period;cpp文件的区别

    关于头文件和源文件的分别 首先,我们可以将所有东西都放在一个.cpp文件内. 然后编译器就将这个.cpp编译成.obj,obj是什么东西? 就是编译单元了.一个程序,可以由一个编译单元组成, 也可以有 ...

  9. 配置caffe过程中,生成解决方案出错。无法打开包括文件&colon; &OpenCurlyDoubleQuote;gpu&sol;mxGPUArray&period;h”

    ------ 已启动生成: 项目: matcaffe, 配置: Release x64 ------12> MatlabPreBuild.cmd : Create output director ...

随机推荐

  1. Canvas坐标系转换

    默认坐标系与当前坐标系 canvas中的坐标是从左上角开始的,x轴沿着水平方向(按像素)向右延伸,y轴沿垂直方向向下延伸.左上角坐标为x=0,y=0的点称作原点.在默认坐标系中,每一个点的坐标都是直接 ...

  2. 使用 SharedPreferences 实现数据的存储和读取

    在开发的过程中我们必须遇到的就是如何对用户的数据进行有效的存储以及读取.我们举个例子,现在我们使用app,当我们登陆一个账号的时候选择记住密码软件就会记住我们的账号以及密码,我们退出当前账号,就可以直 ...

  3. HDU 5968 异或密码 【模拟】 2016年中国大学生程序设计竞赛(合肥)

    异或密码 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others) Problem Des ...

  4. HDOJ&sol;HDU 1256 画8&lpar;绞下思维~水题&rpar;

    Problem Description 谁画8画的好,画的快,今后就发的快,学业发达,事业发达,祝大家发,发,发. Input 输入的第一行为一个整数N,表示后面有N组数据. 每组数据中有一个字符和一 ...

  5. HDOJ&sol;HDU 1073 Online Judge&lpar;字符串处理~&rpar;

    Problem Description Ignatius is building an Online Judge, now he has worked out all the problems exc ...

  6. 纯js实现DIV拖拽

    写代码的时候遇到需要对绝对布局的div进行拖拽的功能,起初为了省事直接在网上扒拉了一番,看到大神张鑫旭的一篇文章<JavaScript实现最简单的拖拽效果>,便直接拿来使用(膜拜大神).但 ...

  7. ASP&period;NET Aries 高级开发教程:Excel导入之多表高级导入配置(中)

    前言: 在面对Excel的各种复杂导入情况中,多表导入是很常见的情景. 今天就来写一下多表导入是如何配置的. 1.自定义导入模板 怎么自定义: 其实就是自己新建一个Excel了,把列头都写好. 不过有 ...

  8. Linux 总线、设备、驱动模型 与 设备树

    1.总线.设备.驱动模型 本着高内聚.低耦合的原则,Linux 把设备驱动模型分为了总线.设备和驱动三个实体,这三个实体在内核里的职责分别如下: 设备和驱动向总线进行注册,总线负责把设备和对应的驱动绑 ...

  9. IE8不支持数组的indexOf方法 如何解决

    转自:http://www.jbxue.com/article/8367.html 原因分析: 这是一个js bug, 在IE8下,js数组没有indexOf方法,会报错:而在其它浏览器下(Firef ...

  10. jQuery CircleCounter的环形倒计时效果

    在线演示1 本地下载 使用jQuery插件CircleCounter生成的环形倒计时效果,这个插件使用HTML5画布生成动画效果,还不错,大家可以试试! 顺带手录制了个代码,大家不吝赐教:http:/ ...