主頁 > 後端開發 > 700行代碼, 用python實作一個HTTP客戶端

700行代碼, 用python實作一個HTTP客戶端

2020-09-24 08:51:43 後端開發

本文用python在TCP的基礎上實作一個HTTP客戶端, 該客戶端能夠復用TCP連接, 使用HTTP1.1協議. 

一. 創建HTTP請求

  HTTP是基于TCP連接的, 它的請求報文格式如下:

  

  因此, 我們只需要創建一個到服務器的TCP連接, 然后按照上面的格式寫好報文并發給服務器, 就實作了一個HTTP請求.

1. HTTPConnection類

  基于以上的分析, 我們首先定義一個HTTPConnection類來管理連接和請求內容:

class HTTPConnection:
    default_port = 80
    _http_vsn = 11
    _http_vsn_str = 'HTTP/1.1'

    def __init__(self, host: str, port: int = None) -> None:
        self.sock = None
        self._buffer = []
        self.host = host
        self.port = port if port is not None else self.default_port
        self._state = _CS_IDLE
        self._response = None
        self._method = None
        self.block_size = 8192

    def _output(self, s: Union[str, bytes]) -> None:
        if hasattr(s, 'encode'):
            s = s.encode('latin-1')
        self._buffer.append(s)

    def connect(self) -> None:
        self.sock = socket.create_connection((self.host, self.port))

  對于這個HTTPConnection物件, 我們只需要創建TCP連接, 然后按照HTTP協議的格式把請求資料寫入buffer中, 最后把buffer中的資料發送出去就行了.

2. 撰寫請求行

  請求行的內容比較簡單, 就是說明請求方法, 請求路徑和HTTP協議. 使用下面的方法來撰寫一個請求行:

def put_request(self, method: str, url: str) -> None:
    self._method = method

    url = url or '/'

    request = f'{method} {url} {self._http_vsn_str}'
    self._output(request)

3. 添加請求頭

  HTTP請求頭和python的字典類似, 每行都是一個欄位名與值的映射關系. HTTP協議并不要求設定所有合法的請求頭的值, 我們只需要按照需要, 設定特定的請求頭即可. 使用如下代碼添加請求頭:

def put_header(self, header: Union[bytes, str], value: Union[bytes, str, int]) -> None:
    if hasattr(header, 'encode'):
        header = header.encode('ascii')

    if hasattr(value, 'encode'):
        value = value.encode('latin-1')
    elif isinstance(value, int):
        value = str(value).encode('ascii')

    header = header + b': ' + value
    self._output(header)

  此外, 在HTTP請求中, Host請求頭欄位是必須的, 否則網站可能會拒絕回應. 因此, 如果用戶沒有設定這個欄位, 這里就應該主動把它加上去:

def _add_host(self, url: str) -> None:
    # 所有HTTP / 1.1請求報文中必須包含一個Host頭欄位
    # 如果用戶沒給,就呼叫這個函式來生成
    netloc = ''
    if url.startswith('http'):
        nil, netloc, nil, nil, nil = urllib.parse.urlsplit(url)

    if netloc:
        try:
            netloc_enc = netloc.encode('ascii')
        except UnicodeEncodeError:
            netloc_enc = netloc.encode('idna')
        self.put_header('Host', netloc_enc)
    else:
        host = self.host
        port = self.port

        try:
            host_enc = host.encode('ascii')
        except UnicodeEncodeError:
            host_enc = host.encode('idna')

        # 對IPv6的地址進行額外處理
        if host.find(':') >= 0:
            host_enc = b'[' + host_enc + b']'

        if port == self.default_port:
            self.put_header('Host', host_enc)
        else:
            host_enc = host_enc.decode('ascii')
            self.put_header('Host', f'{host_enc}:{port}')

4. 發送請求正文

  我們接受兩種形式的body資料: 一個基于io.IOBase的可讀檔案物件, 或者是一個能通過迭代得到資料的物件. 在傳輸資料之前, 我們首先要確定資料是否采用分塊傳輸:

def request(self, method: str, url: str, headers: dict = None, body: Union[io.IOBase, Iterable] = None,
            encode_chunked: bool = False) -> None:
    ...
    if 'content-length' not in header_names:
        if 'transfer-encoding' not in header_names:
            encode_chunked = False
            content_length = self._get_content_length(body, method)
            if content_length is None:
                if body is not None:
                    # 在這種情況下, body一般是個生成器或者可讀檔案之類的東西,應該分塊傳輸
                    encode_chunked = True
                    self.put_header('Transfer-Encoding', 'chunked')
            else:
                self.put_header('Content-Length', str(content_length))
        else:
            # 如果設定了transfer-encoding,則根據用戶給的encode_chunked引數決定是否分塊
            pass
    else:
        # 只要給了content-length,那么一定不是分塊傳輸
        encode_chunked = False
    ...


@staticmethod
def _get_content_length(body: Union[str, bytes, bytearray, Iterable, io.IOBase], method: str) -> Optional[int]:
    if body is None:
        # PUT,POST,PATCH三個方法默認是有body的
        if method.upper() in _METHODS_EXPECTING_BODY:
            return 0
        else:
            return None

    if hasattr(body, 'read'):
        return None

    try:
        # 對于bytes或者bytearray格式的資料,通過memoryview獲取它的長度
        return memoryview(body).nbytes
    except TypeError:
        pass

    if isinstance(body, str):
        return len(body)

    return None

   在確定了是否分塊之后, 就可以把正文發出去了. 如果body是一個可讀檔案的話, 就呼叫_read_readable方法把它封裝為一個生成器:

def _send_body(self, message_body: Union[str, bytes, bytearray, Iterable, io.IOBase], encode_chunked: bool) -> None:
    if hasattr(message_body, 'read'):
        chunks = self._read_readable(message_body)
    else:
        try:
            memoryview(message_body)
        except TypeError:
            try:
                chunks = iter(message_body)
            except TypeError:
                raise TypeError(
                    f'message_body should be a bytes-like object or an iterable, got {repr(type(message_body))}')
        else:
            # 如果是位元組型別的,通過一次迭代把它發出去
            chunks = (message_body,)

    for chunk in chunks:
        if not chunk:
            continue

        if encode_chunked:
            chunk = f'{len(chunk):X}\r\n'.encode('ascii') + chunk + b'\r\n'
        self.send(chunk)

    if encode_chunked:
        self.send(b'0\r\n\r\n')


def _read_readable(self, readable: io.IOBase) -> Generator[bytes, None, None]:
    need_encode = False
    if isinstance(readable, io.TextIOBase):
        need_encode = True
    while True:
        data_block = readable.read(self.block_size)
        if not data_block:
            break
        if need_encode:
            data_block = data_block.encode('utf-8')
        yield data_block

二. 獲取回應資料

  HTTP回應報文的格式與請求報文大同小異, 它大致是這樣的:

  因此, 我們只要用HTTPConnection的socket物件讀取服務器發送的資料, 然后按照上面的格式對資料進行決議就行了.

