今天复习C++ Primer的时候,看到了关于C++类的内联成员函数的放置,应该放在头文件中。那么这到底是为什么
呢?仅仅是一种代码规范问题还是必须这样做呢?
下面我就来讲讲我自己的理解吧。要彻底理解这个问题,首先就要了解下函数的声明和定义了。我们知道,函数可以
在多处声明,但只能在一个地方定义,不然就会出现重定义。大部分函数默认是外部链接,而inline函数默认为内部链
接。也就是说inline函数只能在本文件中使用,对其他文件是不可见的。一般我们使用某个类的时候,都是在文件中加
上该类的头文件,以便我们可以使用该类的接口。而我们类的成员函数的实现都是放在相应的.cpp文件中的,而在.h
文件中声明。这样我们便可以通过.h文件中的成员函数的声明找到其定义,继而使用成员函数了。但如果将inline函数
放在.cpp文件中,那么其只对.cpp文件有效,这样我们就无法访问它了。所以我们将其放在类的声明的头文件中,这
样通过包含该头文件来使用它。
下面写个实际的例子来说明一下,我先把内联函数放到类声明的头文件中:
/*test.h*/ #ifndef TEST_H #define TEST_H #include <iostream> using std::cout; using std::endl; class test { public: test():x(10){} inline void print(); void display (int y); private: int x; }; void test::print() { cout << x << endl; } #endif
/*test.cpp*/ #include <iostream> #include "test.h" using std::cout; using std::endl; void test::display(int y) { cout << x * y << endl; }
/*main.cpp*/ #include <iostream> #include "test.h" using namespace std; int main() { test T; T.display(10); T.print(); system("pause"); return 0; }
运行结果正常,下面来看看将内联函数放到.cpp中去:
/*test.h*/ #ifndef TEST_H #define TEST_H #include <iostream> using std::cout; using std::endl; class test { public: test():x(10){} inline void print(); void display (int y); private: int x; }; #endif
/*test.cpp*/ #include <iostream> #include "test.h" using std::cout; using std::endl; void test::print() { cout << x << endl; } void test::display(int y) { cout << x * y << endl; }
测试函数和上面的main.cpp是一样的。这是出现了错误:
error LNK2019: 无法解析的外部符号 "public: void __thiscall test::print(void)" (?print@test@@QAEXXZ),该符号在函
数 _main 中被引用。如果我将测试函数改为:
int main() { test T; T.display(10); //T.print(); system("pause"); return 0; }
那么运行结果正常。从此可以得出结论:内联函数放在头文件或者.cpp中都是没有错的,但如果我们需要在程序中访
问它,那么就必须将其放在头文件中。