亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

python Socket網(wǎng)絡(luò)編程實(shí)現(xiàn)C/S模式和P2P

 更新時(shí)間:2020年06月22日 09:52:33   作者:阿君的貓  
這篇文章主要介紹了python Socket網(wǎng)絡(luò)編程實(shí)現(xiàn)C/S模式和P2P,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧

C/S模式

由于網(wǎng)絡(luò)課需要實(shí)現(xiàn)Socket網(wǎng)絡(luò)編程,所以簡(jiǎn)單實(shí)現(xiàn)了一下,C/S模式分別用TCP/IP協(xié)議與UDP協(xié)議實(shí)現(xiàn),下面將分別講解。

TCP/IP協(xié)議

TCP/IP協(xié)議是面向連接的,即客戶端與服務(wù)器需要先建立連接后才能傳輸數(shù)據(jù),以下是服務(wù)器端的代碼實(shí)現(xiàn)。

服務(wù)端:

import socket
from threading import Thread

def deal(sock,addr):
 print('Accept new connection from {}:{}'.format(addr[0],addr[1]))
 sock.send('與服務(wù)器連接成功!'.encode('utf-8'))
 while True:
  data = sock.recv(1024).decode('utf-8') #1024為接收數(shù)據(jù)的最大大小
  print('receive from {}:{} :{}'.format(addr[0],addr[1],data))
  sock.send('信息已成功收到'.encode('utf-8'))

##創(chuàng)建tcp/IPV4協(xié)議的socket
s = socket.socket(socket.AF_INET,socket.SOCK_STREAM)

#為socket綁定端口
s.bind(('127.0.0.1',10240))
#監(jiān)聽(tīng)端口,參數(shù)5為等待的最大連接量
s.listen(5)
print("Waiting for connection...")

while True:
 sock,addr = s.accept()
 t1 = Thread(target=deal,args=(sock,addr))
 t1.start()

#斷開(kāi)與該客戶端的連接
sock.close()
s.close()

需要注意的是,服務(wù)器在等待客戶端連接時(shí),即accept()函數(shù)這里是阻塞的,如下代碼每次只能接受一個(gè)客戶端的連接。

while True:
  #接受一個(gè)新連接,accept等待并返回一個(gè)客戶端連接
  sock,addr = s.accept()
  print('Accept new connection from {}:{}'.format(addr[0],addr[1]))
  #給客戶端發(fā)送消息
  sock.send('連接成功!'.encode('utf-8'))
  while True:
    data = sock.recv(1024).decode('utf-8')  #1024為接收數(shù)據(jù)的最大大小
    print('receive from {}:{} :{}'.format(addr[0],addr[1],data))
    sock.send('信息已成功收到'.encode('utf-8'))
  #斷開(kāi)與該客戶端的連接
  sock.close()

也就是說(shuō)如果采用以上方式,一個(gè)客戶端與服務(wù)器建立連接后,服務(wù)器就會(huì)進(jìn)入一個(gè)死循環(huán)去收發(fā)該客戶端的信息,因此需要引入多線程,每與一個(gè)客戶端建立連接,就為其創(chuàng)建一個(gè)線程用于控制信息的收發(fā),這樣便可以接受多個(gè)客戶端的連接了。

客戶端:

import socket

s = socket.socket(socket.AF_INET,socket.SOCK_STREAM)

##建立連接
s.connect(('127.0.0.1',10240))

#接收客戶端連接成功服務(wù)器發(fā)來(lái)的消息
print(s.recv(1024).decode('utf-8'))

while True:
  data = input('發(fā)送給服務(wù)器:')
  if len(data)>0:
    
    s.send(data.encode('utf-8'))
    print('form sever:{}'.format(s.recv(1024).decode('utf-8')))
s.close()

客戶端是比較簡(jiǎn)單的,需要與服務(wù)器建立連接后,再進(jìn)行收發(fā)信息,這里不再贅述了。

