欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Python中的Socket 與 ScoketServer 通信及遇到問(wèn)題解決方法

 更新時(shí)間:2019年04月01日 10:40:42   作者:東小東  
Socket有一個(gè)緩沖區(qū),緩沖區(qū)是一個(gè)流,先進(jìn)先出,發(fā)送和取出的可自定義大小的,如果取出的數(shù)據(jù)未取完緩沖區(qū),則可能存在數(shù)據(jù)怠慢。本文通過(guò)實(shí)例代碼給大家介紹Python中的Socket 與 ScoketServer 通信及遇到問(wèn)題解決方法 ,需要的朋友參考下吧

Socket有一個(gè)緩沖區(qū),緩沖區(qū)是一個(gè)流,先進(jìn)先出,發(fā)送和取出的可自定義大小的,如果取出的數(shù)據(jù)未取完緩沖區(qū),則可能存在數(shù)據(jù)怠慢。其中【recv(1024)】表示從緩沖區(qū)里取最大為1024個(gè)字節(jié),但實(shí)際取值大小是不確定的,推薦其值小于等于8192。

黏包問(wèn)題:

Socket發(fā)送兩條連續(xù)數(shù)據(jù)時(shí),可能最終會(huì)拼接成一條進(jìn)行發(fā)送

解決方法一:

兩條數(shù)據(jù)間進(jìn)行延時(shí)發(fā)送,如【tiem.sleep(0.5) #延時(shí)0.5s】

解決方法二:

每次發(fā)送后等待對(duì)方確認(rèn)接收信息數(shù)據(jù),發(fā)送一條后就立即接收等待

解決方法三:

設(shè)定接收數(shù)據(jù)大小,發(fā)送端每次發(fā)送需要發(fā)送的數(shù)據(jù)的數(shù)據(jù)大小,接收端通過(guò)設(shè)置【recv(xx)】只接收確定的大小

Socket基本使用:

簡(jiǎn)單的服務(wù)器:

import socket
sser=socket.socket()#得到socket對(duì)象
sser.bind(("0.0.0.0",2699))#建立監(jiān)聽(tīng)
sser.listen(3)
print("等等客戶(hù)端連接")
conn,addr=sser.accept() #等待連接,會(huì)一直處于阻塞,返回連接對(duì)象和對(duì)方地址
print("有客戶(hù)端已經(jīng)連接,IP地址和端口為:",addr)
#接收數(shù)據(jù),會(huì)一直阻塞,建議低于8192
#使用連接對(duì)象操作
rdata=conn.recv(1024)
print(rdata.decode("gbk"))
conn.send("服務(wù)器返回,收到數(shù)據(jù)".encode("gbk"))#發(fā)送數(shù)據(jù),使用連接對(duì)象操作
#關(guān)閉連接
sser.close()

簡(jiǎn)單的客戶(hù)端:

import socket
sclient=socket.socket()#得到socket對(duì)象
#連接服務(wù)器
#失敗會(huì)報(bào)錯(cuò):ConnectionRefusedError
sclient.connect(("192.168.1.135",2699))
sclient.send("東小東".encode("gbk"))#發(fā)送數(shù)據(jù)
#接收數(shù)據(jù),會(huì)一直阻塞,建議低于8192
rdata=sclient.recv(1024)
print(rdata.decode("gbk"))
#關(guān)閉連接
sclient.close()

Soket進(jìn)階:

服務(wù)器進(jìn)階:

實(shí)現(xiàn)客戶(hù)循環(huán)連接及數(shù)據(jù)循環(huán)收發(fā)和判斷客戶(hù)端是否斷開(kāi)

