django signal 拾遗

来源:互联网 发布:大数据分析技术 编辑:程序博客网 时间:2024/05/24 11:14
signal实现了发送者和接受者的解偶,同时源码也演示了适用弱引用实现缓存的例子。

首先看看Signal类的一些主要的接口。

?
1
2
3
4
5
6
7
8
9
10
11
12
class Signal(object):
    def __init__(self, providing_args=None, use_caching=False):
        ......
         
    def connect(self, receiver, sender=None, weak=True, dispatch_uid=None):
        ......
 
    def disconnect(self, receiver=None, sender=None, weak=True, dispatch_uid=None):
        ......
         
    def send(self, sender, **named):
        ......

Signal类,不是指一个具体的信号,而是表示一种类型的信号。

这个信号,可以有多个发送者,也可以有多个接受者。

首先必须将接收者和发送者绑定,这里connect方法实现了这一点。如果不指定sender,

这个接收者会接收所有此信号的发送者。

然后使用send方法,需要指定sender。表示由此sender发送的信号,

对于某些指定sender的接受者,这是很有必要的。

disconnect来断开放送者和接收者的信号联系。


可以看见信号的建立和发射都是由Signal类提供的。所以Signal类也承担着管理发送者和发送者的责任。

Signal类存储它们的数据结构,是通过类里面的receiver属性。通过__init__构造方法,可以看到receivers

是一个列表。receivers里面的每个元素,由key和receiver组成,代表着一个信号的连接。而key是由receiver和sender

生成的。结构如图所示:( (r_key, s_key),  receiver  )。

每个connect和disconnect都会更新receivers。

send也会根据receivers找到相应的接收者。


另外这个模块值得注意的一点是,对receivers的查询,它使用了缓存的技术,通过weakref即若引用的技术。

?
1
 self.sender_receivers_cache = weakref.WeakKeyDictionary()

sender_receiver_cache是一个弱引用的字典。key为sender, value为相应绑定的recierver列表。

sender_receiver_cache在每connect和disconnect里,会被清空。

在send里面会通过 _live_receivers(self, sender)方法,查找对应的receivers, 然后会将查找的结果记录到缓存里。

0 0