UDP協(xié)議

UDP協(xié)議是面向無(wú)連接的,即服務(wù)器與客戶端不需要提前建立連接,只需要向指定的端口直接發(fā)送數(shù)據(jù)即可。

服務(wù)端

import socket

#為服務(wù)器創(chuàng)建socket并綁定端口  SOCK_DGRAM指定了socket的類型為udp
s = socket.socket(socket.AF_INET,socket.SOCK_DGRAM)

s.bind(('127.0.0.1',7890))

print('Waiting for data...')
#upd無(wú)需監(jiān)聽(tīng)
while True:
  data,addr = s.recvfrom(1024)
  print('Recevie from {}:{} :{}'.format(addr[0],addr[1],data.decode('utf-8')))
  #sendto的另一個(gè)參數(shù)為客戶端socket地址
  s.sendto('信息已成功收到!'.encode('utf-8'),addr)

客戶端

import socket

#為服務(wù)器創(chuàng)建socket并綁定端口  SOCK_DGRAM指定了socket的類型為udp
s = socket.socket(socket.AF_INET,socket.SOCK_DGRAM)
while True:
  data = input('發(fā)送給服務(wù)器:')
  s.sendto(data.encode('utf-8'),('127.0.0.1',7890))
  
  print('Receive from sever:{}'.format(s.recv(1024).decode('utf-8')))

可以看到UDP協(xié)議是非常簡(jiǎn)單的,由于不需要建立連接,所以也不需要?jiǎng)?chuàng)建線程來(lái)管理數(shù)據(jù)的收發(fā)。

C/S模式的應(yīng)用程序


使用PyQt5對(duì)以上的程序進(jìn)行封裝,這是基于TCP/IP協(xié)議實(shí)現(xiàn)的。

服務(wù)端

from PyQt5.QtWidgets import (QApplication,QPushButton,
             QWidget,QLineEdit,QTextEdit)
import sys
import socket
from threading import Thread

import datetime

class UI(QWidget):
  def __init__(self):
    super().__init__()
    self.initUI()
    
  def initUI(self):
    
    #控件
    self.clear_btn = QPushButton('清空內(nèi)容',self)
    self.text = QTextEdit(self)
    
    #布局
    self.clear_btn.setGeometry(150,400,100,40)
    self.text.setGeometry(20,20,360,370)
    
    self.text.setReadOnly(True)
    
    #信號(hào)連接
    self.clear_btn.clicked.connect(self.commit)
    #初始化socket
    self.s = socket.socket(socket.AF_INET,socket.SOCK_STREAM)

    ##建立連接
    self.s.bind(('127.0.0.1',10240))
    
    self.s.listen(5)
    self.text.setText("Waiting for connection...")
    self.t = Thread(target = self.recv,args = ())
    self.t.start()
    #主窗口布局
    self.setGeometry(300, 300, 400, 450)
    self.setWindowTitle('Server')
    self.show()
    
    
  def commit(self):
    self.text.clear()
      
  def recv(self):
    while True:
      sock,addr = self.s.accept()
      t1 = Thread(target=self.deal,args=(sock,addr))
      t1.start()
    sock.close()
      
  def deal(self,sock,addr):
    #sock,addr = s.accept()
    self.text.append('Accept new connection from {}:{}'.format(addr[0],addr[1]))
    sock.send('與服務(wù)器連接成功!'.encode('utf-8'))
    while True:
      data = sock.recv(1024).decode('utf-8')  #1024為接收數(shù)據(jù)的最大大小
      self.text.append('[{}] receive from {}:{} :{}'.format(datetime.datetime.now().strftime("%Y-%m-%d %H:%M:%S"),addr[0],addr[1],data))
      sock.send('信息已成功收到'.encode('utf-8'))
    sock.close()
    
  def closeEvent(self,event):
    self.s.close()
    event.accept()

if __name__ == '__main__':
  app = QApplication(sys.argv)
  ex = UI()
  sys.exit(app.exec_())

