eXile 的专栏

我的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所在线程中执行的。

posted on 2008-01-09 10:59 eXile 阅读(4145) 评论(5)  编辑 收藏 引用 所属分类: C/C++代码片段

评论

# re: 我的signal/slot实现(修订版) 2008-01-09 15:34 eXile

修正了一个bug ( signal.h: 92)

template<...>
class signal
{
//....
template <class T>
void connect(SP_NS::shared_ptr<T> const& f)
{
_connect(slot_type(f.get()), f);
}
};  回复  更多评论   

# re: 我的signal/slot实现(修订版) 2008-01-22 21:58 姓名

错别字?
void f()(int)
{ printf("Bar"); }
function returns function  回复  更多评论   

# re: 我的signal/slot实现(修订版) 2008-01-23 00:47 eXile

谢谢,改过来了。  回复  更多评论   

# re: 我的signal/slot实现(修订版) 2008-01-29 16:30 zambiafrog@gmail.com

我还没来得及读Boost的。Loki的Functor比较类似,比较轻量级,也容易看懂。
  回复  更多评论   

# re: 我的signal/slot实现(修订版) 2008-03-25 09:41 suxiaojack

Functor!呵呵,很多实现版本。  回复  更多评论   


只有注册用户登录后才能发表评论。
网站导航: 博客园   IT新闻   BlogJava   知识库   博问   管理


导航

<2008年1月>
303112345
6789101112
13141516171819
20212223242526
272829303112
3456789

统计

常用链接

留言簿(18)

随笔分类

随笔档案

服务器编程

搜索

最新评论

阅读排行榜

评论排行榜