網路編程之tcp協議以及粘包問題

来源:https://www.cnblogs.com/yafeng666/archive/2019/12/05/11991720.html
-Advertisement-
Play Games

網路編程tcp協議與socket以及單例的補充 一、單例補充 實現單列的幾種方式 二、tcp協議以及粘包問題 理論知識 三、socket套接字 socket套接字初級版本 演示 socket套接字升級版本 socket套接字高級版本 socket套接字終級版本 四、粘包問題 粘包問題的出現以及幾種情 ...


網路編程tcp協議與socket以及單例的補充

一、單例補充

  • 實現單列的幾種方式
#方式一:classmethod

# class Singleton:
#
#     __instance = None
#
#     @classmethod
#     def singleton(cls):
#
#         if not cls.__instance:
#             cls.__instance = cls()
#
#         return cls.__instance
#
# obj1 = Singleton.singleton()
# obj2 = Singleton.singleton()
# print(obj1)
# print(obj2)
# <__main__.Singleton object at 0x000002127F230D08>
# <__main__.Singleton object at 0x000002127F230D08>


#方式二: __new__
# class Singleton:
#
#     __instance = None
#
#     def __new__(cls, *args, **kwargs):
#
#         if not cls.__instance:
#             cls.__instance = object.__new__(cls)
#
#         return cls.__instance
#
# obj1 = Singleton()
# obj2 = Singleton()
# print(obj1)
# print(obj2)
# <__main__.Singleton object at 0x00000257AAE10A88>
# <__main__.Singleton object at 0x00000257AAE10A88>


#方式三:裝飾器
# def singleton(cls):  #cls---> Father
#
#     __instance = {}
#
#     def inner(*args, **kwargs):
#         if cls not in __instance:
#             obj = cls(*args, **kwargs)
#             __instance[cls] = obj
#
#
#         return __instance[cls]
#     return inner
#
# @singleton
# class Father:
#     pass
#
# print(Father())
# print(Father())
# <__main__.Father object at 0x000001F17EB21548>
# <__main__.Father object at 0x000001F17EB21548>



#方式四
先定義一個Singletoncls的py文件內容如下:
class Singletoncls:
    pass

obj = Singletoncls()
# from Singletoncls import obj
# print(obj)
# from Singletoncls import obj
# print(obj)
# from Singletoncls import obj
# print(obj)
# <Singletoncls.Singletoncls object at 0x00000249CD25BE48>
# <Singletoncls.Singletoncls object at 0x00000249CD25BE48>
# <Singletoncls.Singletoncls object at 0x00000249CD25BE48>

# 方式五:元類

二、tcp協議以及粘包問題

  • 理論知識
傳輸層:
    -tcp協議
    -udp協議
    埠(port):標識一臺電腦上的某一個軟體。
        -0-1024:禁止使用,因為是操作系統在用
        -8000---->以後接著用

    - 以下的一些軟體的固定埠不要碰:
        django:8000
        mysql:3306
        redis:6379
        flask:5000
        tomcat:8080
        mongodb:27017
        ...
    要想要傳輸數據,必須建立雙向通道


1、tcp協議:三次握手,四次揮手
    -tcp協議建立雙向通道

    - 三次握手,鍵連接:
        1:客戶端向服務端發送建立連接的請求
        2:服務端返回收到的請求信息給客戶端,並且發送往客戶端建立的請求
        3:客戶端接收到服務端的請求,返回請求成功給服務端,完成雙向連接


    - 反饋機制:
        客戶端往服務端發送請求,服務端必須返迴響應
        告訴客戶端收到請求了,並且將服務端的數據一併返回給客戶端
        c--->s:一次請求,必須有一次響應

        - 缺點:
            - 洪水攻擊:
                指的是通過偽造大量的請求,往對方伺服器發送請求
                導致對方伺服器跟不上,以至於癱瘓。
                Linux系統有個參數可以限制

            - 半連接池listen:限制用戶在同一時間段內的訪問數量

    - 四次揮手,斷開連接
        1:客戶端向服務端發送斷開連接的請求
        2:服務端返回收到請求的信息給客戶端
        3:服務端確認所有的數據發送完成以後,再發送同意斷開連接的請求給客戶端
        4:客戶端返回收到斷開連接的請求,給服務端

