python3中socket套接字

一、TCP

1、tcp服務(wù)器創(chuàng)建

#創(chuàng)建服務(wù)器
from socket import *
from time import ctime #導(dǎo)入ctime

HOST = ''    #任意主機(jī)
PORT = 21567  #隨機(jī)提供個(gè)端口號(hào)
BUFSIZ = 1024  # 緩沖區(qū)大小設(shè)置為1KB,可以根據(jù)網(wǎng)絡(luò)性能和程序需要改變這個(gè)容量
ADDR = (HOST, PORT)

tcpSerSock = socket(AF_INET, SOCK_STREAM)  #分配了 TCP 服務(wù)器套接字
tcpSerSock.bind(ADDR)  #綁定到服務(wù)器地址以及開啟 TCP 監(jiān)聽器的調(diào)用。
tcpSerSock.listen(5)  
#listen()方法的參數(shù)是在連接被轉(zhuǎn)接或拒絕之前,傳入連接請求的最大數(shù)

一旦進(jìn)入服務(wù)器的無限循環(huán)之中,我們就(被動(dòng)地)等待客戶端的連接。當(dāng)一個(gè)連接請求出現(xiàn)時(shí),我們進(jìn)入對話循環(huán)中,在該循環(huán)中我們等待客戶端發(fā)送的消息。如果消息是空白的,這意味著客戶端已經(jīng)退出,所以此時(shí)我們將跳出對話循環(huán),關(guān)閉當(dāng)前客戶端連接,然后等待另一個(gè)客戶端連接。如果確實(shí)得到了客戶端發(fā)送的消息,就將其格式化并返回相同的數(shù)據(jù),但是會(huì)在這些數(shù)據(jù)中加上當(dāng)前時(shí)間戳的前綴。最后一行永遠(yuǎn)不會(huì)執(zhí)行,它只是用來提醒讀者,如果寫了一個(gè)處理程序來考慮一個(gè)更加優(yōu)雅的退出方式,正如前面討論的,那么應(yīng)該調(diào)用 close()方法。

while True:
  print("waiting for connection")
  tcpCliSock, addr = tcpSerSock.accept() # 接收客戶端連接,返回客戶端和地址
  print("...connected from:", addr)

  while True:
    data = tcpCliSock.recv(BUFSIZ).decode() 
    #對話(接收 / 發(fā)送) 接收客戶端的data
    if not data:
      break
    tcpCliSock.send(('service:'+ctime()+'--'+data).encode()) 
 #發(fā)送時(shí)間戳 和data信息給客戶端

  tcpCliSock.close()
tcpSerSock.close()

2、tcp客戶端創(chuàng)建

from socket import *

HOST = 'localhost'
PORT = 21567      #端口號(hào) PORT 應(yīng)該與你為服務(wù)器設(shè)置的完全相同(否則,將無法進(jìn)行通信)
BUFSIZ = 1024
ADDR = (HOST, PORT)

tcpCliSock = socket(AF_INET, SOCK_STREAM)  #分配 TCP 客戶端套接字
tcpCliSock.connect(ADDR)  #主動(dòng)連接

"""
  我們必須解碼來自服務(wù)器端的字符串(借助于distutils.log.warn()

"""
while True:
  data = input("> ")
  if not data:   #用戶如果沒有輸入,則終止
    break
  tcpCliSock.send(data.encode())   #發(fā)送客戶端的data給服務(wù)器
  data = tcpCliSock.recv(BUFSIZ).decode() #接收服務(wù)器的data
  if not data:  #或者服務(wù)器終止且對 recv()方法的調(diào)用失敗
    break
  print('返回:%s'%data)
tcpCliSock.close()

二、UDP

1、UDP服務(wù)器創(chuàng)建

這個(gè)腳本創(chuàng)建一個(gè) UDP 服務(wù)器,它接受客戶端發(fā)來的消息,并將加了時(shí)間戳前綴的該消息返回給客戶端。

from socket import *
from time import ctime

HOST = ""
PORT = 21567
BUFSIZ = 1024
ADDR = (HOST, PORT)

udpServer = socket(AF_INET, SOCK_DGRAM)
udpServer.bind(ADDR)

while True:
  print("waiting for masssage")
  data,addr = udpServer.recvfrom(BUFSIZ) #接收
  data = data.decode()
  udpServer.sendto((ctime()+"--"+ data).encode(),addr)
  print("received from and returned to ",addr)

udpServer.close() 

2、UDP客戶端創(chuàng)建

這個(gè)腳本創(chuàng)建一個(gè) UDP 客戶端,它提示用戶輸入發(fā)送給服務(wù)器的消息,并接收服務(wù)器加了時(shí)間戳前綴的消息,然后將它們顯示給用戶。

from socket import *
HOST = 'localhost'
PORT = 21567      #端口號(hào) PORT 應(yīng)該與你為服務(wù)器設(shè)置的完全相同(否則,將無法進(jìn)行通信)
BUFSIZ = 1024
ADDR = (HOST, PORT)

udpCS = socket(AF_INET, SOCK_DGRAM)

while True:
  data = input("> ")
  if not data:
    break
  udpCS.sendto(data.encode(),ADDR)
  data,ADDR = udpCS.recvfrom(BUFSIZ)
  if not data:
    break
  print(data)

udpCS.close()

其實(shí)道理都差不多,在python3中,利用套接字傳輸?shù)膬?nèi)容都以byte形式傳輸,這時(shí)候傳送時(shí)(send/sendto)需要encode,接收(recv)時(shí)需要decode。只要掌握這個(gè)重點(diǎn),處理這個(gè)問題就很簡單了。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

推薦閱讀更多精彩內(nèi)容

  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn),斷路器,智...
    卡卡羅2017閱讀 134,836評(píng)論 18 139
  • Teredo 原理概述 http://www.ipv6bbs.cn/thread-144-1-1.html (出處...
    我是葉問小盆友閱讀 2,052評(píng)論 0 1
  • __block和__weak修飾符的區(qū)別其實(shí)是挺明顯的:1.__block不管是ARC還是MRC模式下都可以使用,...
    LZM輪回閱讀 3,354評(píng)論 0 6
  • 轉(zhuǎn)。。。。。。。。 SOCKET,TCP/UDP,HTTP,FTP (一)TCP/UDP,SOCKET,HTTP,...
    zeqinjie閱讀 3,310評(píng)論 1 53
  • 發(fā)面,尖椒肉末雞蛋和餡兒,把面搟成餅鋪上餡兒卷成卷切塊,平底鍋兩面烙金黃,出鍋…… (由于老張給的尖椒太少,所以做...
    風(fēng)箏2017閱讀 231評(píng)論 0 0