1. HTTPResponse類

  我們首先定義一個簡單的HTTPResponse類. 它的屬性大致上就是socket的檔案物件以及一些請求的資訊等等, 呼叫它的begin方法來決議回應行和回應頭的資料, 然后呼叫read方法讀取回應正文:

class HTTPResponse:

    def __init__(self, sock: socket.socket, method: str = None) -> None:
        self.fp = sock.makefile('rb')
        self._method = method
        self.headers = None
        self.version = _UNKNOWN
        self.status = _UNKNOWN
        self.reason = _UNKNOWN
        self.chunked = _UNKNOWN
        self.chunk_left = _UNKNOWN
        self.length = _UNKNOWN
        self.will_close = _UNKNOWN

    def begin(self) -> None:
        ...

    def read(self, amount: int = None) -> bytes:
        ...

2. 決議狀態行

  狀態行的決議比較簡單, 我們只需要讀取回應的第一行資料, 然后把它決議為HTTP協議版本,狀態碼和原因短語三部分就行了:

def _read_status(self) -> Tuple[str, int, str]:
    line = str(self._read_line(), 'latin-1')
    if not line:
        raise RemoteDisconnected('Remote end closed connection without response')
    try:
        version, status, reason = line.split(None, 2)
    except ValueError:
        # reason只是給人看的, 一般和status對應, 所以它有可能不存在
        try:
            version, status = line.split(None, 1)
            reason = ''
        except ValueError:
            version, status, reason = '', '', ''
    if not version.startswith('HTTP/'):
        self._close_conn()
        raise BadStatusLine(line)

    try:
        status = int(status)
        if status < 100 or status > 999:
            raise BadStatusLine(line)
    except ValueError:
        raise BadStatusLine(line)
    return version, status, reason.strip()

  如果狀態碼為100, 則客戶端需要決議多個回應狀態行. 它的原理是這樣的: 在請求資料過大的時候, 有的客戶端會先不發送請求資料, 而是先在header中添加一個Expect: 100-continue, 如果服務器愿意接收資料, 會回傳100的狀態碼, 這時候客戶端再把資料發過去. 因此, 如果讀取到100的狀態碼, 那么后面往往還會收到一個正式的回應資料, 應該繼續讀取回應頭. 這部分的代碼如下:

def begin(self) -> None:
    while True:
        version, status, reason = self._read_status()
        if status != HTTPStatus.CONTINUE:
            break
        # 跳過100狀態碼部分的回應頭
        while True:
            skip = self._read_line().strip()
            if not skip:
                breakself.status = status
    self.reason = reason
    if version in ('HTTP/1.0', 'HTTP/0.9'):
        self.version = 10
    elif version.startswith('HTTP/1.'):
        self.version = 11
    else:
        # HTTP2還沒研究, 這里就不寫了
        raise UnknownProtocol(version)

    ...

3. 決議回應頭

  決議回應頭比回應行還要簡單. 因為每個header欄位占一行, 我們只需要一直呼叫read_line方法讀取欄位, 直到讀完header為止就行了.

def _parse_header(self) -> None:
    headers = {}
    while True:
        line = self._read_line()
        if len(headers) > _MAX_HEADERS:
            raise HTTPException('got more than %d headers' % _MAX_HEADERS)
        if line in _EMPTY_LINE:
            break
        line = line.decode('latin-1')
        i = line.find(':')
        if i == -1:
            raise BadHeaderLine(line)
        # 這里默認沒有重名的情況
        key, value = https://www.cnblogs.com/q1214367903/p/line[:i].lower(), line[i + 1:].strip()
        headers[key] = value
    self.headers = headers

4. 接收回應正文

  在接收回應正文之前, 首先要確定它的傳輸方式和長度:

def _set_chunk(self) -> None:
    transfer_encoding = self.get_header('transfer-encoding')
    if transfer_encoding and transfer_encoding.lower() == 'chunked':
        self.chunked = True
        self.chunk_left = None
    else:
        self.chunked = False


def _set_length(self) -> None:
    # 首先要知道資料是否是分塊傳輸的
    if self.chunked == _UNKNOWN:
        self._set_chunk()

    # 如果狀態碼是1xx或者204(無回應內容)或者304(使用上次快取的內容),則沒有回應正文
    # 如果這是個HEAD請求,那么也不能有回應正文
    if (self.status == HTTPStatus.NO_CONTENT or
            self.status == HTTPStatus.NOT_MODIFIED or
            100 <= self.status < 200 or
            self._method == 'HEAD'):
        self.length = 0
        return

    length = self.get_header('content-length')
    if length and not self.chunked:
        try:
            self.length = int(length)
        except ValueError:
            self.length = None
        else:
            if self.length < 0:
                self.length = None
    else:
        self.length = None

   然后, 我們實作一個read方法, 從body中讀取指定大小的資料:

def read(self, amount: int = None) -> bytes:
    if self.is_closed():
        return b''
    if self._method == 'HEAD':
        self.close()
        return b''
    if amount is None:
        return self._read_all()
    return self._read_amount(amount)

  如果沒有指定需要的資料大小, 就默認讀取所有資料:

def _read_all(self) -> bytes:
    if self.chunked:
        return self._read_all_chunk()
    if self.length is None:
        s = self.fp.read()
    else:
        try:
            s = self._read_bytes(self.length)
        except IncompleteRead:
            self.close()
            raise
        self.length = 0
    self.close()
    return s


def _read_all_chunk(self) -> bytes:
    assert self.chunked != _UNKNOWN
    value = []
    try:
        while True:
            chunk = self._read_chunk()
            if chunk is None:
                break
            value.append(chunk)
        return b''.join(value)
    except IncompleteRead:
        raise IncompleteRead(b''.join(value))


def _read_chunk(self) -> Optional[bytes]:
    try:
        chunk_size = self._read_chunk_size()
    except ValueError:
        raise IncompleteRead(b'')
    if chunk_size == 0:
        self._read_and_discard_trailer()
        self.close()
        return None
    chunk = self._read_bytes(chunk_size)
    # 每塊的結尾會有一個\r\n,這里把它讀掉
    self._read_bytes(2)
    return chunk


def _read_chunk_size(self) -> int:
    line = self._read_line(error_message='chunk size')
    i = line.find(b';')
    if i >= 0:
        line = line[:i]
    try:
        return int(line, 16)
    except ValueError:
        self.close()
        raise


def _read_and_discard_trailer(self) -> None:
    # chunk的尾部可能會掛一些額外的資訊,比如MD5值,過期時間等等,一般會在header中用trailer欄位說明
    # 當chunk讀完之后呼叫這個函式, 這些資訊就先舍棄掉得了
    while True:
        line = self._read_line(error_message='chunk size')
        if line in _EMPTY_LINE:
            break

  否則的話, 就讀取部分資料, 如果正好是分塊資料的話, 就比較復雜了. 簡單來說, 就是用bytearray制造一個所需大小的陣列, 然后依次讀取chunk把資料往里面填, 直到填滿或者沒資料為止.  然后用chunk_left記錄下當前塊剩余的量, 以便下次讀取.