2、socket套接字通信:
    - 什麼是socket?
        socket是一個模塊,又稱套接字,用來封裝 互聯網協議(應用層以下的層)

    - 為什麼要有socket?
        socket可以實現 互聯網協議應用層已下的層的工作
        - 提高開發效率

    - 怎麼使用socket?
        import socket
        寫socket套接字:
            client
            sever

3、粘包問題
    -1)問題:無法確認對方發送過來的數據的大小

    -2)問題:在發送數據間隔短並且數據量小的情況下,會將所有數據一次性發送

    解決:確認對方數據的大小

4、解決粘包問題(struct模塊)
    -無論那一段發送數據
        -客戶端
            -1)先製作報頭,併發送(struct)
            -2)發送真實數據

        服務端
            -1)接收報頭,並解包獲取 真實的數據長度
            -2)根據真實數據長度 接收真實數據
            recv(真實數據長度)

三、socket套接字

  • socket套接字初級版本
  • 演示
-sever:
    
'''
先啟動套接服務端
'''

import socket

#買手機
sever = socket.socket()

#綁定手機卡
sever.bind(
    ('127.0.0.1', 9876)
)

#半連接池
sever.listen(5)  #最多同時五個人坐椅子,實際上==6

print(
    '服務端正在運行...'
)

#等待電話接入
#coon:指的是服務端往客戶端的管道
coon, addr = sever.accept()

#接聽對方講話的內容
#data客戶端發送過來的消息
data = coon.recv(1024) #一次性可接受1024bytes的數據
print(data)

>>>服務端正在運行...
>>>b'hello'

#掛電話
coon.close()


-client:
'''
先啟動服務端後再啟動客戶端
'''

import socket

#買手機
client = socket.socket()

#撥打電話
client.connect(
    ('127.0.0.1', 9876)
)

print('客戶端正在運行...')

#必鬚髮送bytes類型的數據
#開始講話
client.send(b'hello')
# 或client.send('hello'.encode('utf_8'))
>>>客戶端正在運行...
  • socket套接字升級版本
- sever

'''
註意:
    客戶端法送一次,我服務端就得先接受一次,之後才可以再向客戶端發送消息
'''

import socket

#買手機
sever = socket.socket()

#綁定手機卡
#裡面綁定的是一個元祖
sever.bind(
    ('127.0.0.1', 9876)
)


#半連接池
sever.listen(5)

print('服務端正在等待服務...')

#等待電話接入
#coon:指的是服務端往客戶端的管道

coon, addr = sever.accept()

#接聽對方講話的內容
data = coon.recv(1024)
print(data)

#服務端往客戶端發送消息
coon.send(b'hi i am sever')

#掛電話
coon.close()

>>>服務端正在等待服務...
>>>b'hello i am client...'


- client
'''
啟動服務端後再啟動客戶端
'''
import socket

#買手機
client = socket.socket()

#撥號
client.connect(
    ('127.0.0.1', 9876)
)

print('客戶端正在發送請求...')

client.send(b'hello i am client...')

#接收服務端的請求
data = client.recv(1024)

print(data)

client.close()
>>>客戶端正在發送請求...
>>>b'hi i am sever'
  • socket套接字高級版本
-sever

import socket

#買手機
sever = socket.socket()


#綁定手機卡
sever.bind(
    ('127.0.0.1', 9876)
)

#半連接池
sever.listen(5)

print('服務端正在運行...')

#等待電話接入
coon, addr = sever.accept()

