首先看看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, 然后会将查找的结果记录到缓存里。