成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

Python 的異步 IO:Aiohttp Client 代碼分析

fai1017 / 2695人閱讀

摘要:的異步代碼分析是的一個框架,基于,所以叫。不可避免的,可讀性會比較差。想找教程的話,請移步官方教程,寫得還是挺不錯的。建議不要直接使用,而只把它當(dāng)成的一個樣例。

Python 的異步 IO:Aiohttp Client 代碼分析

Aiohttp 是 Python 的一個 HTTP 框架,基于 asyncio,所以叫 Aiohttp。

我主要是看源碼,想理解它的設(shè)計,所以附上了類圖與時序圖。不可避免的,可讀性會比較差。
想找教程的話,請移步 官方教程,寫得還是挺不錯的。

一個例子

下面這個例子,通過 HTTP GET 列出 GitHub 的 public events:

import asyncio
import aiohttp

async def main():
    async with aiohttp.ClientSession() as session:
        async with session.get("https://api.github.com/events") as resp:
            print(resp.status)
            print(await resp.text())

loop = asyncio.get_event_loop()
loop.run_until_complete(main())

Response 是一個 JSON 格式的文本:

[
  {
    "id": "6888907432",
    "type": "PushEvent",
    "actor": {
      "id": 3956266,
      "login": "sekineh",
      "display_login": "sekineh",
      "gravatar_id": "",
      "url": "https://api.github.com/users/sekineh",
      "avatar_url": "https://avatars.githubusercontent.com/u/3956266?"
    },
    ...
]

ClientSession 是一個 Asynchronous Context Manager,所以搭配 async with 語句一起使用。像下面這樣應(yīng)該也是可以的:

async def main():
    session = aiohttp.ClientSession()
    ...
    await session.close()

不過肯定是不推薦的,就當(dāng)是幫助理解吧。

ClientSession.get() 返回一個 ClientResponse 對象,通過 text() 方法,可以拿到 response 的文本:

print(await resp.text())

當(dāng)然,text() 是一個協(xié)程:

    @asyncio.coroutine
    def text(self, encoding=None, errors="strict"):
        """Read response payload and decode."""
        ...
Connector

ClientSession 依賴 Connector 來創(chuàng)建連接,缺省為 TCPConnector,它繼承自 BaseConnector,此外還有 UnixConnector(應(yīng)該是 Unix Domain Socket)。

Connector 的接口比較簡單,主要提供了 connect() 方法(也是協(xié)程):

    @asyncio.coroutine
    def connect(self, req):
        """Get from pool or create new connection."""
        ...

以及 close() 方法:

    def close(self):
        """Close all opened transports."""
        ...
ConnectionKey

ClientRequest 有個屬性 connection_key

class ClientRequest:
    @property
    def connection_key(self):
        return ConnectionKey(self.host, self.port, self.ssl)

它是一個 namedtuple

ConnectionKey = namedtuple("ConnectionKey", ["host", "port", "ssl"])

host,portssl 三個元素組成,這三個元素可以唯一定義一個連接,所以叫 ConnectionKey。
文章開頭的那個例子中,ConnectionKey 為:

ConnectionKey(host="api.github.com", port=443, ssl=True)
全局函數(shù) request()

Aiohttp 為 Client 程序提供了一個全局函數(shù) request(),用法如下:

async def main():
    resp = await aiohttp.request("GET", "http://python.org/")
    print(resp)
    
loop = asyncio.get_event_loop()
loop.run_until_complete(main())

可見 request() 只是 ClientSession 的一個簡單封裝,其步驟大致為:

創(chuàng)建 TCPConnector

創(chuàng)建 ClientSession;

調(diào)用 ClientSession._request()。

建議不要直接使用 request(),而只把它當(dāng)成 ClientSession 的一個樣例。因為 Aiohttp 官方文檔是 這樣說的:

Don’t create a session per request. Most likely you need a session per application which performs all requests altogether.

A session contains a connection pool inside, connection reusage and keep-alives (both are on by default) may speed up total performance.

即,一個 request 用一個 session,太浪費;通常是一個 application 用一個 session。

一些小問題

我經(jīng)常發(fā)現(xiàn)一個變量,明明可以是局部變量,卻被當(dāng)成了成員變量。

Request 里放了一個 response?

class ClientRequest:
    def send(self, conn):
        ...
        self.response = self.response_class(
            self.method, self.original_url,
            writer=self._writer, ...
        )

        self.response._post_init(self.loop, self._session)
        return self.response

self.responseClientRequest 其他地方并沒有用到,是否可以改成局部變量?

ClientResponse.start() 里的 _protocol 應(yīng)該用局部變量吧?

class ClientResponse:
   @asyncio.coroutine
    def start(self, connection, read_until_eof=False):
        """Start response processing."""
        self._closed = False
        self._protocol = connection.protocol
類圖

時序圖

The End

文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/44503.html

相關(guān)文章

  • 基于Sanic微服務(wù)基礎(chǔ)架構(gòu)

    摘要:在中,官方的異步協(xié)程庫正式成為標(biāo)準(zhǔn)。本項目就是以為基礎(chǔ)搭建的微服務(wù)框架。使用做單元測試,并且使用來避免訪問其他微服務(wù)。跟蹤每一個請求,記錄請求所經(jīng)過的每一個微服務(wù),以鏈條的方式串聯(lián)起來,對分析微服務(wù)的性能瓶頸至關(guān)重要。 介紹 使用python做web開發(fā)面臨的一個最大的問題就是性能,在解決C10K問題上顯的有點吃力。有些異步框架Tornado、Twisted、Gevent 等就是為了解...

    seasonley 評論0 收藏0
  • Python爬蟲入門教程 7-100 蜂鳥網(wǎng)圖片爬取之二

    摘要:蜂鳥網(wǎng)圖片簡介今天玩點新鮮的,使用一個新庫,利用它提高咱爬蟲的爬取速度。上下文不在提示,自行搜索相關(guān)資料即可創(chuàng)建一個對象,然后用該對象去打開網(wǎng)頁??梢赃M行多項操作,比如等代碼中等待網(wǎng)頁數(shù)據(jù)返回創(chuàng)建線程,方法負(fù)責(zé)安排執(zhí)行中的任務(wù)。 1. 蜂鳥網(wǎng)圖片-簡介 今天玩點新鮮的,使用一個新庫 aiohttp ,利用它提高咱爬蟲的爬取速度。 安裝模塊常規(guī)套路 pip install aiohtt...

    z2xy 評論0 收藏0
  • Python爬蟲入門教程 7-100 蜂鳥網(wǎng)圖片爬取之二

    摘要:蜂鳥網(wǎng)圖片簡介今天玩點新鮮的,使用一個新庫,利用它提高咱爬蟲的爬取速度。上下文不在提示,自行搜索相關(guān)資料即可創(chuàng)建一個對象,然后用該對象去打開網(wǎng)頁。可以進行多項操作,比如等代碼中等待網(wǎng)頁數(shù)據(jù)返回創(chuàng)建線程,方法負(fù)責(zé)安排執(zhí)行中的任務(wù)。 1. 蜂鳥網(wǎng)圖片-簡介 今天玩點新鮮的,使用一個新庫 aiohttp ,利用它提高咱爬蟲的爬取速度。 安裝模塊常規(guī)套路 pip install aiohtt...

    hufeng 評論0 收藏0
  • Python爬蟲入門教程 7-100 蜂鳥網(wǎng)圖片爬取之二

    摘要:蜂鳥網(wǎng)圖片簡介今天玩點新鮮的,使用一個新庫,利用它提高咱爬蟲的爬取速度。上下文不在提示,自行搜索相關(guān)資料即可創(chuàng)建一個對象,然后用該對象去打開網(wǎng)頁??梢赃M行多項操作,比如等代碼中等待網(wǎng)頁數(shù)據(jù)返回創(chuàng)建線程,方法負(fù)責(zé)安排執(zhí)行中的任務(wù)。 1. 蜂鳥網(wǎng)圖片-簡介 今天玩點新鮮的,使用一個新庫 aiohttp ,利用它提高咱爬蟲的爬取速度。 安裝模塊常規(guī)套路 pip install aiohtt...

    mcterry 評論0 收藏0

發(fā)表評論

0條評論

最新活動
閱讀需要支付1元查看
<