while True:
    #接收對方講話內容
    #data客戶端發送過來的消息
    data = coon.recv(1024)
    if len(data) == 0:
        break
    if data.decode('utf-8') == 'q':
        break
    print(data.decode('utf-8'))

    send_data = input('服務端...')

    coon.send(send_data.encode('utf-8'))

#掛電話
coon.close()
服務端正在運行...
服務端...你好啊亞峰

-client

import socket

#買手機
client = socket.socket()

#撥打電話
client.connect(
    ('127.0.0.1', 9876)
)

print('客戶端正在發送請求...')

while True:
    send_data = input('客戶端>>>:').strip()
    client.send(send_data.encode('utf-8'))
    data = client.recv(1024)
    if data.decode('utf-8') == 'q':
        break
    if len(data) == 0:
        break

    print(data.decode('utf-8'))

client.close()
>>>客戶端正在發送請求...
>>>客戶端>>>:你好啊熱巴
>>>好啊亞峰
  • socket套接字終級版本
- sever


import socket

#買手機
sever = socket.socket()

#綁定手機卡
sever.bind(
    ('127.0.0.1', 9876)
)

#半連接池
sever.listen(5)

print('服務端正在服務...')

#迴圈實現可接受多個用戶訪問
while True:

    coon, addr = sever.accept()
    print(addr)

    #迴圈實現通信
    while True:
        try:
            #監聽代碼是否有異常出現
            #接聽對方講話的內容
            #data客戶端發送過來的消息
            data = coon.recv(1024)

            if len(data) == 0:
                break

            if data.decode('utf-8') =='q':
                break

            print(data.decode('utf-8'))

            send_data = input('服務端>>>...')

            #服務端向客戶端發送消息
            coon.send(send_data.encode('utf-8'))

        except Exception as e:
            print(e)
            break

    #掛電話
    coon.close()

>>>
服務端正在服務...
('127.0.0.1', 52467)
我想找迪麗熱巴
服務端>>>...你好啊亞峰,我是熱巴,有什麼能幫你嗎
熱巴youare beautiful
服務端>>>...謝謝亞峰


-client

import socket

#買手機

client = socket.socket()

#撥打號碼
client.connect(
    ('127.0.0.1', 9876)
)

print('客戶端正在發送請求...')

while True:
    send_data = input('客戶端>>>:')
    client.send(send_data.encode('utf-8'))
    data = client.recv(1024)

    if data.decode('utf-8') == 'q':
        break

    if len(data) == 0:
        break

    print(data.decode('utf-8'))
>>>
client.close()
客戶端正在發送請求...
客戶端>>>:我想找熱巴
你好啊亞峰,我是熱巴,有什麼能幫你嗎
客戶端>>>:熱巴you are beautiful
謝謝亞峰
客戶端>>>:

四、粘包問題

  • 粘包問題的出現以及幾種情況
  • 第一個問題
-sever
#問題一不知道數據的具體長度
# import socket
#
# import subprocess
#
# #買手機
# sever = socket.socket()
#
# #綁定電話卡
# sever.bind(
#     ('127.0.0.1', 9867)
# )
#
# #半整數池
# sever.listen(5)
#
# while True:
#     coon, addr = sever.accept()
#     print(addr)
#
#     while True:
#         try:
#             #recv從記憶體中獲取數據
#             cmd = coon.recv(1024)
#
#             if len(cmd) == 0:
#                 continue
#             cmd = cmd.decode('utf-8')
#             if cmd == 'q':
#                 break
#
#             #調用subprocess連接終端,對終端進行操作,並獲取操作後正確或錯誤的結果
#             obj = subprocess.Popen(
#                 cmd, shell=True, stdout=subprocess.PIPE,
#                 stderr=subprocess.PIPE
#             )
#             #結果交給result變數名
#             result = obj.stdout.read() + obj.stderr.read()
#             print(len(result))
#
#             #windows預設是gbk
#             print(result.decode('gbk'))
#
#             #將結果返回給客戶端
#             coon.send(result)
#         except Exception as e:
#             print(e)
#             break
#
#     coon.close()