這里需要注意的是,由于Qt的主程序本身一直處于循環(huán),如果直接阻塞等待客戶端連接會(huì)導(dǎo)致程序崩潰,因此需要在Qt初始化時(shí)創(chuàng)建一個(gè)線程用于等待客戶端的連接,要想同時(shí)多個(gè)客戶端訪問(wèn)服務(wù)器,還需要在連接成功后再創(chuàng)建一個(gè)線程單獨(dú)用于接收該客戶端的數(shù)據(jù)。

客戶端

from PyQt5.QtWidgets import (QApplication,QPushButton,
             QWidget,QLineEdit,QTextEdit)
import sys
import socket
from threading import Thread

import datetime

class UI(QWidget):
  def __init__(self):
    super().__init__()
    self.initUI()
    
  def initUI(self):
    
    #控件
    self.edit = QLineEdit(self)
    self.commit_btn = QPushButton('發(fā)送',self)
    self.text = QTextEdit(self)
    
    #布局
    self.edit.setGeometry(20, 410, 280, 30)
    self.commit_btn.setGeometry(310,410,70,30)
    self.text.setGeometry(20,20,360,380)
    
    self.text.setReadOnly(True)
    
    #信號(hào)連接
    self.commit_btn.clicked.connect(self.commit)
    #初始化socket
    self.s = socket.socket(socket.AF_INET,socket.SOCK_STREAM)

    ##建立連接
    self.s.connect(('127.0.0.1',10240))
    self.text.setText('服務(wù)器 [{}]:{}\n'.format(datetime.datetime.now().strftime("%Y-%m-%d %H:%M:%S"),self.s.recv(1024).decode('utf-8')))
    #主窗口布局
    self.setGeometry(300, 300, 400, 450)
    self.setWindowTitle('Client')
    self.show()
    
  def commit(self):
    if len(self.edit.text()):
      text = self.edit.text()
      self.s.send(text.encode('utf-8'))
      self.text.append('本機(jī) [{}]:{}\n'.format(datetime.datetime.now().strftime("%Y-%m-%d %H:%M:%S"),text))
      self.text.append('服務(wù)器 [{}]:{}\n'.format(datetime.datetime.now().strftime("%Y-%m-%d %H:%M:%S"),self.s.recv(1024).decode('utf-8')))
      self.edit.clear()
      
  def closeEvent(self,event):
    self.s.close()
    event.accept()
    
  def recv(self):
    while True:
      pass

if __name__ == '__main__':
  app = QApplication(sys.argv)
  ex = UI()
  sys.exit(app.exec_())

客戶端還是比較簡(jiǎn)單,不需要?jiǎng)?chuàng)建線程,在發(fā)送按紐點(diǎn)擊時(shí)觸發(fā)事件,向服務(wù)器發(fā)送數(shù)據(jù),并將發(fā)送的數(shù)據(jù)與服務(wù)器返回的數(shù)據(jù)顯示在textEdit上。

P2P模式


老師說(shuō)P2P模式就是用兩個(gè)服務(wù)器相互連接通信(我以為是要客戶端發(fā)送給服務(wù)器,服務(wù)器再轉(zhuǎn)發(fā)給另一個(gè)客戶端),為了實(shí)現(xiàn)方便,直接采用UDP協(xié)議,也不用創(chuàng)建那么多線程了。代碼如下:

from PyQt5.QtWidgets import (QApplication,QPushButton,
             QWidget,QLineEdit,QTextEdit,QLabel)
import sys
import socket
from threading import Thread

import datetime

