转自:
用bind来解耦应用了boost::bind, boost::function, boost::test
代码稍有修改。
#define BOOST_AUTO_TEST_MAIN
#include <boost/test/auto_unit_test.hpp>
#include <boost/test/included/unit_test_framework.hpp>
#include <boost/thread/thread.hpp>
#include <boost/bind.hpp>
#include <boost/function.hpp>
#include <string>
using namespace std;
//声明一个函数指针的类型
typedef void (* ON_SEGMENT) (char *);
//用function<> 和指针回调指针来解除耦合
//注意解耦效果: CReader 和 CSaver 没有任何关系
class CReader
{
public:
CReader()
{
m_pFunc=NULL;
m_functor=NULL;
};
void UseFreeFunc(ON_SEGMENT pFunc)
{
m_pFunc=pFunc;
};
void UseFunctor(boost::function<void(char *)> functor)
{
m_functor= functor;
};
void Read(char * pchar)
{
if(m_pFunc)
{
m_pFunc(pchar);
};
if(m_functor)
{
m_functor(pchar);
};
}
private:
ON_SEGMENT m_pFunc;//函数指针
boost::function<void(char *)> m_functor; //函数对象
};
class CSaver
{
public:
//接受输入的字符,保存在内部变量m_str
void Input(char * pchar)
{
m_str.append(pchar);
};
string m_str;
};
BOOST_AUTO_TEST_CASE(test_bind)
{
//两个类并不知道对方的任何细节
CSaver saver;
CReader reader;
//预先 把函数 CSaver::Input 和要操作的数据 &saver 注册到 reader
//只要函数形式符合(返回值是void,传入参数是char *),都可以注册
//解除CSaver 和 CReader的耦合
//bind产生的函数对象,重载操作符()
//函数对象的本质是 函数指针加this指针,即 操作和数据
//函数对象 比 函数指针优越的地方是可以带有内部状态
reader.UseFunctor( boost::bind(&CSaver::Input, &saver, _1) );
//延时调用,解除CSaver 和 CReader的耦合
//把字符从reader 放入 saver里面
reader.Read("Hello");
reader.Read("Mafeitao");
BOOST_CHECK(saver.m_str=="HelloMafeitao");
}
//asio 里面的 async_accept 的简化版本
//可以接受 函数指针; 比如 async_accept(FreeFunc,Para)
//可以接受 函数对象; 比如
// CSaverFunctor func;
//async_accept(func,Para)
//可以接受 bind
template <typename AcceptHandler,typename Para>
void async_accept(AcceptHandler handler, Para p)
{
handler(p);
};
// 自由函数,没有内部状态
void FreeFunc(char * pChar)
{
printf("In FreeFunc %s\n", pChar);
}
BOOST_AUTO_TEST_CASE(bind_asio)
{
//类 CSaver 和 模板函数 async_accept,没有任何关系
CSaver saver;
char *pchar="Hello";
//bind生成一个临时的functor类 CSaverTemp
//类似于
/*
class CSaverTemp
{
public:
void operator ()(char * pchar)
{
m_str.append(pchar);
};
string m_str
}
CSaverTemp saver;
async_accept(saver,pchar);
}
*/
//全局函数 async_accept,接受一个bind产生的函数对象( 函数指针和this指针)
async_accept( boost::bind(&CSaver::Input, &saver,_1) , pchar);
BOOST_CHECK(saver.m_str=="Hello");
pchar="Mafeitao";
async_accept( boost::bind(&CSaver::Input, &saver,_1) , pchar);
BOOST_CHECK(saver.m_str=="HelloMafeitao");
//接受 全局函数指针,和下面的效果一样
async_accept( boost::bind(FreeFunc,_1), pchar);
//接受 全局函数指针,和上面的效果一样
async_accept( FreeFunc, pchar);
}
//带有模板函数的类
class TReader
{
public:
//模板函数
template <typename OnReadHandler,typename Para>
void Read(OnReadHandler & handler, Para p)
{
//可以接受 函数指针,参数
//可以接受 函数对象,参数
//可以接受 bind
handler(p);
};
};
BOOST_AUTO_TEST_CASE(bind_template)
{
TReader reader;
CSaver saver;
char *pchar="Hello";
boost::function<void (char*)> handler
= boost::bind(&CSaver::Input, &saver,_1);
reader.Read(handler, pchar);
BOOST_CHECK(saver.m_str=="Hello");
pchar="Mafeitao";
reader.Read(handler, pchar);
BOOST_CHECK(saver.m_str=="HelloMafeitao");
}
//函数对象,重载操作符()
//函数对象的本质是 函数指针加this指针,即 操作和数据
//函数对象 比 函数指针优越的地方是可以带有内部状态
class CSaverFunctor
{
public:
void operator ()(char * pchar)
{
m_str.append(pchar);
};
string m_str;
};
BOOST_AUTO_TEST_CASE(test_functor)
{
//带有模板函数的类
TReader reader;
//函数对象
CSaverFunctor saver;
//把字符从reader 放入 saver里面
char *pchar="Hello";
reader.Read( saver , pchar);
pchar="Mafeitao";
reader.Read( saver , pchar);
BOOST_CHECK(saver.m_str=="HelloMafeitao");
}