关于模板类中友元函数的声明与定义

//Widget.h
#ifndef _WIDGET_H_
#define _WIDGET_H_
#include <iostream>
using namespace std;

template<class T>
class Widget
{
	friend void AccessWidget(const Widget<T>& w);
public:
	explicit Widget(const T& a):a_(a)
	{}
private:
	T a_;
};

//Widget.cpp
#include"widget.h"

template<class T>
inline void AccessWidget(const Widget<T>& w)
{
	cout<<w.a_<<endl;
}

int test_Widget()
{
	Widget<int> w(5);
	AccessWidget(w);
	return 0;
}
#endif
调用test_Widget后,编译将出现以下错误:
1>main.obj : error LNK2001: unresolved external symbol "void __cdecl AccessWidget(class Widget<int> const &)" (?AccessWidget@@YAXABV?$Widget@H@@@Z)


原因分析:
1、模板的定义文件和声明头文件不要分开,目前来说,gcc和msvc80还都不支持分开。
2、这个问题属于模板friend函数问题,不能简单地将声明与定义分开,然而即使放在同一个文件,编译也会报错的。


解决办法:
1、最直接的办法是将函数的定义放在声明的地方,目前大部分的编译器都支持。有些人可能会采用将友元函数定义的cpp文件改为inl文件,然后在h头文件尾部包含inl文件的方法来实现,但事实上这样做依然无法编译。

2、另外一种办法较为麻烦,并且有的编译器还不一定支持;需要预先声明模板类和友元函数,以及在类中友元函数声明时候加以特化处理,以上代码修改如下就可以顺利通过编译:

//widget.hpp
#ifndef _WIDGET_HPP_
#define _WIDGET_HPP_
#include <iostream>
using namespace std;

//预先声明 Widget
template<class T>
class Widget;

//预先声明AccessWidget
template<class T>
void AccessWidget(const Widget<T>& w);

template<class T>
class Widget
{
	friend void AccessWidget<>(const Widget<T>& w);//需要在此处加特化处理
public:
	explicit Widget(const T& a):a_(a)
	{}
private:
	T a_;
};

template<class T>
inline void AccessWidget(const Widget<T>& w)
{
	cout<<w.a_<<endl;
}

int test_Widget()
{
	Widget<int> w(5);
	AccessWidget(w);
	return 0;
}
#endif


  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值