模板编译模型

学习C++ Primer第四版,P542的模板编译模型的时候,按照上面的说法写“包含模型”程序

utilities.h

1 #ifndef GUARD_UTILITIES_H
2 #define GUARD_UTILITIES_H
3 
4 template <class T> int compare(const T&, const T&);  //模板函数的声明
5 
6 #include "utilities.cpp"
7 #endif

utilities.cpp

 1 #include "utilities.h"
 2 
 3 //implementation file utilities.cpp
 4 template <class T> int compare(const T &v1,const T &v2)
 5 {
 6     if(v1 < v2){return -1;}
 7     if(v2 < v1){return 1;}
 8     return 0;
 9 }
10 #endif

main.cpp

 1 #include "utilities.h"
 2 #include <iostream>
 3 
 4 
 5 using namespace std;
 6 
 7 int main()
 8 {
 9     const int a = 20;
10     const int b = 20;
11     int c;
12     c = compare(a, b);
13     cout << c << endl;
14 }

结果出现错误:error C2995: “int compare(const T &,const T &)”: 函数模板已经定义

1>        c:usersleedocumentsvisual studio 2008projects est estutilities.h(4) : 参见“compare”的声明

后来在csdn上看了下,有以下三种解释:

1.在.h文件中不包含.cpp;在main函数里#include .cpp也可以

2.将定义和声明放在一个文件中,不要分开弄

3.将utilities.cpp文件不包含在工程中,其他代码比便,编译通过可运行

非常好的解释:

查了一些资料,终于弄明白了

原来C++ Primer是采用这样的技术让模板定义和实现分离

1. 在Queue.h文件的确写有#include "Queue.cpp"(这是对的)
2. 而建立的工程里面,并不把Queue.cpp包含进来(放在磁盘目录下,但别加入工程),也就是Queue.cpp并不直接参与编译链接,只有main()参加编译,
3. 在main()所在的文件中,#include "Queue.h"

由于1和3的效果,使分离的模板定义和实现合并到了main()所在的文件中
而由于Queue.cpp本身不直接编译(仅间接参与main()所在的文件的编译),所以也不会出现重定义错误

你之所以在windows下用命令行中用nmake可以编译成功,是因为他的makefile文件里面就是按上述要求写的,即makefile指明只编译main()所在的文件

参考地址:http://bbs.csdn.net/topics/320028181


原文地址:https://www.cnblogs.com/laobusi/p/3144994.html