def _read_amount(self, amount: int) -> bytes:
    if self.chunked:
        return self._read_amount_chunk(amount)
    if isinstance(self.length, int) and amount > self.length:
        amount = self.length
    container = bytearray(amount)
    n = self.fp.readinto(container)
    if not n and container:
        # 如果讀不到位元組了,也就可以關了
        self.close()
    elif self.length is not None:
        self.length -= n
        if not self.length:
            self.close()
    return memoryview(container)[:n].tobytes()


def _read_amount_chunk(self, amount: int) -> bytes:
    # 呼叫這個方法,讀取amount大小的chunk型別資料,不足就全部讀取
    assert self.chunked != _UNKNOWN
    total_bytes = 0
    container = bytearray(amount)
    mvb = memoryview(container)
    try:
        while True:
            # mvb可以理解為容器的空的那一部分
            # 這里一直呼叫_full_readinto把資料填進去,讓mvb越來越小,同時記錄填入的量
            # 等沒資料或者當前資料足夠把mvb填滿之后,跳出回圈
            chunk_left = self._get_chunk_left()
            if chunk_left is None:
                break
            if len(mvb) <= chunk_left:
                n = self._full_readinto(mvb)
                self.chunk_left = chunk_left - n
                total_bytes += n
                break
            temp_mvb = mvb[:chunk_left]
            n = self._full_readinto(temp_mvb)
            mvb = mvb[n:]
            total_bytes += n
            self.chunk_left = 0

    except IncompleteRead:
        raise IncompleteRead(bytes(container[:total_bytes]))

    return memoryview(container)[:total_bytes].tobytes()


def _full_readinto(self, container: memoryview) -> int:
    # 回傳讀取的量.如果沒能讀滿,這個方法會報警
    amount = len(container)
    n = self.fp.readinto(container)
    if n < amount:
        raise IncompleteRead(bytes(container[:n]), amount - n)
    return n


def _get_chunk_left(self) -> Optional[int]:
    # 如果當前塊讀了一半,那么直接回傳self.chunk_left就行了
    # 否則,有三種情況
    # 1). chunk_left為None,說明body壓根沒開始讀,于是回傳當前這一整塊的長度
    # 2). chunk_left為0,說明這塊讀完了,于是回傳下一塊的長度
    # 3). body資料讀完了,回傳None,順便做好善后作業
    chunk_left = self.chunk_left
    if not chunk_left:
        if chunk_left == 0:
            # 如果剩余零,說明上一塊已經讀完了,這里把\r\n讀掉
            # 如果是None,就說明chunk壓根沒開始讀
            self._read_bytes(2)
        try:
            chunk_left = self._read_chunk_size()
        except ValueError:
            raise IncompleteRead(b'')
        if chunk_left == 0:
            self._read_and_discard_trailer()
            self.close()
            chunk_left = None
        self.chunk_left = chunk_left
    return chunk_left

三. 復用TCP連接

  HTTP通信本質上是基于TCP連接發送和接收HTTP請求和回應, 因此, 只要TCP連接不斷開, 我們就可以繼續用它進行HTTP請求, 這樣就避免了創建和銷毀TCP連接產生的消耗.

1. 判斷連接是否會斷開

  在下面幾種情況中, 服務端會自動斷開連接:

  • HTTP協議小于1.1且沒有在頭部設定了keep-alive
  • HTTP協議大于等于1.1但是在頭部設定了connection: close
  • 資料沒有分塊傳輸, 也沒有說明資料的長度, 這種情況下, 服務器一般會在發送完成后斷開連接, 讓客戶端知道資料發完了

  根據上面列出來的幾種情況, 通過下面的代碼來判斷連接是否會斷開:

def _check_close(self) -> bool:
    conn = self.get_header('connection')

    if not self.chunked and self.length is None:
        return True

    if self.version == 11:
        if conn and 'close' in conn.lower():
            return True
        return False
    else:
        if self.headers.get('keep-alive'):
            return False

        if conn and 'keep-alive' in conn.lower():
            return False

    return True

2. 正確地關閉HTTPResponse物件

  由于TCP連接的復用, 一個HTTPConnection可以產生多個HTTPResponse物件, 而這些物件在同一個TCP連接上, 會共用這個連接的讀緩沖區. 這就導致, 如果上一個HTTPResponse物件沒有把它的那部分資料讀完, 就會對下一個回應產生影響.

  另一方面來看, 我們也需要及時地關閉與這個TCP關聯的檔案物件來避免占用資源. 因此, 我們定義如下的close方法關閉一個HTTPResponse物件:

def close(self) -> None:
    if self.is_closed():
        return
    fp = self.fp
    self.fp = None
    fp.close()


def is_closed(self) -> bool:
    return self.fp is None

  用戶呼叫HTTPResponse物件的read方法, 把緩沖區資料讀完之后, 就會自動呼叫close方法(具體實作見上一章的第四節: 讀取回應資料這部分). 因此, 在獲取下一個回應資料之前, 我們只需要呼叫這個物件的is_closed方法, 就能判斷讀緩沖區是否已經讀完, 能否繼續接收回應了.

3. HTTP請求的生命周期

  對于HTTP/1.1以及更低版本的HTTP協議來說, 不同的HTTP請求必須按次序進行, 相互之間不能重疊. 基于這個原因, 我們為HTTPConnection物件設定IDLE, REQ_STARTED和REQ_SENT三種狀態, 一個完整的請求應該經歷這幾種狀態:

  根據上面的流程, 對HTTPConnection中對應的方法進行修改:

def get_response(self) -> HTTPResponse:
    if self._response and self._response.is_closed():
        self._response = None
    if self._state != _CS_REQ_SENT or self._response:
        raise ResponseNotReady(self._state)

    response = HTTPResponse(self.sock, method=self._method)

    try:
        try:
            response.begin()
        except ConnectionError:
            self.close()
            raise
        assert response.will_close != _UNKNOWN
        self._state = _CS_IDLE

        if response.will_close:
            self.close()
        else:
            self._response = response

        return response
    except Exception as _:
        response.close()
        raise

def put_request(self, method: str, url: str) -> None:
    # 呼叫這個函式開始新一輪的請求,它負責寫好請求行輸出到快取里面去
    # 呼叫它的前提是當前處于空閑狀態
    # 如果之前的response還在并且已結束,會自動把它消除掉
    if self._response and self._response.is_closed():
        self._response = None

    if self._state == _CS_IDLE:
        self._state = _CS_REQ_STARTED
    else:
        raise CannotSendRequest(self._state)

    ...

def put_header(self, header: Union[bytes, str], value: Union[bytes, str, int]) -> None:
    if self._state != _CS_REQ_STARTED:
        raise CannotSendHeader()

    ...

