类模板的模板友元函数定义(转载)

本文介绍了C++中类模板的模板友元函数的两种定义方式:直接定义在类模板内和声明在类模板内而定义在外部。通过具体示例说明了如何避免链接时无法解析的错误,并强调了正确声明友元模板函数的重要性。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

转自:http://www.cppblog.com/zerolee/archive/2010/11/03/132344.html

类模板的模板友元函数定义有2种方式:
1. 将友元模板函数直接定义在类模板中。这种方式比较简单直接。
2. 将友元模板函数声明在类模板中,定义在类模板之外。这种方式的写法,如果不小心,通常会出现编译没问题,链接时无法解析的错误。
以下是一个简单的正确的例子:

 1  #include  < iostream >
 2  #include  < vector >
 3 
 4  template  < typename T >
 5  class  Number;
 6 
 7  template  < typename T >
 8  void  print( const  Number < T >&  n);
 9 
10  template  < typename T >
11  std::ostream &   operator   <<  (std::ostream &  os,  const  Number < T >&  n);
12 
13  template  < typename T >
14  std::istream &   operator >> (std::istream &   is , Number < T >&  n);
15 
16  template  < typename T, typename T2 >
17  void  printVector( const  std::vector < T2 >&  vt,  const  Number < T >&  n);
18 
19  template  < typename T >
20  class  Number {
21  public :
22      Number(T v) 
23          : val(v) {}
24       ~ Number() {}
25 
26  private :
27      T val;
28  public :
29      friend  void  print<T>  ( const  Number < T >&  n);
30      friend std::ostream &   operator   <<   <T> (std::ostream &  os,  const  Number < T >&  n);
31      friend std::istream &   operator >>   <T> (std::istream &   is , Number < T >&  n);
32 
33      friend Number < T >&   operator   +=  (Number < T >&  a,  const  Number < T >&  b)
34      {
35          a.val  +=  b.val;
36           return  a;
37      }
38      template <typename T2>
39      friend  void  printVector<T> ( const  std::vector < T2 >&  vt,  const  Number < T >&  n);
40      template <typename T2>
41       void  memFunc( const  std::vector < T2 >&  vt,  const  Number < T >&  n);
42  };
43 
44  template  < typename T >
45  std::ostream &   operator   << (std::ostream &  os,  const  Number < T >&  n)
46  {
47       os  <<  n.val  <<  std::endl;
48        return  os;
49  }
50 
51  template  < typename T >
52  std::istream &   operator   >> (std::istream &   is , Number < T >&  n)
53  {
54       is   >>  n.val;
55       return   is ;
56  }
57 
58  template  < typename T >
59  void  print < T >  ( const  Number < T >&  n)
60  {
61      std::cout  <<  n;
62  }
63 
64  template  < typename T, typename T2 >
65  void  printVector( const  std::vector < T2 >&  vt,  const  Number < T >&  n)
66  {
67       for  (unsigned  int  i  =   0 ; i  <  vt.size(); i ++ )
68          std::cout  <<  vt.at(i)  <<   "   " ;
69      std::cout  <<   " =>  "   <<  n;
70  }
71 
72  template <typename T>
73  template <typename T2>
74  void  Number < T > ::memFunc( const  std::vector < T2 >&  vt,  const  Number < T >&  n)
75  {
76       for  (unsigned  int  i  =   0 ; i  <  vt.size(); i ++ )
77          std::cout  <<  vt.at(i)  <<   "   " ;
78      std::cout  <<   " =>  "   <<  n;
79  }
80 

1) 以上代码中,operator +=被定义在类模板内部。其他3个函数先被声明(需提前声明类模板,如果模板函数的参数中含有类模板),然后在类模板中被声明为友元函数, 之后被定义在类模板体之外。
2) 请注意当模板函数被声明为类模板的友元时, 在函数名之后必须紧跟模板实参表,用来代表该友元声明指向函数模板的实例。否则友元函数会被解释为一个非模板函数,链接时无法解析。
3) 友元模板函数的模板参数类型,并不一定要求是类模板的参数类型,也可以另外声明。

转载于:https://www.cnblogs.com/charm/archive/2011/03/05/1971838.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值