一、socket的定義

  Socket是應(yīng)用層與TCP/IP協(xié)議族通信的中間軟件抽象層,它是一組接口。在設(shè)計(jì)模式中,Socket其實(shí)就是一個(gè)門面模式,它把復(fù)雜的TCP/IP協(xié)議族隱藏在Socket接口后面,對(duì)用戶來(lái)說(shuō),一組簡(jiǎn)單的接口就是全部,讓Socket去組織數(shù)據(jù),以符合指定的協(xié)議。所以,我們無(wú)需深入理解tcp/udp協(xié)議,socket已經(jīng)為我們封裝好了,我們只需要遵循socket的規(guī)定去編程,寫出的程序自然就是遵循tcp/udp標(biāo)準(zhǔn)的。

補(bǔ)充:也有人將socket說(shuō)成ip+port,ip是用來(lái)標(biāo)識(shí)互聯(lián)網(wǎng)中的一臺(tái)主機(jī)的位置,而port是用來(lái)標(biāo)識(shí)這臺(tái)機(jī)器上的一個(gè)應(yīng)用程序,ip地址是配置到網(wǎng)卡上的,而port是應(yīng)用程序開啟的,ip與port的綁定就標(biāo)識(shí)了互聯(lián)網(wǎng)中獨(dú)一無(wú)二的一個(gè)應(yīng)用程序,而程序的pid是同一臺(tái)機(jī)器上不同進(jìn)程或者線程的標(biāo)識(shí)

大數(shù)據(jù)培訓(xùn),云培訓(xùn),數(shù)據(jù)挖掘培訓(xùn),云計(jì)算培訓(xùn),高端軟件開發(fā)培訓(xùn),項(xiàng)目經(jīng)理培訓(xùn)

 

二、套接字發(fā)展史及分類

  套接字起源于 20 世紀(jì) 70 年代加利福尼亞大學(xué)伯克利分校版本的 Unix,即人們所說(shuō)的 BSD Unix。 因此,有時(shí)人們也把套接字稱為“伯克利套接字”或“BSD 套接字”。一開始,套接字被設(shè)計(jì)用在同 一臺(tái)主機(jī)上多個(gè)應(yīng)用程序之間的通訊。這也被稱進(jìn)程間通訊,或 IPC。套接字有兩種(或者稱為有兩個(gè)種族),分別是基于文件型的和基于網(wǎng)絡(luò)型的。 

  • 基于文件類型的套接字家族

套接字家族的名字:AF_UNIX

unix一切皆文件,基于文件的套接字調(diào)用的就是底層的文件系統(tǒng)來(lái)取數(shù)據(jù),兩個(gè)套接字進(jìn)程運(yùn)行在同一機(jī)器,可以通過(guò)訪問(wèn)同一個(gè)文件系統(tǒng)間接完成通信

  • 基于網(wǎng)絡(luò)類型的套接字家族

套接字家族的名字:AF_INET

(還有AF_INET6被用于ipv6,還有一些其他的地址家族,不過(guò),他們要么是只用于某個(gè)平臺(tái),要么就是已經(jīng)被廢棄,或者是很少被使用,或者是根本沒(méi)有實(shí)現(xiàn),所有地址家族中,AF_INET是使用最廣泛的一個(gè),python支持很多種地址家族,但是由于我們只關(guān)心網(wǎng)絡(luò)編程,所以大部分時(shí)候我么只使用AF_INET)

 

三、套接字的工作流程

    一個(gè)生活中的場(chǎng)景。你要打電話給一個(gè)朋友,先撥號(hào),朋友聽到電話鈴聲后提起電話,這時(shí)你和你的朋友就建立起了連接,就可以講話了。等交流結(jié)束,掛斷電話結(jié)束此次交談。

生活中的場(chǎng)景就解釋了套接字的工作原理

大數(shù)據(jù)培訓(xùn),云培訓(xùn),數(shù)據(jù)挖掘培訓(xùn),云計(jì)算培訓(xùn),高端軟件開發(fā)培訓(xùn),項(xiàng)目經(jīng)理培訓(xùn)

先從服務(wù)器端說(shuō)起。服務(wù)器端先初始化Socket,然后與端口綁定(bind),對(duì)端口進(jìn)行監(jiān)聽(listen),調(diào)用accept阻塞,等待客戶端連接。在這時(shí)如果有個(gè)客戶端初始化一個(gè)Socket,然后連接服務(wù)器(connect),如果連接成功,這時(shí)客戶端與服務(wù)器端的連接就建立了。客戶端發(fā)送數(shù)據(jù)請(qǐng)求,服務(wù)器端接收請(qǐng)求并處理請(qǐng)求,然后把回應(yīng)數(shù)據(jù)發(fā)送給客戶端,客戶端讀取數(shù)據(jù),最后關(guān)閉連接,一次交互結(jié)束。

 