def end_headers(self, message_body=None, encode_chunked=False) -> None:
    if self._state == _CS_REQ_STARTED:
        self._state = _CS_REQ_SENT
    else:
        raise CannotSendHeader()
    ...

  需要注意的是, 如果第二個請求已經進入到獲取回應的階段了, 而上一個請求的回應還沒關閉, 那么就應該直接報錯, 否則讀取到的會是上一個請求剩余的回應部分資料, 導致決議回應出現問題.

四. 總結

1. 完整代碼

  HTTPConnection的完整代碼如下:

class HTTPConnection:
    default_port = 80
    _http_vsn = 11
    _http_vsn_str = 'HTTP/1.1'

    def __init__(self, host: str, port: int = None) -> None:
        self.sock = None
        self._buffer = []
        self.host = host
        self.port = port if port is not None else self.default_port
        self._state = _CS_IDLE
        self._response = None
        self._method = None
        self.block_size = 8192

    def request(self, method: str, url: str, headers: dict = None, body: Union[io.IOBase, Iterable] = None,
                encode_chunked: bool = False) -> None:
        self.put_request(method, url)
        headers = headers or {}
        header_names = frozenset(k.lower() for k in headers.keys())
        if 'host' not in header_names:
            self._add_host(url)

        if 'content-length' not in header_names:
            if 'transfer-encoding' not in header_names:
                encode_chunked = False
                content_length = self._get_content_length(body, method)
                if content_length is None:
                    if body is not None:
                        encode_chunked = True
                        self.put_header('Transfer-Encoding', 'chunked')
                else:
                    self.put_header('Content-Length', str(content_length))
            else:
                # 如果設定了transfer-encoding,則根據用戶給的encode_chunked引數決定是否分塊
                pass
        else:
            # 只要給了content-length,那么一定不是分塊傳輸
            encode_chunked = False

        for hdr, value in headers.items():
            self.put_header(hdr, value)
        if isinstance(body, str):
            body = _encode(body)
        self.end_headers(body, encode_chunked=encode_chunked)

    def send(self, data: bytes) -> None:
        if self.sock is None:
            self.connect()

        self.sock.sendall(data)

    def get_response(self) -> HTTPResponse:
        if self._response and self._response.is_closed():
            self._response = None
        if self._state != _CS_REQ_SENT or self._response:
            raise ResponseNotReady(self._state)

        response = HTTPResponse(self.sock, method=self._method)

        try:
            try:
                response.begin()
            except ConnectionError:
                self.close()
                raise
            assert response.will_close != _UNKNOWN
            self._state = _CS_IDLE

            if response.will_close:
                self.close()
            else:
                self._response = response

            return response
        except Exception as _:
            response.close()
            raise

    def connect(self) -> None:
        self.sock = socket.create_connection((self.host, self.port))

    def close(self) -> None:
        self._state = _CS_IDLE
        try:
            sock = self.sock
            if sock:
                self.sock = None
                sock.close()
        finally:
            response = self._response
            if response:
                self._response = None
                response.close()

    def put_request(self, method: str, url: str) -> None:
        # 呼叫這個函式開始新一輪的請求,它負責寫好請求行輸出到快取里面去
        # 呼叫它的前提是當前處于空閑狀態
        # 如果之前的response還在并且已結束,會自動把它消除掉
        if self._response and self._response.is_closed():
            self._response = None

        if self._state == _CS_IDLE:
            self._state = _CS_REQ_STARTED
        else:
            raise CannotSendRequest(self._state)

        self._method = method

        url = url or '/'

        request = f'{method} {url} {self._http_vsn_str}'
        self._output(request)

    def put_header(self, header: Union[bytes, str], value: Union[bytes, str, int]) -> None:
        if self._state != _CS_REQ_STARTED:
            raise CannotSendHeader()

        if hasattr(header, 'encode'):
            header = header.encode('ascii')

        if hasattr(value, 'encode'):
            value = value.encode('latin-1')
        elif isinstance(value, int):
            value = str(value).encode('ascii')

        header = header + b': ' + value
        self._output(header)

    def end_headers(self, message_body=None, encode_chunked=False) -> None:
        if self._state == _CS_REQ_STARTED:
            self._state = _CS_REQ_SENT
        else:
            raise CannotSendHeader()
        self._send_output(message_body, encode_chunked=encode_chunked)

    def _add_host(self, url: str) -> None:
        # 所有HTTP / 1.1請求報文中必須包含一個Host頭欄位
        # 如果用戶沒給,就呼叫這個函式來生成
        netloc = ''
        if url.startswith('http'):
            nil, netloc, nil, nil, nil = urlsplit(url)

        if netloc:
            try:
                netloc_enc = netloc.encode('ascii')
            except UnicodeEncodeError:
                netloc_enc = netloc.encode('idna')
            self.put_header('Host', netloc_enc)
        else:
            host = self.host
            port = self.port

            try:
                host_enc = host.encode('ascii')
            except UnicodeEncodeError:
                host_enc = host.encode('idna')

            # 對IPv6的地址進行額外處理
            if host.find(':') >= 0:
                host_enc = b'[' + host_enc + b']'

            if port == self.default_port:
                self.put_header('Host', host_enc)
            else:
                host_enc = host_enc.decode('ascii')
                self.put_header('Host', f'{host_enc}:{port}')

    def _output(self, s: Union[str, bytes]) -> None:
        # 將資料添加到緩沖區
        if hasattr(s, 'encode'):
            s = s.encode('latin-1')
        self._buffer.append(s)

    def _send_output(self, message_body=None, encode_chunked=False) -> None:
        # 發送并清慷訓沖資料.然后,如果有請求正文,就也順便發送

        self._buffer.extend((b'', b''))
        msg = b'\r\n'.join(self._buffer)
        self._buffer.clear()
        self.send(msg)

        if message_body is not None:
            self._send_body(message_body, encode_chunked)

    def _send_body(self, message_body: Union[bytes, str, bytearray, Iterable, io.IOBase], encode_chunked: bool) -> None:
        if hasattr(message_body, 'read'):
            chunks = self._read_readable(message_body)
        else:
            try:
                memoryview(message_body)
            except TypeError:
                try:
                    chunks = iter(message_body)
                except TypeError:
                    raise TypeError(
                        f'message_body should be a bytes-like object or an iterable, got {repr(type(message_body))}')
            else:
                # 如果是位元組型別的,通過一次迭代把它發出去
                chunks = (message_body,)

        for chunk in chunks:
            if not chunk:
                continue

            if encode_chunked:
                chunk = f'{len(chunk):X}\r\n'.encode('ascii') + chunk + b'\r\n'
            self.send(chunk)

        if encode_chunked:
            self.send(b'0\r\n\r\n')

    def _read_readable(self, readable: io.IOBase) -> Generator[bytes, None, None]:
        need_encode = False
        if isinstance(readable, io.TextIOBase):
            need_encode = True
        while True:
            data_block = readable.read(self.block_size)
            if not data_block:
                break
            if need_encode:
                data_block = data_block.encode('utf-8')
            yield data_block

    @staticmethod
    def _get_content_length(body: Union[str, bytes, bytearray, Iterable, io.IOBase], method: str) -> Optional[int]:
        if body is None:
            # PUT,POST,PATCH三個方法默認是有body的
            if method.upper() in _METHODS_EXPECTING_BODY:
                return 0
            else:
                return None

        if hasattr(body, 'read'):
            return None

        try:
            # 對于bytes或者bytearray格式的資料,通過memoryview獲取它的長度
            return memoryview(body).nbytes
        except TypeError:
            pass

        if isinstance(body, str):
            return len(body)

        return None