class UI(QWidget):
  def __init__(self):
    super().__init__()
    self.initUI()
    
  def initUI(self):
    
    #控件
    self.edit = QLineEdit(self)
    self.commit_btn = QPushButton('發(fā)送',self)
    self.text = QTextEdit(self)
    self.host_label = QLabel('ip地址:',self)
    self.host = QLineEdit(self)
    self.dst_port_label = QLabel('目標(biāo)端口:',self)
    self.dst_port_edit = QLineEdit(self)
    self.src_port_label = QLabel('本機(jī)端口:',self)
    self.src_port_edit = QLineEdit(self)
    self.que_ren_btn = QPushButton('確認(rèn)',self)
    
    #self.host_label.setStyleSheet("QLabel{font-size:25px}")
    #self.dst_port_label.setStyleSheet("QLabel{font-size:25px}")
    #self.src_port_label.setStyleSheet("QLabel{font-size:25px}")
    #布局
    self.edit.setGeometry(20, 480, 280, 30)
    self.commit_btn.setGeometry(310,480,70,30)
    self.text.setGeometry(20,90,360,380)
    self.host_label.setGeometry(20,20,65,25)
    self.host.setGeometry(90,20,110,25)
    self.dst_port_label.setGeometry(205,20,65,25)
    self.dst_port_edit.setGeometry(275,20,110,25)
    self.src_port_label.setGeometry(20,55,65,25)
    self.src_port_edit.setGeometry(90,55,110,25)
    self.que_ren_btn.setGeometry(205,55,70,25)
    
    self.text.setReadOnly(True)
    
    #信號(hào)連接
    self.commit_btn.clicked.connect(self.commit)
    self.que_ren_btn.clicked.connect(self.que_ren)
    #初始化socket
    self.s = socket.socket(socket.AF_INET,socket.SOCK_DGRAM)

    #主窗口布局
    self.setGeometry(300, 300, 400, 520)
    self.setWindowTitle('Client')
    self.show()
    
  def commit(self):
    if len(self.edit.text()):
      text = self.edit.text()
      self.s.sendto(text.encode('utf-8'),('127.0.0.1',self.dst_port))
      self.text.append('本機(jī) [{}]:\n{}\n'.format(datetime.datetime.now().strftime("%Y-%m-%d %H:%M:%S"),text))
      self.edit.clear()
      
  def closeEvent(self,event):
    self.s.close()
    event.accept()
    
  def recv(self):
    while True:
      data,addr = self.s.recvfrom(1024)
      self.text.append('{}:{}[{}]:\n{}\n'.format(addr[0],addr[1],datetime.datetime.now().strftime("%Y-%m-%d %H:%M:%S"),data.decode('utf-8')))
    
  def que_ren(self):
    self.src_port = int(self.src_port_edit.text())
    self.dst_port = int(self.dst_port_edit.text())
    #綁定ip地址與端口
    self.s.bind(('127.0.0.1',self.src_port))
    #開(kāi)啟接收消息的線程
    self.t = Thread(target=self.recv,args=())
    self.t.start()

if __name__ == '__main__':
  app = QApplication(sys.argv)
  ex = UI()
  sys.exit(app.exec_())

首先需要輸入要傳送信息的IP地址,以及端口號(hào),以及設(shè)置自己的端口號(hào)(IP地址沒(méi)有用到,我設(shè)置了是127.0.0.1),點(diǎn)擊確定按鈕時(shí)觸發(fā)事件,會(huì)為socket綁定端口號(hào),并且創(chuàng)建一個(gè)用于接收消息的線程,在點(diǎn)擊發(fā)送按鈕時(shí)會(huì)觸發(fā)另一個(gè)事件用于發(fā)送消息,發(fā)送與接收的消息最后會(huì)顯示在TextEdit上。

注意

這里要統(tǒng)一說(shuō)明一下,在使用Qt封裝后程序會(huì)一直循環(huán)運(yùn)行,導(dǎo)致關(guān)閉程序時(shí)socket也沒(méi)有關(guān)閉(因?yàn)槲乙矂倢W(xué),不清楚不關(guān)閉的后果,可能會(huì)占用這個(gè)端口一段時(shí)間吧),因此需要重寫Qt的closeEvent函數(shù),在該函數(shù)中進(jìn)行關(guān)閉。