-client
# import socket
#
# client = socket.socket()
#
# client.connect(
#     ('127.0.0.1', 9867)
# )
#
# while True:
#
#     cmd = input('客戶端輸入的內容:')
#
#     client.send(cmd.encode('utf-8'))
#
#     data = client.recv(19190)
#
#     print(len(data))
#
#     print(data.decode('gbk'))
  • 第二種問題
-sever

#問題二:當發送多次傳入的數據長度卻不是很長的時候,服務端多次接收後面接收的沒內容

import socket

sever = socket.socket()

sever.bind(
    ('127.0.0.1', 9000)
)

sever.listen(5)

coon, addr = sever.accept()

data = coon.recv(10)
print(data)

data = coon.recv(1024)
print(data)

data = coon.recv(1024)
print(data)
>>>b'hellohello'
>>>b'hello'
>>>b''


- client
#問題二
import socket

client = socket.socket()

client.connect(
    ('127.0.0.1', 9000)
)

client.send(b'hello')
client.send(b'hello')
client.send(b'hello')

五、解決粘包問題

  • 演示
- sever

import socket
import subprocess
import struct

sever = socket.socket()

sever.bind(
    ('127.0.0.1', 9000)
)

sever.listen(5)

while True:
    coon, addr = sever.accept()
    print(addr)

    while True:
        try:

            #獲取客戶端傳過來的報頭
            header = coon.recv(4)

            #解包獲取真實的數據長度
            data_len = struct.unpack('i', header)[0]

            #準備接收真實數據
            cmd = coon.recv(data_len)

            if len(cmd) == 0:
                continue

            cmd = cmd.decode('utf-8')

            if cmd == 'q':
                break

            #調用subprocess連接終端,對終端進行操作,並獲取操作後的正確或錯誤的結果
            obj = subprocess.Popen(
                cmd, shell=True,
                stdout=subprocess.PIPE,
                stderr=subprocess.PIPE
            )

            #結果交給result變數名
            result = obj.stdout.read() + obj.stderr.read()

            print('發送給客戶端的真實長度', len(result))

            #將結果返回給客戶端,做一個報頭,返回給客戶端
            header = struct.pack('i', len(result))
            print(len(header))
            coon.send(header)
            coon.send(result)

        except Exception as e:
            print(e)
            break

    coon.close()
    
    
- client


import struct
import socket

client = socket.socket()

client.connect(
    ('127.0.0.1', 9000)
)

while True:

    cmd = input('客戶端輸入的內容:')

    cmd_bytes = cmd.encode('utf-8')

    #做一個報頭
    header = struct.pack('i', len(cmd_bytes))

    print(len(header))

    client.send(header)

    #待伺服器確認長度以後,再發送真實數據長度
    client.send(cmd_bytes)

    #接收服務端返回的報頭
    s_header = client.recv(4)

    #解包,接收服務端返回的真實數據
    data_len = struct.unpack('i', s_header)[0]

    result = client.recv(data_len)

    print('接收伺服器返回的真實數據長度', len(result))
    print(result.decode('gbk'))
  • 演示二
-sever

import socket
import struct
import json

sever = socket.socket()

sever.bind(
    ('127.0.0.1', 9000)
)

sever.listen(5)

while True:

    coon, addr = sever.accept()
    print(addr)

    while True:
        try:
            #獲取客戶端傳來的報頭
            header = coon.recv(4)

            #解包,獲取真實的數據長度
            json_len = struct.unpack('i', header)[0]

            #接收json(dictionary)的真實長度
            json_bytes_data = coon.recv(json_len)

            #反序列化將bytes類型數據變成json數據
            json_data = json_bytes_data.decode('utf-8')

            back_dic = json.loads(json_data)
            print(back_dic)
            print(back_dic.get('movie_len'))

        except Exception as e:
            print(e)
            break

    coon.close()
    