HTTPConnection

  HTTPResponse的完整代碼如下:

class HTTPResponse:

    def __init__(self, sock: socket.socket, method: str = None) -> None:
        self.fp = sock.makefile('rb')
        self._method = method
        self.headers = None
        self.version = _UNKNOWN
        self.status = _UNKNOWN
        self.reason = _UNKNOWN
        self.chunked = _UNKNOWN
        self.chunk_left = _UNKNOWN
        self.length = _UNKNOWN
        self.will_close = _UNKNOWN

    def begin(self) -> None:
        if self.headers is not None:
            return
        self._parse_status_line()
        self._parse_header()
        self._set_chunk()
        self._set_length()
        self.will_close = self._check_close()

    def _read_line(self, limit: int = _MAX_LINE + 1, error_message: str = '') -> bytes:
        # 注意,這個方法默認不去除line尾部的\r\n
        line = self.fp.readline(limit)
        if len(line) > _MAX_LINE:
            raise LineTooLong(error_message)
        return line

    def _read_bytes(self, amount: int) -> bytes:
        data = self.fp.read(amount)
        if len(data) < amount:
            raise IncompleteRead(data, amount - len(data))
        return data

    def _parse_status_line(self) -> None:
        while True:
            version, status, reason = self._read_status()
            if status != HTTPStatus.CONTINUE:
                break
            while True:
                skip = self._read_line(error_message='header line').strip()
                if not skip:
                    break

        self.status = status
        self.reason = reason
        if version in ('HTTP/1.0', 'HTTP/0.9'):
            self.version = 10
        elif version.startswith('HTTP/1.'):
            self.version = 11
        else:
            raise UnknownProtocol(version)

    def _read_status(self) -> Tuple[str, int, str]:
        line = str(self._read_line(error_message='status line'), 'latin-1')
        if not line:
            raise RemoteDisconnected('Remote end closed connection without response')
        try:
            version, status, reason = line.split(None, 2)
        except ValueError:
            # reason只是給人看的, 和status對應, 所以它有可能不存在
            try:
                version, status = line.split(None, 1)
                reason = ''
            except ValueError:
                version, status, reason = '', '', ''
        if not version.startswith('HTTP/'):
            self.close()
            raise BadStatusLine(line)

        try:
            status = int(status)
            if status < 100 or status > 999:
                raise BadStatusLine(line)
        except ValueError:
            raise BadStatusLine(line)
        return version, status, reason.strip()

    def _parse_header(self) -> None:
        headers = {}
        while True:
            line = self._read_line(error_message='header line')
            if len(headers) > _MAX_HEADERS:
                raise HTTPException('got more than %d headers' % _MAX_HEADERS)
            if line in _EMPTY_LINE:
                break
            line = line.decode('latin-1')
            i = line.find(':')
            if i == -1:
                raise BadHeaderLine(line)
            # 這里默認沒有重名的情況
            key, value = https://www.cnblogs.com/q1214367903/p/line[:i].lower(), line[i + 1:].strip()
            headers[key] = value
        self.headers = headers

    def _set_chunk(self) -> None:
        transfer_encoding = self.get_header('transfer-encoding')
        if transfer_encoding and transfer_encoding.lower() == 'chunked':
            self.chunked = True
            self.chunk_left = None
        else:
            self.chunked = False

    def _set_length(self) -> None:
        # 首先要知道資料是否是分塊傳輸的
        if self.chunked == _UNKNOWN:
            self._set_chunk()

        # 如果狀態碼是1xx或者204(無回應內容)或者304(使用上次快取的內容),則沒有回應正文
        # 如果這是個HEAD請求,那么也不能有回應正文
        assert isinstance(self.status, int)
        if (self.status == HTTPStatus.NO_CONTENT or
                self.status == HTTPStatus.NOT_MODIFIED or
                100 <= self.status < 200 or
                self._method == 'HEAD'):
            self.length = 0
            return

        length = self.get_header('content-length')
        if length and not self.chunked:
            try:
                self.length = int(length)
            except ValueError:
                self.length = None
            else:
                if self.length < 0:
                    self.length = None
        else:
            self.length = None

    def _check_close(self) -> bool:
        conn = self.get_header('connection')

        if not self.chunked and self.length is None:
            return True

        if self.version == 11:
            if conn and 'close' in conn.lower():
                return True
            return False
        else:
            if self.headers.get('keep-alive'):
                return False

            if conn and 'keep-alive' in conn.lower():
                return False

        return True

    def close(self) -> None:
        if self.is_closed():
            return
        fp = self.fp
        self.fp = None
        fp.close()

    def is_closed(self) -> bool:
        return self.fp is None

    def read(self, amount: int = None) -> bytes:
        if self.is_closed():
            return b''
        if self._method == 'HEAD':
            self.close()
            return b''
        if amount is None:
            return self._read_all()
        print(amount, amount is None)
        return self._read_amount(amount)

    def _read_all(self) -> bytes:
        if self.chunked:
            return self._read_all_chunk()
        if self.length is None:
            s = self.fp.read()
        else:
            try:
                s = self._read_bytes(self.length)
            except IncompleteRead:
                self.close()
                raise
            self.length = 0
        self.close()
        return s

    def _read_all_chunk(self) -> bytes:
        assert self.chunked != _UNKNOWN
        value = []
        try:
            while True:
                chunk = self._read_chunk()
                if chunk is None:
                    break
                value.append(chunk)
            return b''.join(value)
        except IncompleteRead:
            raise IncompleteRead(b''.join(value))

    def _read_chunk(self) -> Optional[bytes]:
        try:
            chunk_size = self._read_chunk_size()
        except ValueError:
            raise IncompleteRead(b'')
        if chunk_size == 0:
            self._read_and_discard_trailer()
            self.close()
            return None
        chunk = self._read_bytes(chunk_size)
        # 每塊的結尾會有一個\r\n,這里把它讀掉
        self._read_bytes(2)
        return chunk

    def _read_chunk_size(self) -> int:
        line = self._read_line(error_message='chunk size')
        i = line.find(b';')
        if i >= 0:
            line = line[:i]
        try:
            return int(line, 16)
        except ValueError:
            self.close()
            raise

    def _read_and_discard_trailer(self) -> None:
        # chunk的尾部可能會掛一些額外的資訊,比如MD5值,過期時間等等,一般會在header中用trailer欄位說明
        # 當chunk讀完之后呼叫這個函式, 這些資訊就先舍棄掉得了
        while True:
            line = self._read_line(error_message='chunk size')
            if line in _EMPTY_LINE:
                break

    def _read_amount(self, amount: int) -> bytes:
        if self.chunked:
            return self._read_amount_chunk(amount)
        if isinstance(self.length, int) and amount > self.length:
            amount = self.length
        container = bytearray(amount)
        n = self.fp.readinto(container)
        if not n and container:
            # 如果讀不到位元組了,也就可以關了
            self.close()
        elif self.length is not None:
            self.length -= n
            if not self.length:
                self.close()
        return memoryview(container)[:n].tobytes()

    def _read_amount_chunk(self, amount: int) -> bytes:
        # 呼叫這個方法,讀取amount大小的chunk型別資料,不足就全部讀取
        assert self.chunked != _UNKNOWN
        total_bytes = 0
        container = bytearray(amount)
        mvb = memoryview(container)
        try:
            while True:
                # mvb可以理解為容器的空的那一部分
                # 這里一直呼叫_full_readinto把資料填進去,讓mvb越來越小,同時記錄填入的量
                # 等沒資料或者當前資料足夠把mvb填滿之后,跳出回圈
                chunk_left = self._get_chunk_left()
                if chunk_left is None:
                    break
                if len(mvb) <= chunk_left:
                    n = self._full_readinto(mvb)
                    self.chunk_left = chunk_left - n
                    total_bytes += n
                    break
                temp_mvb = mvb[:chunk_left]
                n = self._full_readinto(temp_mvb)
                mvb = mvb[n:]
                total_bytes += n
                self.chunk_left = 0

        except IncompleteRead:
            raise IncompleteRead(bytes(container[:total_bytes]))

        return memoryview(container)[:total_bytes].tobytes()

    def _full_readinto(self, container: memoryview) -> int:
        # 回傳讀取的量.如果沒能讀滿,這個方法會報警
        amount = len(container)
        n = self.fp.readinto(container)
        if n < amount:
            raise IncompleteRead(bytes(container[:n]), amount - n)
        return n

    def _get_chunk_left(self) -> Optional[int]:
        # 如果當前塊讀了一半,那么直接回傳self.chunk_left就行了
        # 否則,有三種情況
        # 1). chunk_left為None,說明body壓根沒開始讀,于是回傳當前這一整塊的長度
        # 2). chunk_left為0,說明這塊讀完了,于是回傳下一塊的長度
        # 3). body資料讀完了,回傳None,順便做好善后作業
        chunk_left = self.chunk_left
        if not chunk_left:
            if chunk_left == 0:
                # 如果剩余零,說明上一塊已經讀完了,這里把\r\n讀掉
                # 如果是None,就說明chunk壓根沒開始讀
                self._read_bytes(2)
            try:
                chunk_left = self._read_chunk_size()
            except ValueError:
                raise IncompleteRead(b'')
            if chunk_left == 0:
                self._read_and_discard_trailer()
                self.close()
                chunk_left = None
            self.chunk_left = chunk_left
        return chunk_left

    def get_header(self, name, default: str = None) -> Optional[str]:
        if self.headers is None:
            raise ResponseNotReady()
        return self.headers.get(name, default)

    @property
    def info(self) -> str:
        return repr(self.headers)
