本文解决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。
如果这篇博文对您有帮助,为了好的网络环境,不建议转载,建议收藏!如果您一定要转载,请带上后缀和本文地址。