Django Channels - 根据打开的网络套接字数多次调用接收器函数

Django Channels - receiver function called multiple times depending on number of open web sockets

我不明白 Django Channels 2.4.0 的行为。当多个 Web 套接字打开时,接收器函数的调用频率与 Web 套接字打开的频率一样高。

此处是频道文档中著名的聊天示例中的最少代码

    # Receive message from WebSocket
async def receive(self, text_data):
    text_data_json = json.loads(text_data)
    message = text_data_json['message']

    # Send message to room group
    await self.channel_layer.group_send(
        self.room_group_name,
        {
            'type': 'chat_message',
            'message': message
        }
    )

# Receive message from room group
async def chat_message(self, event):
    print(event) # this is called as often as sockets are open
    message = event['message']
    # Send message to WebSocket
    await self.send(text_data=json.dumps({
        'message': message
    }))

当我想在由接收函数触发的数据库中执行逻辑操作时,问题变得很严重。然后根据打开的网络套接字数,事情开始多次完成。知道我在这里遗漏了什么吗?

Django 通道为每个打开的 websocket 连接创建一个 Consumer class 实例。

group_send 的想法是调用对应于您的 type 值的函数。在已订阅 (group_add) 该组的每个消费者实例上调用此函数。

如果您有不依赖于各个连接的工作要做(例如,每个连接的用户 ID 可能不同),请在 group_send 消息之前完成。

因此,如果您需要查询数据库,请在 group_send 之前执行此操作,然后序列化结果并将它们作为 group_send 消息的一部分通过通道层发送。

然而,如果您的数据库查询需要考虑订阅的用户(也就是过滤掉他们不允许看到的信息),那么您需要在 chat_message 处理程序中执行此操作,并且它会在每个订阅的连接。或者,您可以在 group_send 之前进行查询以公开所需的信息,以便在您的 chat_message 中您可以检查消息以确定是否可以将其发送给用户。