import socket
sser=socket.socket()#得到socket對(duì)象
sser.bind(("0.0.0.0",2699))#建立監(jiān)聽(tīng)
sser.listen(3)
while True:
  print("等等客戶(hù)端連接")
  conn,addr=sser.accept() #等待連接,會(huì)一直處于阻塞,返回連接對(duì)象和對(duì)方地址
  print("有客戶(hù)端已經(jīng)連接,IP地址和端口為:",addr)
  conn.send(("服務(wù)器歡迎你:%s\r\n"%(str(addr))).encode("gbk")) # 發(fā)送數(shù)據(jù),使用連接對(duì)象操作
  while True:
    #接收數(shù)據(jù),會(huì)一直阻塞
    #使用連接對(duì)象操作
    rdata=conn.recv(1024)
    if not rdata: break #判斷客戶(hù)端是否斷開(kāi),斷開(kāi)則收到空數(shù)據(jù)
    print(rdata.decode("gbk"))
    conn.send("服務(wù)器返回,收到數(shù)據(jù)\r\n".encode("gbk"))#發(fā)送數(shù)據(jù),使用連接對(duì)象操作
#關(guān)閉連接
sser.close()

客戶(hù)端進(jìn)階:

實(shí)現(xiàn)循環(huán)收發(fā)和判斷服務(wù)器是否斷開(kāi)

import socket
sclient=socket.socket()#得到socket對(duì)象
#連接服務(wù)器
#失敗會(huì)報(bào)錯(cuò):ConnectionRefusedError
sclient.connect(("192.168.1.135",2699))
while True:
 #接收數(shù)據(jù),會(huì)一直阻塞
 rdata=sclient.recv(1024)
 if not rdata: break # 判斷服務(wù)器是否斷開(kāi),斷開(kāi)則收到空數(shù)據(jù)
 print(rdata.decode("gbk"))
 sclient.send(("客戶(hù)端收到數(shù)據(jù):%s\r\n"%rdata.decode("gbk")).encode("gbk")) # 發(fā)送數(shù)據(jù)
#關(guān)閉連接
sclient.close()

注意:

發(fā)送數(shù)據(jù)不可發(fā)送空字符,否則會(huì)卡住,解決方法為判斷輸入的是否為空值,空值則進(jìn)行數(shù)據(jù)發(fā)送

strx=input("輸入:").strip()#得到控制臺(tái)輸入值
if(len(strx)==0):continue #如果為空字符,則跳出本次循環(huán)
print(strx) #打印
sclient.send(strx.encode("gbk")) # 發(fā)送數(shù)據(jù)

發(fā)送大數(shù)據(jù):

先發(fā)送總體數(shù)據(jù)大小,socket另一端判斷實(shí)際接收的數(shù)據(jù)大小與總數(shù)據(jù)大小進(jìn)行比較,循環(huán)recv()進(jìn)行數(shù)據(jù)接收

 簡(jiǎn)單的ssh實(shí)現(xiàn):

服務(wù)端:

import os
import socket
sser=socket.socket()#得到socket對(duì)象
sser.bind(("0.0.0.0",2697))#建立監(jiān)聽(tīng)
sser.listen(3)
while True:
  print("等等客戶(hù)端連接")
  conn,addr=sser.accept() #等待連接,會(huì)一直處于阻塞,返回連接對(duì)象和對(duì)方地址
  print("有客戶(hù)端已經(jīng)連接,IP地址和端口為:",addr)
  conn.send(("服務(wù)器歡迎你:%s\r\n"%(str(addr))).encode("gbk")) # 發(fā)送數(shù)據(jù),使用連接對(duì)象操作
  while True:
    #接收數(shù)據(jù),會(huì)一直阻塞
    rdata=conn.recv(1024)
    if not rdata: break #判斷客戶(hù)端是否斷開(kāi),斷開(kāi)則收到空數(shù)據(jù)
    sendtox=os.popen(rdata.decode("gbk")).read() #執(zhí)行命令
    conn.send(str(len(sendtox)).encode("gbk")) #發(fā)送執(zhí)行命令的結(jié)果長(zhǎng)度
    #如果長(zhǎng)度大于0 則發(fā)送命令結(jié)果數(shù)據(jù)
    if len(sendtox)>0:
    if conn.recv(10).decode("gbk")=="1":
     conn.sendall(sendtox.encode("gbk"))#發(fā)送最終數(shù)據(jù)
