分类: C/C++
2008-05-18 18:09:55
来源: |
|
double max(double first, double second); //..该函数的其它版本 尽管这个函数针对不同的数据类型其实现都是一样的,但程序员必须为每一种数据类型定义一个单独的版本: complex max(complex first, complex second) date max(date first, date second)
用普通函数来实现抽象操作会迫使你定义多个函数实例,从而招致不小的维护工作和调试开销。解决办法是使用函数模板代替普通函数。 使用函数模板 函数模板解决了上述所有的问题。类型无关并且只在需要时自动实例化。本文下面将展示如何定义函数模板以便抽象通用操作,示范其使用并讨论优化技术。 第一步:定义 函数模板的声明是在关键字 template 后跟随一个或多个模板在尖括弧内的参数和原型。与普通函数相对,它通常是在一个单元里声明,而在另一个单元中定义,你可以在某个头文件中定义模板。例如: // file max.h int n=10,m=16; std::complex<double> c1, c2; 第二步:改进设计 上述的 max() 的实现还有些土气——参数t1和t2是用值来的。对于像 int,float 这样的内建数据类型来说不是什么问题。但是,对于像std::complex 和 std::sting这样的用户定义的数据类型来说,通过引用来传递参数会更有效。此外,因为 max() 会认为其参数是不会被改变的,我们应该将 t1和t2声明为 const (常量)。下面是 max() 的改进版本: template <class T> T max(const T& t1, const T& t2) 额外的问题 很幸运,标准模板库或 STL 已经在 <algorithm> 里定义了一个叫 std::max()的算法。因此,你不必重新发明。让我们考虑更加现实的例子,即字节排序。众所周知,TCP/IP 协议在传输多字节值时,要求使用 big endian 字节次序。因此,big endian 字节次序也被称为网络字节次序.Network byte order)。如果目的主机使用 little endian 次序,必须将所有过来的所字节值转换成 little endian 次序。同样,在通过 TCP/IP 传输多字节值之前,主机必须将它们转换成网络字节次序。你的 socket 库声明四个函数,它们负责主机字节次序和网络字节次序之间的转换: unsigned int htonl (unsigned int hostlong); 这些函数实现相同的操作:反转多字节值的字节。其唯一的差别是方向性以及参数的大小。非常适合模板化。使用一个模板函数来替代这四个函数,我们可以定义一个的模板,它会处理所有这四种情况以及更多种情形: template <class T> T byte_reverse(T val); 为了确定 T 实际的类型,我们使用 sizeof 操作符。此外,我们还使用 STL 的 std::reverse 算法来反转值的字节: template <class T> T byte_reverse(T val) int main() 注:模板使用不当会影响.exe 文件的大小,也就是常见的代码浮肿问题 |