四、socket函數(shù)使用

  • socket函數(shù)用法

大數(shù)據(jù)培訓(xùn),云培訓(xùn),數(shù)據(jù)挖掘培訓(xùn),云計(jì)算培訓(xùn),高端軟件開發(fā)培訓(xùn),項(xiàng)目經(jīng)理培訓(xùn)

import socket socket.socket(socket_family,socket_type,protocal=0) #socket_family 可以是 AF_UNIX 或 AF_INET。socket_type 可以是 SOCK_STREAM 或 SOCK_DGRAM。protocol 一般不填,默認(rèn)值為 0。 #獲取tcp/ip套接字 tcpSock = socket.socket(socket.AF_INET, socket.SOCK_STREAM) #獲取udp/ip套接字 udpSock = socket.socket(socket.AF_INET, socket.SOCK_DGRAM) #由于 socket 模塊中有太多的屬性。我們?cè)谶@里破例使用了'from module import *'語(yǔ)句。使用 'from socket import *',我們就把 socket 模塊里的所有屬性都帶到我們的命名空間里了,這樣能 大幅減短我們的代碼。 #例如tcpSock = socket(AF_INET, SOCK_STREAM)

大數(shù)據(jù)培訓(xùn),云培訓(xùn),數(shù)據(jù)挖掘培訓(xùn),云計(jì)算培訓(xùn),高端軟件開發(fā)培訓(xùn),項(xiàng)目經(jīng)理培訓(xùn)



  • 服務(wù)端套接字函數(shù)

s.bind()    #綁定(主機(jī),端口號(hào))到套接字 s.listen()  #開始TCP監(jiān)聽 s.accept()  #被動(dòng)接受TCP客戶的連接,(阻塞式)等待連接的到來(lái)

 

  • 客戶端套接字函數(shù)

s.connect()     #主動(dòng)初始化TCP服務(wù)器連接 s.connect_ex()  #connect()函數(shù)的擴(kuò)展版本,出錯(cuò)時(shí)返回出錯(cuò)碼,而不是拋出異常

 

  • 公共用途的套接字函數(shù)

大數(shù)據(jù)培訓(xùn),云培訓(xùn),數(shù)據(jù)挖掘培訓(xùn),云計(jì)算培訓(xùn),高端軟件開發(fā)培訓(xùn),項(xiàng)目經(jīng)理培訓(xùn)

s.recv()            #接收TCP數(shù)據(jù) s.send()            #發(fā)送TCP數(shù)據(jù)(send在待發(fā)送數(shù)據(jù)量大于己端緩存區(qū)剩余空間時(shí),數(shù)據(jù)丟失,不會(huì)發(fā)完) s.sendall()         #發(fā)送完整的TCP數(shù)據(jù)(本質(zhì)就是循環(huán)調(diào)用send,sendall在待發(fā)送數(shù)據(jù)量大于己端緩存區(qū)剩余空間時(shí),數(shù)據(jù)不丟失,循環(huán)調(diào)用send直到發(fā)完) s.recvfrom()        #接收UDP數(shù)據(jù) s.sendto()          #發(fā)送UDP數(shù)據(jù) s.getpeername()     #連接到當(dāng)前套接字的遠(yuǎn)端的地址 s.getsockname()     #當(dāng)前套接字的地址 s.getsockopt()      #返回指定套接字的參數(shù) s.setsockopt()      #設(shè)置指定套接字的參數(shù) s.close()           #關(guān)閉套接字

大數(shù)據(jù)培訓(xùn),云培訓(xùn),數(shù)據(jù)挖掘培訓(xùn),云計(jì)算培訓(xùn),高端軟件開發(fā)培訓(xùn),項(xiàng)目經(jīng)理培訓(xùn)

 

  • 面向鎖的套接字方法

s.setblocking()     #設(shè)置套接字的阻塞與非阻塞模式 s.settimeout()      #設(shè)置阻塞套接字操作的超時(shí)時(shí)間 s.gettimeout()      #得到阻塞套接字操作的超時(shí)時(shí)間

 

  • 面向文件的套接字方法

s.fileno()          #套接字的文件描述符 s.makefile()        #創(chuàng)建一個(gè)與該套接字相關(guān)的文件

 