#關(guān)閉連接
sser.close()

客戶(hù)端:

import socket
sclient=socket.socket()#得到socket對(duì)象
#連接服務(wù)器
#失敗會(huì)報(bào)錯(cuò):ConnectionRefusedError
sclient.connect(("192.168.43.21",2697))
# 接收數(shù)據(jù),會(huì)一直阻塞
rdata = sclient.recv(1024)
print(rdata.decode("gbk"))
while True:
 strx = input("請(qǐng)輸入命令:").strip() # 得到控制臺(tái)輸入值
 if (len(strx) == 0): continue # 如果為空字符,則跳出本次循環(huán)
 sclient.send(strx.encode("gbk")) # 發(fā)送數(shù)據(jù)
 rdata = sclient.recv(10)
 if not rdata: break # 判斷服務(wù)器是否斷開(kāi),斷開(kāi)則收到空數(shù)據(jù)
 #判斷命令是否執(zhí)行成功,0為失敗
 dataall=int(rdata.decode("gbk"))
 if dataall==0:
  continue
 sclient.send("1".encode("gbk")) # 發(fā)送確認(rèn)接收數(shù)據(jù)命令
 #循環(huán)接收數(shù)據(jù)
 datanew=0
 while dataall !=datanew:
  rdata=sclient.recv(1024).decode("gbk")
  datanew+=len(rdata)
  print(rdata)
#關(guān)閉連接
sclient.close()

發(fā)送文件:

發(fā)送文件,使用read()讀取文件數(shù)據(jù)后,可循環(huán)調(diào)用send()發(fā)送數(shù)據(jù),或者使用sendall()一次性發(fā)送所有數(shù)據(jù),socket另一端接收可循環(huán)recv()進(jìn)行數(shù)據(jù)接收,且每次接收的數(shù)據(jù)大小是不確定的。文件傳輸需要驗(yàn)證發(fā)送和接受的數(shù)據(jù)是否完全一致,可以通過(guò)數(shù)據(jù)大小加md5雙重驗(yàn)證,發(fā)送端:md5在每次發(fā)送一條數(shù)據(jù)時(shí)進(jìn)行update(),在數(shù)據(jù)發(fā)送完成后再發(fā)送md5值;接受端:md5在每次接收到一條數(shù)據(jù)后進(jìn)行update(),在文件接收完成后再接收發(fā)送端發(fā)送的md5值,將兩值進(jìn)行比較,相同則表示傳輸無(wú)丟包,但加入md5校驗(yàn),將會(huì)影響傳輸速率。

發(fā)送文件數(shù)據(jù):先發(fā)送總體數(shù)據(jù)大小,socket另一端判斷實(shí)際接收的數(shù)據(jù)大小與總數(shù)據(jù)大小進(jìn)行比較,循環(huán)recv()進(jìn)行數(shù)據(jù)接收

示例:客戶(hù)端發(fā)送文件名,服務(wù)器判斷文件是否存在,如果不存在或者是空文件則不進(jìn)行傳輸,服務(wù)器進(jìn)行文件發(fā)送,客戶(hù)端實(shí)現(xiàn)文件接收

服務(wù)器