HTTPResponse

  這兩個類應該放到同一個py檔案中, 同時這個檔案內還有其他一些輔助性質的代碼:

import io
import socket
from typing import Generator, Iterable, Optional, Tuple, Union
from urllib.parse import urlsplit

_CS_IDLE = 'Idle'
_CS_REQ_STARTED = 'Request-started'
_CS_REQ_SENT = 'Request-sent'

_METHODS_EXPECTING_BODY = {'PATCH', 'POST', 'PUT'}
_UNKNOWN = 'UNKNOWN'

_MAX_LINE = 65536
_MAX_HEADERS = 100

_EMPTY_LINE = (b'\r\n', b'\n', b'')


class HTTPStatus:
    CONTINUE = 100
    SWITCHING_PROTOCOLS = 101
    PROCESSING = 102
    OK = 200
    CREATED = 201
    ACCEPTED = 202
    NON_AUTHORITATIVE_INFORMATION = 203
    NO_CONTENT = 204
    RESET_CONTENT = 205
    PARTIAL_CONTENT = 206
    MULTI_STATUS = 207
    ALREADY_REPORTED = 208
    IM_USED = 226
    MULTIPLE_CHOICES = 300
    MOVED_PERMANENTLY = 301
    FOUND = 302
    SEE_OTHER = 303
    NOT_MODIFIED = 304
    USE_PROXY = 305
    TEMPORARY_REDIRECT = 307
    PERMANENT_REDIRECT = 308
    BAD_REQUEST = 400
    UNAUTHORIZED = 401
    PAYMENT_REQUIRED = 402
    FORBIDDEN = 403
    NOT_FOUND = 404
    METHOD_NOT_ALLOWED = 405
    NOT_ACCEPTABLE = 406
    PROXY_AUTHENTICATION_REQUIRED = 407
    REQUEST_TIMEOUT = 408
    CONFLICT = 409
    GONE = 410
    LENGTH_REQUIRED = 411
    PRECONDITION_FAILED = 412
    REQUEST_ENTITY_TOO_LARGE = 413
    REQUEST_URI_TOO_LONG = 414
    UNSUPPORTED_MEDIA_TYPE = 415
    REQUESTED_RANGE_NOT_SATISFIABLE = 416
    EXPECTATION_FAILED = 417
    MISDIRECTED_REQUEST = 421
    UNPROCESSABLE_ENTITY = 422
    LOCKED = 423
    FAILED_DEPENDENCY = 424
    UPGRADE_REQUIRED = 426
    PRECONDITION_REQUIRED = 428
    TOO_MANY_REQUESTS = 429
    REQUEST_HEADER_FIELDS_TOO_LARGE = 431
    UNAVAILABLE_FOR_LEGAL_REASONS = 451
    INTERNAL_SERVER_ERROR = 500
    NOT_IMPLEMENTED = 501
    BAD_GATEWAY = 502
    SERVICE_UNAVAILABLE = 503
    GATEWAY_TIMEOUT = 504
    HTTP_VERSION_NOT_SUPPORTED = 505
    VARIANT_ALSO_NEGOTIATES = 506
    INSUFFICIENT_STORAGE = 507
    LOOP_DETECTED = 508
    NOT_EXTENDED = 510
    NETWORK_AUTHENTICATION_REQUIRED = 511


class HTTPResponse:
    ...


class HTTPConnection:
    ...


def _encode(data: str, encoding: str = 'latin-1', name: str = 'data') -> bytes:
    # 給請求正文等不知道能怎么轉碼的東西轉碼時用這個,默認使用latin-1編碼
    # 它的好處是,轉碼失敗后能拋出詳細的錯誤資訊,一目了然
    try:
        return data.encode(encoding)
    except UnicodeEncodeError as err:
        raise UnicodeEncodeError(
            err.encoding,
            err.object,
            err.start,
            err.end,
            "{} ({:.20!r}) is not valid {}. Use {}.encode('utf-8') if you want to send it encoded in UTF-8.".format(
                name.title(), data[err.start:err.end], encoding, name)
        ) from None