打電話的流程演示

服務(wù)端.py

大數(shù)據(jù)培訓(xùn),云培訓(xùn),數(shù)據(jù)挖掘培訓(xùn),云計(jì)算培訓(xùn),高端軟件開發(fā)培訓(xùn),項(xiàng)目經(jīng)理培訓(xùn)

import socket phone=socket.socket(socket.AF_INET,socket.SOCK_STREAM) #買手機(jī) phone.bind(('127.0.0.1',8080)) #插電話卡  phone.listen(5) #開機(jī),backlog print('starting....') conn,addr=phone.accept() #接電話 print(conn) print('client addr',addr) print('ready to read msg') client_msg=conn.recv(1024) #收消息 print('client msg: %s' %client_msg) conn.send(client_msg.upper()) #發(fā)消息  conn.close() phone.close()

大數(shù)據(jù)培訓(xùn),云培訓(xùn),數(shù)據(jù)挖掘培訓(xùn),云計(jì)算培訓(xùn),高端軟件開發(fā)培訓(xùn),項(xiàng)目經(jīng)理培訓(xùn)

 

客戶端.py

大數(shù)據(jù)培訓(xùn),云培訓(xùn),數(shù)據(jù)挖掘培訓(xùn),云計(jì)算培訓(xùn),高端軟件開發(fā)培訓(xùn),項(xiàng)目經(jīng)理培訓(xùn)

import socket phone=socket.socket(socket.AF_INET,socket.SOCK_STREAM) phone.connect(('127.0.0.1',8080)) #撥通電話  phone.send('hello'.encode('utf-8')) #發(fā)消息  back_msg=phone.recv(1024) print(back_msg) phone.close()

大數(shù)據(jù)培訓(xùn),云培訓(xùn),數(shù)據(jù)挖掘培訓(xùn),云計(jì)算培訓(xùn),高端軟件開發(fā)培訓(xùn),項(xiàng)目經(jīng)理培訓(xùn)

 

輸出

服務(wù)端:

starting.... <socket.socket fd=4, family=AddressFamily.AF_INET, type=SocketKind.SOCK_STREAM, proto=0, laddr=('127.0.0.1', 8080), raddr=('127.0.0.1', 65142)> client addr ('127.0.0.1', 65142) ready to read msg client msg: b'hello'

客戶端

b'HELLO'

 

五、基于TCP的套接字

  • tcp服務(wù)端 

大數(shù)據(jù)培訓(xùn),云培訓(xùn),數(shù)據(jù)挖掘培訓(xùn),云計(jì)算培訓(xùn),高端軟件開發(fā)培訓(xùn),項(xiàng)目經(jīng)理培訓(xùn)

ss = socket() #創(chuàng)建服務(wù)器套接字 ss.bind()      #把地址綁定到套接字 ss.listen()      #監(jiān)聽鏈接 inf_loop:      #服務(wù)器無(wú)限循環(huán)     cs = ss.accept() #接受客戶端鏈接     comm_loop:         #通訊循環(huán)         cs.recv()/cs.send() #對(duì)話(接收與發(fā)送)     cs.close()    #關(guān)閉客戶端套接字 ss.close()        #關(guān)閉服務(wù)器套接字(可選)

大數(shù)據(jù)培訓(xùn),云培訓(xùn),數(shù)據(jù)挖掘培訓(xùn),云計(jì)算培訓(xùn),高端軟件開發(fā)培訓(xùn),項(xiàng)目經(jīng)理培訓(xùn)

  • tcp客戶端

cs = socket()    # 創(chuàng)建客戶套接字 cs.connect()    # 嘗試連接服務(wù)器 comm_loop:        # 通訊循環(huán)     cs.send()/cs.recv()    # 對(duì)話(發(fā)送/接收) cs.close()            # 關(guān)閉客戶套接字

socket通信流程與打電話流程類似,我們就以打電話為例來(lái)實(shí)現(xiàn)一個(gè)low版的套接字通信

服務(wù)端

大數(shù)據(jù)培訓(xùn),云培訓(xùn),數(shù)據(jù)挖掘培訓(xùn),云計(jì)算培訓(xùn),高端軟件開發(fā)培訓(xùn),項(xiàng)目經(jīng)理培訓(xùn)