import os
import socket
sser=socket.socket()#得到socket對(duì)象
sser.bind(("0.0.0.0",2697))#建立監(jiān)聽(tīng)
sser.listen(3)
while True:
  print("等等客戶(hù)端連接")
  conn,addr=sser.accept() #等待連接,會(huì)一直處于阻塞,返回連接對(duì)象和對(duì)方地址
  print("有客戶(hù)端已經(jīng)連接,IP地址和端口為:",addr)
  conn.send(("服務(wù)器歡迎你:%s\r\n"%(str(addr))).encode("gbk")) # 發(fā)送數(shù)據(jù),使用連接對(duì)象操作
  while True:
    #接收數(shù)據(jù),會(huì)一直阻塞
    rdata=conn.recv(1024)
    if not rdata: break #判斷客戶(hù)端是否斷開(kāi),斷開(kāi)則收到空數(shù)據(jù)
    filesize=0
    filenamex=rdata.decode("gbk")
    if os.path.isfile(filenamex): #判斷是否是文件
     filesize=os.stat(filenamex).st_size #得到文件大小
    conn.send(str(filesize).encode("gbk")) #不是文件則發(fā)送0,是文件則是實(shí)際大小
    #如果文件大小大于0 則發(fā)送文件
    if filesize>0:
    if conn.recv(10).decode("gbk")=="1": #等待確認(rèn)接收命令
     #一行一行發(fā)送數(shù)據(jù)
     f=open(filenamex,"rb")
     for linex in f:
      conn.sendall(linex)#發(fā)送最終數(shù)據(jù)
#關(guān)閉連接
sser.close()

客戶(hù)端

import socket
sclient=socket.socket()#得到socket對(duì)象
#連接服務(wù)器
#失敗會(huì)報(bào)錯(cuò):ConnectionRefusedError
sclient.connect(("192.168.43.21",2697))
# 接收數(shù)據(jù),會(huì)一直阻塞
rdata = sclient.recv(1024)
print(rdata.decode("gbk"))
while True:
 filenamex = input("請(qǐng)輸入文件名:").strip() # 得到控制臺(tái)輸入值
 if (len(filenamex) == 0): continue # 如果為空字符,則跳出本次循環(huán)
 sclient.send(filenamex.encode("gbk")) # 發(fā)送數(shù)據(jù)
 rdata = sclient.recv(10)
 if not rdata: break # 判斷服務(wù)器是否斷開(kāi),斷開(kāi)則收到空數(shù)據(jù)
 #判斷命令是否執(zhí)行成功,0為失敗
 dataall=int(rdata.decode("gbk"))
 if dataall==0:
  print("文件不存在或者為空")
  continue
 sclient.send("1".encode("gbk")) # 發(fā)送確認(rèn)接收數(shù)據(jù)命令
 #打開(kāi)文件
 f=open(filenamex,"wb")
 #循環(huán)接收數(shù)據(jù)
 datanew=0
 while dataall !=datanew:
  rdata=sclient.recv(1024)
  datanew+=len(rdata)
  f.write(rdata)
 print("文件(%s)接收完畢"%filenamex)
 f.close() #關(guān)閉文件
#關(guān)閉連接
sclient.close()

ScoketServer

服務(wù)端實(shí)現(xiàn)多并發(fā)效果,可以同時(shí)接入多個(gè)客戶(hù)端

import socketserver
#建立一個(gè)類(lèi),必須繼承 socketserver.BaseRequestHandler 類(lèi)
class DongSocket(socketserver.BaseRequestHandler):
 #必須重寫(xiě)handle方法
 def handle(self):
  print("建立新連接,對(duì)方地址為:{}".format(self.client_address))
  while True:
   try:
    self.datax=self.request.recv(1024).decode("gbk") #接收數(shù)據(jù)
    print("接收的數(shù)據(jù)為:%s"%self.datax)
    self.request.send(("服務(wù)器返回?cái)?shù)據(jù):%s"%self.datax).encode("gbk"))
   except Exception as e:
    print("斷開(kāi),再見(jiàn):{}".format(self.client_address))
    break
#參數(shù):(("ip",端口),自定義類(lèi))
#ss=socketserver.TCPServer(("0.0.0.0",2351),DongSocket) #與之前的socket服務(wù)器效果一致,同時(shí)只能連接一個(gè)客戶(hù)端
ss=socketserver.ThreadingTCPServer(("0.0.0.0",2351),DongSocket) #同時(shí)可以連接多個(gè)客戶(hù)端,多并發(fā)
ss.serve_forever()

總結(jié)