>>>('127.0.0.1', 53414)
>>>{'movie_name': '色戒', 'movie_len': 100000}
>>>100000


-client
import struct
import socket
import json

client = socket.socket()

client.connect(
    ('127.0.0.1', 9000)
)

while True:

    movie_name = input('請輸入上傳電影的名字:').strip()

    #偽裝電影真實數據
    movie_len = 100000

    send_dic = {
        'movie_name': movie_name,
        'movie_len': movie_len
    }

    #序列化
    json = json.dumps(send_dic)
    # print(json)
    # print(json.encode('utf-8'))
    # print(len(json.encode('utf-8')))

    json_bytes = json.encode('utf-8')

    #做一個報頭
    header = struct.pack('i', len(json_bytes))

    #發送報頭
    client.send(header)

    #發送真實數據
    client.send(json_bytes)

>>>請輸入上傳電影的名字:色戒

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 在posix標準推出後,socket在各大主流OS平臺上都得到了很好的支持。而Golang是自帶runtime的跨平臺編程語言,Go中提供給開發者的socket API是建立在操作系統原生socket介面之上的。但golang 中的socket介面在行為特點與操作系統原生介面有一些不同。本文將對結合 ...
  • Problem Description 輸入N個整數順序建立一個單鏈表,將該單鏈表拆分成兩個子鏈表,第一個子鏈表存放了所有的偶數,第二個子鏈表存放了所有的奇數。兩個子鏈表中數據的相對次序與原鏈表一致。 Input 第一行輸入整數N;;第二行依次輸入N個整數。 Output 第一行分別輸出偶數鏈表與奇 ...
  • 一、GC 第一步:判斷對象是否已死?有兩種方法:第一種是引用計數法,即給對象添加一個引用計數器,當被引用時,計數器就+1;當引用失效時,就-1;當計數器為0時,代表對象沒有被引用。但是計數器的缺點就是:對象之間相互引用時導致計數器不為零,無法被回收。第二種方法是可達性分析法,即通過定義一系列的GC ...
  • 解決方法 方法一:換一個埠 若仍然想要使用該埠,則可以將占用該埠的進程殺死即可。 方法二:殺死占用該埠的進程 若仍然想要使用該埠,則可以將占用該埠的進程殺死即可 查找埠被占用的進程id netstat -nao | findstr “埠號” 查詢埠號對應的進程 tasklist | ...
  • 想象一下,程式員沒有WIFI會怎樣? 程式員沒有網路肯定會寸步難行! 但是對於Python程式員來說,只要附近有熱點,分分鐘就能蹭網成功! 想要WIFI破解,python+字典,這是必少不了的。熱點加弱口令也是核心。字典自己加精,你的字典有強大,你能破解的WIFI就越多,提供兩種方法,一般人學到一種 ...
  • 1.os.path.exists()既可以判斷文件是否存在,又可以判斷文件夾是否存在 2.os.path.isfile()判斷文件是否存在 3.os.path.isdir()判斷文件夾是否存在 ...
  • 前言 這篇文章主要是從pinpoint web界面入手,我們的目標是弄清楚兩個問題: 1、 pinpoint左側服務地圖上的調用量數據是怎麼查詢的? 2、界面查詢條件WasOnly是什麼意思? 左側服務地圖調用量來源 從下圖可以看出,A顯示被USER調用299次,線上數值代表著調用量。 我們F12跟 ...
  • 元組 元組是一個輕量級集合,這個集合可以存儲任意元素 元組是使用小括弧,元素之間使用逗號分隔,元素的類型是任意的 若需要訪問元組中的元素 [元組名._數字] 數字是從1開始, 有多少個元素,這個1遞增多少次 例如: 有一個元組,元組中有三個元素 訪問每一個元素值 --> _1 _2 _3 元組屬於輕 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...