import socket ip_port=('127.0.0.1',9000)  #電話卡 BUFSIZE=1024                #收發(fā)消息的尺寸 s=socket.socket(socket.AF_INET,socket.SOCK_STREAM) #買手機(jī) s.bind(ip_port) #手機(jī)插卡 s.listen(5)     #手機(jī)待機(jī)  conn,addr=s.accept()            #手機(jī)接電話 # print(conn) # print(addr) print('接到來(lái)自%s的電話' %addr[0]) msg=conn.recv(BUFSIZE)             #聽消息,聽話 print(msg,type(msg)) conn.send(msg.upper())          #發(fā)消息,說(shuō)話  conn.close()                    #掛電話  s.close()                       #手機(jī)關(guān)機(jī)

大數(shù)據(jù)培訓(xùn),云培訓(xùn),數(shù)據(jù)挖掘培訓(xùn),云計(jì)算培訓(xùn),高端軟件開發(fā)培訓(xùn),項(xiàng)目經(jīng)理培訓(xùn)

客戶端

大數(shù)據(jù)培訓(xùn),云培訓(xùn),數(shù)據(jù)挖掘培訓(xùn),云計(jì)算培訓(xùn),高端軟件開發(fā)培訓(xùn),項(xiàng)目經(jīng)理培訓(xùn)

import socket ip_port=('127.0.0.1',9000) BUFSIZE=1024 s=socket.socket(socket.AF_INET,socket.SOCK_STREAM) s.connect_ex(ip_port)           #撥電話  s.send('nitouxiang nb'.encode('utf-8'))         #發(fā)消息,說(shuō)話(只能發(fā)送字節(jié)類型)  feedback=s.recv(BUFSIZE)                           #收消息,聽話 print(feedback.decode('utf-8')) s.close()                                       #掛電話

大數(shù)據(jù)培訓(xùn),云培訓(xùn),數(shù)據(jù)挖掘培訓(xùn),云計(jì)算培訓(xùn),高端軟件開發(fā)培訓(xùn),項(xiàng)目經(jīng)理培訓(xùn)

 

輸出

服務(wù)端

接到來(lái)自127.0.0.1的電話 b'nitouxiang nb' <class 'bytes'>

客戶端

NITOUXIANG NB

 

上述流程的問(wèn)題是,服務(wù)端只能接受一次鏈接,然后就徹底關(guān)閉掉了,實(shí)際情況應(yīng)該是,服務(wù)端不斷接受鏈接,然后循環(huán)通信,通信完畢后只關(guān)閉鏈接,服務(wù)器能夠繼續(xù)接收下一次鏈接,下面是修改版

 服務(wù)端

大數(shù)據(jù)培訓(xùn),云培訓(xùn),數(shù)據(jù)挖掘培訓(xùn),云計(jì)算培訓(xùn),高端軟件開發(fā)培訓(xùn),項(xiàng)目經(jīng)理培訓(xùn)

import socket ip_port = ('127.0.0.1',8081)    #電話卡 BUFSIZE=1024 s=socket.socket(socket.AF_INET,socket.SOCK_STREAM)  #買手機(jī) s.bind(ip_port) #手機(jī)插卡 s.listen(5) #手機(jī)待機(jī) while True:                  #新增接收鏈接循環(huán),可以不停的接電話     conn,addr=s.accept()    #手機(jī)接電話     print('接到來(lái)自%s的電話' %addr[0])     while True:                 ##新增通信循環(huán),可以不斷的通信,收發(fā)消息         msg=conn.recv(BUFSIZE)  #聽消息,聽話         if len(msg) == 0:break  #如果不加,那么正在鏈接的客戶端突然斷開,recv便不再阻塞,死循環(huán)發(fā)生         print(msg,type(msg))         conn.send(msg.upper())  #發(fā)消息,說(shuō)話     conn.close()                #掛電話 s.close()               #手機(jī)關(guān)機(jī)

大數(shù)據(jù)培訓(xùn),云培訓(xùn),數(shù)據(jù)挖掘培訓(xùn),云計(jì)算培訓(xùn),高端軟件開發(fā)培訓(xùn),項(xiàng)目經(jīng)理培訓(xùn)

客戶端

大數(shù)據(jù)培訓(xùn),云培訓(xùn),數(shù)據(jù)挖掘培訓(xùn),云計(jì)算培訓(xùn),高端軟件開發(fā)培訓(xùn),項(xiàng)目經(jīng)理培訓(xùn)