class HTTPException(Exception):
    pass


class ImproperConnectionState(HTTPException):
    pass


class CannotSendRequest(ImproperConnectionState):
    pass


class CannotSendHeader(ImproperConnectionState):
    pass


class CannotCloseStream(ImproperConnectionState):
    pass


class ResponseNotReady(ImproperConnectionState):
    pass


class LineTooLong(HTTPException):
    def __init__(self, line_type):
        HTTPException.__init__(self, 'got more than %d bytes when reading %s'
                               % (_MAX_LINE, line_type))


class BadStatusLine(HTTPException):
    def __init__(self, line):
        if not line:
            line = repr(line)
        self.args = line,
        self.line = line


class BadHeaderLine(HTTPException):
    def __init__(self, line):
        if not line:
            line = repr(line)
        self.args = line,
        self.line = line


class RemoteDisconnected(ConnectionResetError, BadStatusLine):
    def __init__(self, *args, **kwargs):
        BadStatusLine.__init__(self, '')
        ConnectionResetError.__init__(self, *args, **kwargs)


class UnknownProtocol(HTTPException):
    def __init__(self, version):
        self.args = version,
        self.version = version


class UnknownTransferEncoding(HTTPException):
    pass


class IncompleteRead(HTTPException):
    def __init__(self, partial, expected=None):
        self.args = partial,
        self.partial = partial
        self.expected = expected

    def __repr__(self):
        if self.expected is not None:
            e = f', {self.expected} more expected'
        else:
            e = ''
        return f'{self.__class__.__name__}({len(self.partial)} bytes read{e})'

    __str__ = object.__str__
client.py

2. 需要注意的點

  總的來說, 本文的內容不算復雜, 畢竟HTTP屬于不難理解, 但知識點很多很雜的型別. 這里把本文中一些需要注意的點總結一下:

  • 請求和回應資料的結構大致相同, 都是狀態行+頭部+正文, 狀態行和頭部的每個欄位都用一個\r\n分割, 與正文之間用兩個分割;
  • 狀態行是必須的, 請求頭則最少需要host這個欄位, 同時為了大家的方便, 你最好也設定一下Accept-encoding和Accept來限制服務器回傳給你的資料內容和格式;
  • 正文不是必須的, 特別是對于除了3P(PATCH, POST, PUT)之外的方法來說. 如果你有正文, 你最好在header中使用Content-Length說明正文的長度, 如果是分塊發送, 則使用Transfer-Encoding欄位說明;
  • 如果對正文使用分塊傳輸, 每塊的格式是: 16進制的資料長度+\r\n+資料+\r\n, 使用0\r\n\r\n來收尾. 收尾之后, 你還可以放一個trailer, 里面放資料的MD5值或者過期時間什么的, 這時候最好在header中設定trailer欄位;
  • 在一個請求的生命周期完成后, TCP連接是否會斷開取決于三點: 回應資料的HTTP版本, 回應頭中的Connection和Keep-Alive欄位, 是否知道回應正文的長度;
  • 最最重要的一點, HTTP協議只是一個約定而非限制, 這就和礦泉水的建議零售價差不多, 你可以選擇遵守, 也可以不遵守, 后果自負. 

3. 結果測驗

  首先, 我們用tornado寫一個簡單的服務器, 它會顯示客戶端的地址和介面;

import tornado.web
import tornado.ioloop

class IndexHandler(tornado.web.RequestHandler):

    def get(self) -> None:
        print(f'new connection from {self.request.connection.context.address}')
        self.write('hello world')


app = tornado.web.Application([(r'/', IndexHandler)])
app.listen(8888)
tornado.ioloop.IOLoop.current().start()

  然后, 使用我們剛寫好的客戶端進行測驗:

from client import HTTPConnection


def fetch(conn: HTTPConnection, url: str = '') -> None:
    conn.request('GET', url)
    res = conn.get_response()
    print(res.read())


connection = HTTPConnection('127.0.0.1', 8888)
for i in range(10):
    fetch(connection)

  結果如下:

轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/117834.html

標籤:Python

上一篇:pickle存盤和讀取資料

下一篇:python 函式進階,遞回,匿名,內置

