首页   注册   登录
V2EX = way to explore
V2EX 是一个关于分享和探索的地方
现在注册
已注册用户请  登录
V2EX 提问指南
宝塔
V2EX  ›  问与答

Flask 跑在 Tornado 上,使用 requests 库导致阻塞的问题

  •  
  •   woostundy · 2016-12-05 10:28:28 +08:00 · 2528 次点击
    这是一个创建于 1079 天前的主题,其中的信息可能已经有所发展或是发生改变。

    服务是 Flask 写的,外面套了层 Tornado ,该怎么让 requests 变成非阻塞?

    18 回复  |  直到 2016-12-05 15:55:26 +08:00
        1
    DozySun   2016-12-05 10:36:19 +08:00   ♥ 1
    tornado 换成 gunicorn+gevent
        2
    woostundy   2016-12-05 10:40:27 +08:00
    @DozySun ……改造规模有点大啊
        3
    ericls   2016-12-05 10:58:44 +08:00 via iPhone
    啊? 还可以这么神奇直接改?
        4
    julyclyde   2016-12-05 11:34:54 +08:00   ♥ 1
    你外边套的是 tornado web server 而不是 tornado web framework 吧?那就直接改 gevent monkey patch 就得了
        5
    woostundy   2016-12-05 11:49:17 +08:00
    @julyclyde 嗯,其实改起来也快。
        6
    loalj   2016-12-05 11:51:41 +08:00
    requests 库换成 tornado 自带的 async httpclient 试一试?
        7
    DozySun   2016-12-05 11:52:05 +08:00
    如果不重度依赖 requests 可以用 tornado 的异步 httpclient
    否则就是 concurrent.futures
        8
    woostundy   2016-12-05 11:55:34 +08:00
    @loalj @DozySun
    需要立刻返回 request 的结果,用异步 httpclient 的话,很难写吧?
    回调怎么返回给 flask 的 response
        9
    timonwong   2016-12-05 12:05:24 +08:00
    我不知道在 tornado 上跑 flask 有什么意义,跟用单个进程,什么 threading, gevent, eventlet 都不用跑有多大区别,有没有老司机出来指点一下?
        10
    loalj   2016-12-05 12:08:33 +08:00   ♥ 1
    @woostundy 大概像这样

    ```
    @gen.coroutine
    def async_response():
    http_client = AsyncHTTPClient()
    res = yield http_client.fetch(url)
    return res
    ```
        11
    woostundy   2016-12-05 12:11:25 +08:00
    @loalj 多谢,完了我试试
        12
    Livid   V2EX Moderator   2016-12-05 12:23:06 +08:00 via iPhone   ♥ 1
    耗时可能会超过半秒的函数一律放进 rq 或者 celery 做异步处理。
        13
    woostundy   2016-12-05 12:27:09 +08:00
    @loalj 不过 2.7 里没法在有 yield 的方法里再写 return ,应该是 yield res 吧?
        14
    woostundy   2016-12-05 12:27:51 +08:00
    @Livid 需要即时返回 request 的结果, celery 没法解决吧?
        15
    wwqgtxx   2016-12-05 13:28:10 +08:00 via iPhone
    你这种需求还是用 gunicorn 或者 uwsgi 包装一下吧
        16
    loalj   2016-12-05 14:12:43 +08:00
    @woostundy raise gen.Return(res)
        17
    woostundy   2016-12-05 14:14:51 +08:00
    @loalj 用 yield 也 OK~ 刚试了可行
        18
    cloverstd   2016-12-05 15:55:26 +08:00
    @woostundy

    @gen.coroutine
    def long_time_function():
    task_id = task_function.delay()
    while True:
    gen.sleep(0.1)
    res = get_res_from(task_id)
    if res:
    raise gen.Return(res)
    关于   ·   FAQ   ·   API   ·   我们的愿景   ·   广告投放   ·   感谢   ·   实用小工具   ·   2663 人在线   最高记录 5043   ·     Select Language
    创意工作者们的社区
    World is powered by solitude
    VERSION: 3.9.8.3 · 29ms · UTC 12:37 · PVG 20:37 · LAX 04:37 · JFK 07:37
    ♥ Do have faith in what you're doing.