刷新客户端网页时如何重新连接到 Web Socket

How to reconnect to a Web Socket when I refresh the client web page

当我刷新客户端网页时,我停止从 tornado 套接字服务器接收数据。如何重新连接到流?

我试过将连接对象附加到列表,然后在连接关闭时从列表中删除连接,但事实证明,当您刷新页面时,连接永远不会关闭,它根据服务器保持活动状态,但它也不再在客户端接收数据:(

这是我的龙卷风服务器

# python 3
from tornado import web, httpserver, ioloop, websocket, options
from time import time, sleep


class ChannelHandler(websocket.WebSocketHandler):
    """Handler that handles a websocket channel"""
    connections = list()

    @classmethod
    def urls(cls):
        return [(r'/websocket', cls, {})]

    def initialize(self):
        self.channel = None

    def open(self):
        # When Client opens a websocket
        # add the new connnection to connections
        self.connections.append(self)

    def on_message(self, message):
        # Message received on channel
        # keep sending all connected clients the time info
        while True:
            [client.write_message({'time()': str(time())}) for client in self.connections]
            sleep(1)
            print('still sending')

    def on_close(self):
        # Channel is closed
        # delete client from active connections if they close connection
        self.connections.remove(self)
        print('CLOSED connection?')

    def check_origin(self, origin):
        # Override the origin check if needed
        return True


def main():
    # Create tornado application and supply URL routes
    app = web.Application(ChannelHandler.urls())
    # Setup HTTP Server
    http_server = httpserver.HTTPServer(app)
    http_server.listen(8000, 'localhost')
    # Start IO/Event loop
    ioloop.IOLoop.instance().start()


if __name__ == '__main__':
    main()

套接字客户端是

<script type="text/javascript">
    var ws = new WebSocket("ws://localhost:8000/websocket");
    ws.onopen = function () {
        ws.send("Hello, world");
    };
    ws.onmessage = function (evt) {
        console.log(evt.data);
    };
</script>

那么,当我刷新我的客户端网页时,如何才能继续从套接字服务器接收数据?

while 循环阻塞了您的整个服务器。您在循环中添加了一个 sleep(1) 调用来暂停循环一秒钟,但是 time.sleep 是一个阻塞函数,因此它也没有任何帮助。

您需要使用 time.sleep 的异步替代方法——Tornado 的 gen.sleep

您还需要将 on_message 函数转换为协程。

示例:

from tornado import gen


async def on_message(...):
    ...
    while True:
        ...
        await gen.sleep(1)
        ...

作为额外的提示,请使用 set() 而不是 list() 来存储您的连接,这样您就不会不小心添加重复的连接。

在这种情况下,您还需要稍微修改一下代码。 set 没有 append 方法,而是有 add 方法。

connections = set()
...
self.connections.add(self) # to add connection
...
self.connections.remove(self) # to remove connetion