標籤雲
其他(157675) Python(38076) JavaScript(25376) Java(17977) C(15215) 區塊鏈(8255) C#(7972) AI(7469) 爪哇(7425) MySQL(7132) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5869) 数组(5741) R(5409) Linux(5327) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4554) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2429) ASP.NET(2402) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) 功能(1967) .NET技术(1958) Web開發(1951) python-3.x(1918) HtmlCss(1915) 弹簧靴(1913) C++(1909) xml(1889) PostgreSQL(1872) .NETCore(1853) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • 【C++】Microsoft C++、C 和匯編程式檔案

    ......

    uj5u.com 2020-09-10 00:57:23 more
  • 例外宣告

    相比于斷言適用于排除邏輯上不可能存在的狀態,例外通常是用于邏輯上可能發生的錯誤。 例外宣告 Item 1:當函式不可能拋出例外或不能接受拋出例外時,使用noexcept 理由 如果不打算拋出例外的話,程式就會認為無法處理這種錯誤,并且應當盡早終止,如此可以有效地阻止例外的傳播與擴散。 示例 //不可 ......

    uj5u.com 2020-09-10 00:57:27 more
  • Codeforces 1400E Clear the Multiset(貪心 + 分治)

    鏈接:https://codeforces.com/problemset/problem/1400/E 來源:Codeforces 思路:給你一個陣列,現在你可以進行兩種操作,操作1:將一段沒有 0 的區間進行減一的操作,操作2:將 i 位置上的元素歸零。最終問:將這個陣列的全部元素歸零后操作的最少 ......

    uj5u.com 2020-09-10 00:57:30 more
  • UVA11610 【Reverse Prime】

    本人看到此題沒有翻譯,就附帶了一個自己的翻譯版本 思考 這一題,它的第一個要求是找出所有 $7$ 位反向質數及其質因數的個數。 我們應該需要質數篩篩選1~$10^{7}$的所有數,這里就不慢慢介紹了。但是,重讀題,我們突然發現反向質數都是 $7$ 位,而將它反過來后的數字卻是 $6$ 位數,這就說明 ......

    uj5u.com 2020-09-10 00:57:36 more
  • 統計區間素數數量

    1 #pragma GCC optimize(2) 2 #include <bits/stdc++.h> 3 using namespace std; 4 bool isprime[1000000010]; 5 vector<int> prime; 6 inline int getlist(int ......

    uj5u.com 2020-09-10 00:57:47 more
  • C/C++編程筆記:C++中的 const 變數詳解,教你正確認識const用法

    1、C中的const 1、區域const變數存放在堆疊區中,會分配記憶體(也就是說可以通過地址間接修改變數的值)。測驗代碼如下: 運行結果: 2、全域const變數存放在只讀資料段(不能通過地址修改,會發生寫入錯誤), 默認為外部聯編,可以給其他源檔案使用(需要用extern關鍵字修飾) 運行結果: ......

    uj5u.com 2020-09-10 00:58:04 more
  • 【C++犯錯記錄】VS2019 MFC添加資源不懂如何修改資源宏ID

    1. 首先在資源視圖中,添加資源 2. 點擊新添加的資源,復制自動生成的ID 3. 在解決方案資源管理器中找到Resource.h檔案,編輯,使用整個專案搜索和替換的方式快速替換 宏宣告 4. Ctrl+Shift+F 全域搜索,點擊查找全部,然后逐個替換 5. 為什么使用搜索替換而不使用屬性視窗直 ......

    uj5u.com 2020-09-10 00:59:11 more
  • 【C++犯錯記錄】VS2019 MFC不懂的批量添加資源

    1. 打開資源頭檔案Resource.h,在其中預先定義好宏 ID(不清楚其實ID值應該設定多少,可以先新建一個相同的資源項,再在這個資源的ID值的基礎上遞增即可) 2. 在資源視圖中選中專案資源,按F7編輯資源檔案,按 ID 型別 相對路徑的形式添加 資源。(別忘了先把檔案拷貝到專案中的res檔案 ......

    uj5u.com 2020-09-10 01:00:19 more
  • C/C++編程筆記:關于C++的參考型別,專供新手入門使用

    今天要講的是C++中我最喜歡的一個用法——參考,也叫別名。 參考就是給一個變數名取一個變數名,方便我們間接地使用這個變數。我們可以給一個變數創建N個參考,這N + 1個變數共享了同一塊記憶體區域。(參考型別的變數會占用記憶體空間,占用的記憶體空間的大小和指標型別的大小是相同的。雖然參考是一個物件的別名,但 ......

    uj5u.com 2020-09-10 01:00:22 more
  • 【C/C++編程筆記】從頭開始學習C ++:初學者完整指南

    眾所周知,C ++的學習曲線陡峭,但是花時間學習這種語言將為您的職業帶來奇跡,并使您與其他開發人員區分開。您會更輕松地學習新語言,形成真正的解決問題的技能,并在編程的基礎上打下堅實的基礎。 C ++將幫助您養成良好的編程習慣(即清晰一致的編碼風格,在撰寫代碼時注釋代碼,并限制類內部的可見性),并且由 ......

    uj5u.com 2020-09-10 01:00:41 more
最新发布
  • Rust中的智能指標:Box<T> Rc<T> Arc<T> Cell<T> RefCell<T> Weak

    Rust中的智能指標是什么 智能指標(smart pointers)是一類資料結構,是擁有資料所有權和額外功能的指標。是指標的進一步發展 指標(pointer)是一個包含記憶體地址的變數的通用概念。這個地址參考,或 ” 指向”(points at)一些其 他資料 。參考以 & 符號為標志并借用了他們所 ......

    uj5u.com 2023-04-20 07:24:10 more
  • Java的值傳遞和參考傳遞

    值傳遞不會改變本身,參考傳遞(如果傳遞的值需要實體化到堆里)如果發生修改了會改變本身。 1.基本資料型別都是值傳遞 package com.example.basic; public class Test { public static void main(String[] args) { int ......

    uj5u.com 2023-04-20 07:24:04 more
  • [2]SpinalHDL教程——Scala簡單入門

    第一個 Scala 程式 shell里面輸入 $ scala scala> 1 + 1 res0: Int = 2 scala> println("Hello World!") Hello World! 檔案形式 object HelloWorld { /* 這是我的第一個 Scala 程式 * 以 ......

    uj5u.com 2023-04-20 07:23:58 more
  • 理解函式指標和回呼函式

    理解 函式指標 指向函式的指標。比如: 理解函式指標的偽代碼 void (*p)(int type, char *data); // 定義一個函式指標p void func(int type, char *data); // 宣告一個函式func p = func; // 將指標p指向函式func ......

    uj5u.com 2023-04-20 07:23:52 more
  • Django筆記二十五之資料庫函式之日期函式

    本文首發于公眾號:Hunter后端 原文鏈接:Django筆記二十五之資料庫函式之日期函式 日期函式主要介紹兩個大類,Extract() 和 Trunc() Extract() 函式作用是提取日期,比如我們可以提取一個日期欄位的年份,月份,日等資料 Trunc() 的作用則是截取,比如 2022-0 ......

    uj5u.com 2023-04-20 07:23:45 more
  • 一天吃透JVM面試八股文

    什么是JVM? JVM,全稱Java Virtual Machine(Java虛擬機),是通過在實際的計算機上仿真模擬各種計算機功能來實作的。由一套位元組碼指令集、一組暫存器、一個堆疊、一個垃圾回收堆和一個存盤方法域等組成。JVM屏蔽了與作業系統平臺相關的資訊,使得Java程式只需要生成在Java虛擬機 ......

    uj5u.com 2023-04-20 07:23:31 more
  • 使用Java接入小程式訂閱訊息!

    更新完微信服務號的模板訊息之后,我又趕緊把微信小程式的訂閱訊息給實作了!之前我一直以為微信小程式也是要企業才能申請,沒想到小程式個人就能申請。 訊息推送平臺🔥推送下發【郵件】【短信】【微信服務號】【微信小程式】【企業微信】【釘釘】等訊息型別。 https://gitee.com/zhongfuch ......

    uj5u.com 2023-04-20 07:22:59 more
  • java -- 緩沖流、轉換流、序列化流

    緩沖流 緩沖流, 也叫高效流, 按照資料型別分類: 位元組緩沖流:BufferedInputStream,BufferedOutputStream 字符緩沖流:BufferedReader,BufferedWriter 緩沖流的基本原理,是在創建流物件時,會創建一個內置的默認大小的緩沖區陣列,通過緩沖 ......

    uj5u.com 2023-04-20 07:22:49 more
  • Java-SpringBoot-Range請求頭設定實作視頻分段傳輸

    老實說,人太懶了,現在基本都不喜歡寫筆記了,但是網上有關Range請求頭的文章都太水了 下面是抄的一段StackOverflow的代碼...自己大修改過的,寫的注釋挺全的,應該直接看得懂,就不解釋了 寫的不好...只是希望能給視頻網站開發的新手一點點幫助吧. 業務場景:視頻分段傳輸、視頻多段傳輸(理 ......

    uj5u.com 2023-04-20 07:22:42 more
  • Windows 10開發教程_編程入門自學教程_菜鳥教程-免費教程分享

    教程簡介 Windows 10開發入門教程 - 從簡單的步驟了解Windows 10開發,從基本到高級概念,包括簡介,UWP,第一個應用程式,商店,XAML控制元件,資料系結,XAML性能,自適應設計,自適應UI,自適應代碼,檔案管理,SQLite資料庫,應用程式到應用程式通信,應用程式本地化,應用程式 ......

    uj5u.com 2023-04-20 07:22:35 more