python异步回调转为同步并实现超时
问题描述
场景:一个服务端A,一个客户端B,存在一个socket连接。现在写的是客户端B部分,服务端不可控。原来是 B先发送一个包,等待A返回指定内容,B再发送下一个包
def do(): s.send(...) yield 1 s.send(...) yield 2# 接收到数据后的回调def callback(): global f next(f) f=do()next(f)
现在想实现一个timeout,并且实现阻塞。B发送数据后阻塞,直到A返回数据(或5秒内未接受到来自A的返回raise一个错误),请教如何实现?
问题解答
回答1:用 Tornado 的话,写不了几行代码吧。
先作个简单的 Server ,以方便演示:
# -*- coding: utf-8 -*-from tornado.ioloop import IOLoopfrom tornado.tcpserver import TCPServerfrom tornado import genclass Server(TCPServer): @gen.coroutine def handle_stream(self, stream, address):while 1: data = yield stream.read_until(’n’) if data.strip() == ’exit’:stream.close()break if data.strip() == ’5’:IOLoop.current().call_at(IOLoop.current().time() + 5, lambda: stream.write(’ok 5n’)) else:stream.write(’okn’)if __name__ == ’__main__’: Server().listen(8000) IOLoop.current().start()
然后,来实现 Client ,基本逻辑是,超时就关闭连接,然后再重新建立连接:
# -*- coding: utf-8 -*-import functoolsfrom tornado.ioloop import IOLoopfrom tornado.tcpclient import TCPClientfrom tornado import gendef when_error(stream): print ’ERROR’ stream.close() main()@gen.coroutinedef main(): client = TCPClient() stream = yield client.connect(’localhost’, 8000) count = 0 IL = IOLoop.current() while 1:count += 1stream.write(str(count) + ’n’)print count, ’...’timer = IL.call_at(IL.time() + 4, functools.partial(when_error, stream))try: data = yield stream.read_until(’n’)except: breakIL.remove_timeout(timer)print datayield gen.Task(IL.add_timeout, IOLoop.current().time() + 1)if __name__ == ’__main__’: main() IOLoop.current().start()
相关文章:
1. angular.js - ng-grid 和tabset一起用时,grid width默认特别小2. python for循环中的函数只能运行一次?3. css - transform-origin 旋转参考点4. html5 - 在一个页面中 初始了两个swiper 不知道哪里错了 一直不对5. selenium-基于python2-126邮箱登陆问题6. python的文件读写问题?7. 能用Nginx服务小型购物网站的web吗?8. python - pyspider爬取了接近1G的数据,无法导出csv?9. android - 美团筛选处筛选条件停靠+条件点击滑动到顶部。10. 请问是对象还是数组