以上所述是小編給大家介紹的Python中的Socket 與 ScoketServer 通信及遇到問(wèn)題解決方法 ,希望對(duì)大家有所幫助,如果大家有任何疑問(wèn)請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)腳本之家網(wǎng)站的支持!

相關(guān)文章

  • Python?nonlocal關(guān)鍵字?與?global?關(guān)鍵字解析

    Python?nonlocal關(guān)鍵字?與?global?關(guān)鍵字解析

    這篇文章主要介紹了Python?nonlocal關(guān)鍵字?與?global?關(guān)鍵字解析,nonlocal關(guān)鍵字用來(lái)在函數(shù)或其他作用域中使用外層變量,global關(guān)鍵字用來(lái)在函數(shù)或其他局部作用域中使用全局變量,更多香瓜內(nèi)容需要的小伙伴可以參考一下
    2022-03-03
  • python 處理微信對(duì)賬單數(shù)據(jù)的實(shí)例代碼

    python 處理微信對(duì)賬單數(shù)據(jù)的實(shí)例代碼

    本文通過(guò)實(shí)例代碼給大家介紹了python 處理微信對(duì)賬單數(shù)據(jù),代碼簡(jiǎn)單易懂,非常不錯(cuò),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2019-07-07
  • Python用5行代碼寫(xiě)一個(gè)自定義簡(jiǎn)單二維碼

    Python用5行代碼寫(xiě)一個(gè)自定義簡(jiǎn)單二維碼

    今天小編就為大家分享一篇關(guān)于Python用5行代碼寫(xiě)一個(gè)自定義簡(jiǎn)單二維碼的文章,小編覺(jué)得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來(lái)看看吧
    2018-10-10
  • python數(shù)字圖像處理之基本圖形的繪制

    python數(shù)字圖像處理之基本圖形的繪制

    這篇文章主要為大家介紹了python數(shù)字圖像處理之基本圖形的繪制,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-06-06
  • 詳解Python中__new__和__init__的區(qū)別與聯(lián)系

    詳解Python中__new__和__init__的區(qū)別與聯(lián)系

    在Python中,每個(gè)對(duì)象都有兩個(gè)特殊的方法:__new__和__init__,本文將詳細(xì)介紹這兩個(gè)方法的不同之處以及它們之間的聯(lián)系,具有一定的參考價(jià)值,感興趣的可以了解一下
    2023-12-12
  • 舉例詳解Python中threading模塊的幾個(gè)常用方法

    舉例詳解Python中threading模塊的幾個(gè)常用方法

    這篇文章主要介紹了舉例詳解Python中threading模塊的幾個(gè)常用方法,threading模塊用來(lái)創(chuàng)建和操作線(xiàn)程,是Python學(xué)習(xí)當(dāng)中的重要知識(shí),需要的朋友可以參考下
    2015-06-06
  • 深入理解python中的ThreadLocal

    深入理解python中的ThreadLocal

    本文主要介紹了深入理解python中的ThreadLocal,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2023-03-03
  • Python常見(jiàn)反爬蟲(chóng)機(jī)制解決方案

    Python常見(jiàn)反爬蟲(chóng)機(jī)制解決方案

    這篇文章主要介紹了Python常見(jiàn)反爬蟲(chóng)機(jī)制解決方案,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-06-06
  • Python 如何操作 SQLite 數(shù)據(jù)庫(kù)

    Python 如何操作 SQLite 數(shù)據(jù)庫(kù)

    這篇文章主要介紹了Python 如何操作 SQLite 數(shù)據(jù)庫(kù),幫助大家更好的理解和學(xué)習(xí),感興趣的朋友可以了解下
    2020-08-08
  • python 協(xié)程 gevent原理與用法分析

    python 協(xié)程 gevent原理與用法分析

    這篇文章主要介紹了python 協(xié)程 gevent原理與用法,結(jié)合實(shí)例形式分析了Python協(xié)程gevent相關(guān)概念、原理、安裝及使用操作技巧,需要的朋友可以參考下
    2019-11-11

最新評(píng)論