我的signal/slot实现(修订版)

我的signal/slot实现(修订版)

  自从掌握了signal/slot机制后,它已经变成了我在设计中一种必不可少的方法,如何在非QT环境中使用这种机制,也是我一直在考虑的问题。目前的这个版本,相对来说还是比较满意的,而且个人认为它优于Boost::signal实现。
  这个实现的主要特点:
  1。不再使用tr1::fucntion, 主要是因为它没有办法提供一致的相等性比较,另外,它也并没有真正解决使用成员函数时指针的有效期问题(当然也可以通过tr1::bind加上tr1::shared_ptr达到这一目的)。

  2。比较好地解决了对象指针的有效性跟踪问题。

使用举例:
struct Foo
{
      void operator()(
int )
     {   printf(
" Foo " );  }
};

struct Bar
{
      void f(
int )
     {   printf(
" Bar " );  }
};

使用方法如下:
signal < int >  sig;
{
     shared_ptr
< Foo >  foo( new  Foo);
     shared_ptr
< Bar >  bar( new  Bar);

     sig.connect(foo);
     sig.connect(bar, 
& Bar::f);
   
     sig(
0 );   //   print: FooBar
//  连接自动断开

sig(
0 );    //  print  nothing

在单线程中,也可以如下使用:
signal<int> sig;
{
     Foo foo;

     Bar  bar;
        trackable t;

     sig.connect(&foo);
     sig.connect(&bar, 
&Bar::f);
         sig.track(&foo, t.weak_this());
         sig.track(&bar, &Bar::f, t.weak_this());
   
     sig(
0);   //  print: FooBar
// 连接自动断开

sig(
0);    // print nothing

或者像下面这样:
struct Bar : trackable
{
    void f()( int )
   {   printf(
" Bar " );  }
};


signal < int >  sig;
{
   Bar  bar;
   sig.connect(
& bar,  & Bar::f);  
   sig(
0 );    //   print: Bar
//  连接自动断开

sig(
0 );     //  print  nothing

我在VS2005和gcc中测试过,没有试过其它的编译器,不过肯定不支持VC6.

下载地址  http://www.cppblog.com/Files/eXile/signal.rar

使用:依赖于tr1::shard_ptr, 若要使用boost::shared_ptr,如下
#include <boost/weak_ptr.hpp>
#define SP_NS boost
所有定义位于namespace my中。


说明:它是线程安全的吗?如何就在多线程中使用它?

Signal本身可看作是一个连接的slot的容器,正象stl中的容器一样,本身不是线程安全的,必须在锁保护下才能用于多线程中。还要注意以下几点:
1)多线程中不要使用trackable跟踪对象,要使用shared_ptr管理对象(如文中的第一种用法)。
2)由于对象销毁而造成的连接断开操作是线程安全的。
3)一个避免加锁的简单方法,就是在线程启动前,连接好所有slot。由于上述第二点,这时不用加锁就可以用于多线程中,要注意的一点是,所有slot函数是在signal所在线程中执行的。

你可能感兴趣的:(我的signal/slot实现(修订版))