總結(jié)

到此這篇關(guān)于python Socket網(wǎng)絡(luò)編程實(shí)現(xiàn)C/S模式和P2P的文章就介紹到這了,更多相關(guān)python Socket C/S模式和P2P內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Pycharm 2020年最新激活碼(親測(cè)有效)

    Pycharm 2020年最新激活碼(親測(cè)有效)

    本文給大家分享Pycharm 2020年最新激活碼,親測(cè)試過(guò)可以放心使用,感興趣的朋友快來(lái)吧
    2020-01-01
  • Python?Matplotlib基本用法詳解

    Python?Matplotlib基本用法詳解

    Matplotlib?是Python中類似?MATLAB?的繪圖工具,熟悉?MATLAB?也可以很快的上手?Matplotlib,這篇文章主要介紹了Python?Matplotlib基本用法,需要的朋友可以參考下
    2023-03-03
  • Django中的ajax請(qǐng)求

    Django中的ajax請(qǐng)求

    今天小編就為大家分享一篇關(guān)于Django中的ajax請(qǐng)求,小編覺(jué)得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來(lái)看看吧
    2018-10-10
  • flask框架json數(shù)據(jù)的拿取和返回操作示例

    flask框架json數(shù)據(jù)的拿取和返回操作示例

    這篇文章主要介紹了flask框架json數(shù)據(jù)的拿取和返回操作,結(jié)合實(shí)例形式分析了flask框架針對(duì)json格式數(shù)據(jù)的解析、數(shù)據(jù)庫(kù)操作與輸出等相關(guān)操作技巧,需要的朋友可以參考下
    2019-11-11
  • tensorboard實(shí)現(xiàn)同時(shí)顯示訓(xùn)練曲線和測(cè)試曲線

    tensorboard實(shí)現(xiàn)同時(shí)顯示訓(xùn)練曲線和測(cè)試曲線

    今天小編就為大家分享一篇tensorboard實(shí)現(xiàn)同時(shí)顯示訓(xùn)練曲線和測(cè)試曲線,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2020-01-01
  • Pytorch模型轉(zhuǎn)onnx模型實(shí)例

    Pytorch模型轉(zhuǎn)onnx模型實(shí)例

    今天小編就為大家分享一篇Pytorch模型轉(zhuǎn)onnx模型實(shí)例,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2020-01-01
  • 詳解Python中常用的激活函數(shù)(Sigmoid、Tanh、ReLU等)

    詳解Python中常用的激活函數(shù)(Sigmoid、Tanh、ReLU等)

    激活函數(shù) (Activation functions) 對(duì)于人工神經(jīng)網(wǎng)絡(luò)模型去學(xué)習(xí)、理解非常復(fù)雜和非線性的函數(shù)來(lái)說(shuō)具有十分重要的作用,這篇文章主要介紹了Python中常用的激活函數(shù)(Sigmoid、Tanh、ReLU等),需要的朋友可以參考下
    2023-04-04
  • Python修改Excel數(shù)據(jù)的實(shí)例代碼

    Python修改Excel數(shù)據(jù)的實(shí)例代碼

    Python修改Excel數(shù)據(jù)的方法。
    2013-11-11
  • python PaddleSpeech實(shí)現(xiàn)嬰兒啼哭識(shí)別

    python PaddleSpeech實(shí)現(xiàn)嬰兒啼哭識(shí)別

    這篇文章主要為大家介紹了python PaddleSpeech實(shí)現(xiàn)嬰兒啼哭識(shí)別操作詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-08-08
  • python中的lambda表達(dá)式用法詳解

    python中的lambda表達(dá)式用法詳解

    這篇文章主要介紹了python中的lambda表達(dá)式用法,詳細(xì)分析了lambda函數(shù)的功能、定義、使用方法與相關(guān)技巧,需要的朋友可以參考下
    2016-06-06

最新評(píng)論