import socket ip_port=('127.0.0.1',8081) BUFSIZE=1024 s=socket.socket(socket.AF_INET,socket.SOCK_STREAM) s.connect_ex(ip_port)           #撥電話 while True:                             #新增通信循環(huán),客戶端可以不斷發(fā)收消息     msg=input('>>: ').strip()     if len(msg) == 0:continue     s.send(msg.encode('utf-8'))         #發(fā)消息,說(shuō)話(只能發(fā)送字節(jié)類型)      feedback=s.recv(BUFSIZE)                           #收消息,聽話     print(feedback.decode('utf-8')) s.close()                                       #掛電話

大數(shù)據(jù)培訓(xùn),云培訓(xùn),數(shù)據(jù)挖掘培訓(xùn),云計(jì)算培訓(xùn),高端軟件開發(fā)培訓(xùn),項(xiàng)目經(jīng)理培訓(xùn)

 

補(bǔ)充:

在重啟服務(wù)端時(shí)可能會(huì)遇到

大數(shù)據(jù)培訓(xùn),云培訓(xùn),數(shù)據(jù)挖掘培訓(xùn),云計(jì)算培訓(xùn),高端軟件開發(fā)培訓(xùn),項(xiàng)目經(jīng)理培訓(xùn)

這個(gè)是由于你的服務(wù)端仍然存在四次揮手的time_wait狀態(tài)在占用地址(如果不懂,請(qǐng)深入研究1.tcp三次握手,四次揮手 2.syn洪水攻擊 3.服務(wù)器高并發(fā)情況下會(huì)有大量的time_wait狀態(tài)的優(yōu)化方法)

解決辦法

方法一

#加入一條socket配置,重用ip和端口  phone=socket(AF_INET,SOCK_STREAM) phone.setsockopt(SOL_SOCKET,SO_REUSEADDR,1) #就是它,在bind前加 phone.bind(('127.0.0.1',8080))

方法二

大數(shù)據(jù)培訓(xùn),云培訓(xùn),數(shù)據(jù)挖掘培訓(xùn),云計(jì)算培訓(xùn),高端軟件開發(fā)培訓(xùn),項(xiàng)目經(jīng)理培訓(xùn)

/etc/= 1= 1= 1= 30/sbin/sysctl -= 1= 1 表示開啟重用。允許將TIME-= 1 表示開啟TCP連接中TIME-

大數(shù)據(jù)培訓(xùn),云培訓(xùn),數(shù)據(jù)挖掘培訓(xùn),云計(jì)算培訓(xùn),高端軟件開發(fā)培訓(xùn),項(xiàng)目經(jīng)理培訓(xùn)

 

六、基于UDP的套接字

  • udp服務(wù)端

ss = socket()   #創(chuàng)建一個(gè)服務(wù)器的套接字 ss.bind()       #綁定服務(wù)器套接字 inf_loop:       #服務(wù)器無(wú)限循環(huán)     cs = ss.recvfrom()/ss.sendto() # 對(duì)話(接收與發(fā)送) ss.close()                         # 關(guān)閉服務(wù)器套接字
  • udp客戶端

cs = socket()   # 創(chuàng)建客戶套接字 comm_loop:      # 通訊循環(huán)     cs.sendto()/cs.recvfrom()   # 對(duì)話(發(fā)送/接收) cs.close()                      # 關(guān)閉客戶套接字

示例

服務(wù)端

大數(shù)據(jù)培訓(xùn),云培訓(xùn),數(shù)據(jù)挖掘培訓(xùn),云計(jì)算培訓(xùn),高端軟件開發(fā)培訓(xùn),項(xiàng)目經(jīng)理培訓(xùn)

import socket ip_port=('127.0.0.1',9000) BUFSIZE=1024 udp_server_client=socket.socket(socket.AF_INET,socket.SOCK_DGRAM) udp_server_client.bind(ip_port) while True:     msg,addr=udp_server_client.recvfrom(BUFSIZE)     print(msg,addr)     udp_server_client.sendto(msg.upper(),addr)

大數(shù)據(jù)培訓(xùn),云培訓(xùn),數(shù)據(jù)挖掘培訓(xùn),云計(jì)算培訓(xùn),高端軟件開發(fā)培訓(xùn),項(xiàng)目經(jīng)理培訓(xùn)

客戶端

大數(shù)據(jù)培訓(xùn),云培訓(xùn),數(shù)據(jù)挖掘培訓(xùn),云計(jì)算培訓(xùn),高端軟件開發(fā)培訓(xùn),項(xiàng)目經(jīng)理培訓(xùn)

