模板编译模型
学习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:\users\lee\documents\visual studio 2008\projects\test\test\utilities.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