django signal 拾遗

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

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即若引用的技术。

 self.sender_receivers_cache = weakref.WeakKeyDictionary()

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

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

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

你可能感兴趣的:(django signal 拾遗)