import socket ip_port=('127.0.0.1',9000) BUFSIZE=1024 udp_server_client=socket.socket(socket.AF_INET,socket.SOCK_DGRAM) while True:     msg=input('>>: ').strip()     if not msg:continue     udp_server_client.sendto(msg.encode('utf-8'),ip_port)     back_msg,addr=udp_server_client.recvfrom(BUFSIZE)     print(back_msg.decode('utf-8'),addr)

大數(shù)據(jù)培訓(xùn),云培訓(xùn),數(shù)據(jù)挖掘培訓(xùn),云計(jì)算培訓(xùn),高端軟件開發(fā)培訓(xùn),項(xiàng)目經(jīng)理培訓(xùn)

輸出

客戶端

>>: 123 123 ('127.0.0.1', 9000) >>: 3 3 ('127.0.0.1', 9000) >>: 4 4 ('127.0.0.1', 9000)

服務(wù)端

b'123' ('127.0.0.1', 53066) b'3' ('127.0.0.1', 53066) b'4' ('127.0.0.1', 53066)

 

模擬QQ聊天,多個(gè)客戶端和服務(wù)端通信

服務(wù)端

大數(shù)據(jù)培訓(xùn),云培訓(xùn),數(shù)據(jù)挖掘培訓(xùn),云計(jì)算培訓(xùn),高端軟件開發(fā)培訓(xùn),項(xiàng)目經(jīng)理培訓(xùn)

import socket ip_port=('127.0.0.1',8081) udp_server_sock=socket.socket(socket.AF_INET,socket.SOCK_DGRAM) #買手機(jī) udp_server_sock.bind(ip_port) while True:     qq_msg,addr=udp_server_sock.recvfrom(1024)     print('來(lái)自[%s:%s]的一條消息:\033[1;44m%s\033[0m' %(addr[0],addr[1],qq_msg.decode('utf-8')))     back_msg=input('回復(fù)消息: ').strip()     udp_server_sock.sendto(back_msg.encode('utf-8'),addr)

大數(shù)據(jù)培訓(xùn),云培訓(xùn),數(shù)據(jù)挖掘培訓(xùn),云計(jì)算培訓(xùn),高端軟件開發(fā)培訓(xùn),項(xiàng)目經(jīng)理培訓(xùn)

客戶端1

大數(shù)據(jù)培訓(xùn),云培訓(xùn),數(shù)據(jù)挖掘培訓(xùn),云計(jì)算培訓(xùn),高端軟件開發(fā)培訓(xùn),項(xiàng)目經(jīng)理培訓(xùn)

import socket BUFSIZE=1024 udp_client_socket=socket.socket(socket.AF_INET,socket.SOCK_DGRAM) qq_name_dic={     'TOM':('127.0.0.1',8081),     'JACK':('127.0.0.1',8081),     '一棵樹':('127.0.0.1',8081),     '武大郎':('127.0.0.1',8081), } while True:     qq_name=input('請(qǐng)選擇聊天對(duì)象: ').strip()     while True:         msg=input('請(qǐng)輸入消息,回車發(fā)送: ').strip()         if msg == 'quit':break         if not msg or not qq_name or qq_name not in qq_name_dic:continue         udp_client_socket.sendto(msg.encode('utf-8'),qq_name_dic[qq_name])         back_msg,addr=udp_client_socket.recvfrom(BUFSIZE)         print('來(lái)自[%s:%s]的一條消息:\033[1;44m%s\033[0m' %(addr[0],addr[1],back_msg.decode('utf-8'))) udp_client_socket.close()

大數(shù)據(jù)培訓(xùn),云培訓(xùn),數(shù)據(jù)挖掘培訓(xùn),云計(jì)算培訓(xùn),高端軟件開發(fā)培訓(xùn),項(xiàng)目經(jīng)理培訓(xùn)

客戶端2

大數(shù)據(jù)培訓(xùn),云培訓(xùn),數(shù)據(jù)挖掘培訓(xùn),云計(jì)算培訓(xùn),高端軟件開發(fā)培訓(xùn),項(xiàng)目經(jīng)理培訓(xùn)

import socket BUFSIZE=1024 udp_client_socket=socket.socket(socket.AF_INET,socket.SOCK_DGRAM) qq_name_dic={     'TOM':('127.0.0.1',8081),     'JACK':('127.0.0.1',8081),     '一棵樹':('127.0.0.1',8081),     '武大郎':('127.0.0.1',8081), } while True:     qq_name=input('請(qǐng)選擇聊天對(duì)象: ').strip()     while True:         msg=input('請(qǐng)輸入消息,回車發(fā)送: ').str
http://www.cnblogs.com/smallmars/p/7064081.html