python socket網(wǎng)絡(luò)編程步驟詳解(socket套接字使用)
一、套接字
套接字是為特定網(wǎng)絡(luò)協(xié)議(例如TCP/IP,ICMP/IP,UDP/IP等)套件對(duì)上的網(wǎng)絡(luò)應(yīng)用程序提供者提供當(dāng)前可移植標(biāo)準(zhǔn)的對(duì)象。它們?cè)试S程序接受并進(jìn)行連接,如發(fā)送和接受數(shù)據(jù)。為了建立通信通道,網(wǎng)絡(luò)通信的每個(gè)端點(diǎn)擁有一個(gè)套接字對(duì)象極為重要。
套接字為BSD UNIX系統(tǒng)核心的一部分,而且他們也被許多其他類(lèi)似UNIX的操作系統(tǒng)包括Linux所采納。許多非BSD UNIX系統(tǒng)(如ms-dos,windows,os/2,mac os及大部分主機(jī)環(huán)境)都以庫(kù)形式提供對(duì)套接字的支持。
三種最流行的套接字類(lèi)型是:stream,datagram和raw。stream和datagram套接字可以直接與TCP協(xié)議進(jìn)行接口,而raw套接字則接口到IP協(xié)議。但套接字并不限于TCP/IP。
二、套接字模塊
套接字模塊是一個(gè)非常簡(jiǎn)單的基于對(duì)象的接口,它提供對(duì)低層BSD套接字樣式網(wǎng)絡(luò)的訪問(wèn)。使用該模塊可以實(shí)現(xiàn)客戶(hù)機(jī)和服務(wù)器套接字。要在python 中建立具有TCP和流套接字的簡(jiǎn)單服務(wù)器,需要使用socket模塊。利用該模塊包含的函數(shù)和類(lèi)定義,可生成通過(guò)網(wǎng)絡(luò)通信的程序。一般來(lái)說(shuō),建立服務(wù)器連接需要六個(gè)步驟。
第1步是創(chuàng)建socket對(duì)象。調(diào)用socket構(gòu)造函數(shù)。
socket=socket.socket(familly,type)
family的值可以是AF_UNIX(Unix域,用于同一臺(tái)機(jī)器上的進(jìn)程間通訊),也可以是AF_INET(對(duì)于IPV4協(xié)議的TCP和 UDP),至于type參數(shù),SOCK_STREAM(流套接字)或者 SOCK_DGRAM(數(shù)據(jù)報(bào)文套接字),SOCK_RAW(raw套接字)。
第2步則是將socket綁定(指派)到指定地址上,socket.bind(address)
address必須是一個(gè)雙元素元組,((host,port)),主機(jī)名或者ip地址+端口號(hào)。如果端口號(hào)正在被使用或者保留,或者主機(jī)名或ip地址錯(cuò)誤,則引發(fā)socke.error異常。
第3步,綁定后,必須準(zhǔn)備好套接字,以便接受連接請(qǐng)求。
socket.listen(backlog)
backlog指定了最多連接數(shù),至少為1,接到連接請(qǐng)求后,這些請(qǐng)求必須排隊(duì),如果隊(duì)列已滿(mǎn),則拒絕請(qǐng)求。
第4步,服務(wù)器套接字通過(guò)socket的accept方法等待客戶(hù)請(qǐng)求一個(gè)連接:
connection,address=socket.accept()
調(diào)用accept方法時(shí),socket會(huì)進(jìn)入'waiting'(或阻塞)狀態(tài)??蛻?hù)請(qǐng)求連接時(shí),方法建立連接并返回服務(wù)器。accept方法返回一個(gè)含有倆個(gè)元素的元組,形如(connection,address)。第一個(gè)元素(connection)是新的socket對(duì)象,服務(wù)器通過(guò)它與客戶(hù)通信;第二個(gè)元素(address)是客戶(hù)的internet地址。
第5步是處理階段,服務(wù)器和客戶(hù)通過(guò)send和recv方法通信(傳輸數(shù)據(jù))。服務(wù)器調(diào)用send,并采用字符串形式向客戶(hù)發(fā)送信息。send方法返回已發(fā)送的字符個(gè)數(shù)。服務(wù)器使用recv方法從客戶(hù)接受信息。調(diào)用recv時(shí),必須指定一個(gè)整數(shù)來(lái)控制本次調(diào)用所接受的最大數(shù)據(jù)量。recv方法在接受數(shù)據(jù)時(shí)會(huì)進(jìn)入'blocket'狀態(tài),最后返回一個(gè)字符串,用它來(lái)表示收到的數(shù)據(jù)。如果發(fā)送的量超過(guò)recv所允許,數(shù)據(jù)會(huì)被截?cái)?。多余的?shù)據(jù)將緩沖于接受端。以后調(diào)用recv時(shí),多余的數(shù)據(jù)會(huì)從緩沖區(qū)刪除。
第6步,傳輸結(jié)束,服務(wù)器調(diào)用socket的close方法以關(guān)閉連接。
建立一個(gè)簡(jiǎn)單客戶(hù)連接則需要4個(gè)步驟。
第1步,創(chuàng)建一個(gè)socket以連接服務(wù)器 socket=socket.socket(family,type)
第2步,使用socket的connect方法連接服務(wù)器 socket.connect((host,port))
第3步,客戶(hù)和服務(wù)器通過(guò)send和recv方法通信。
第4步,結(jié)束后,客戶(hù)通過(guò)調(diào)用socket的close方法來(lái)關(guān)閉連接。
python 編寫(xiě)server的步驟:
第一步是創(chuàng)建socket對(duì)象。調(diào)用socket構(gòu)造函數(shù)。如:
socket = socket.socket( family, type )
family參數(shù)代表地址家族,可為AF_INET或AF_UNIX。AF_INET家族包括Internet地址,AF_UNIX家族用于同一臺(tái)機(jī)器上的進(jìn)程間通信。
type參數(shù)代表套接字類(lèi)型,可為SOCK_STREAM(流套接字)和SOCK_DGRAM(數(shù)據(jù)報(bào)套接字)。
第二步是將socket綁定到指定地址。這是通過(guò)socket對(duì)象的bind方法來(lái)實(shí)現(xiàn)的:
socket.bind( address )
由AF_INET所創(chuàng)建的套接字,address地址必須是一個(gè)雙元素元組,格式是(host,port)。host代表主機(jī),port代表端口號(hào)。如果端口號(hào)正在使用、主機(jī)名不正確或端口已被保留,bind方法將引發(fā)socket.error異常。
第三步是使用socket套接字的listen方法接收連接請(qǐng)求。
socket.listen( backlog )
backlog指定最多允許多少個(gè)客戶(hù)連接到服務(wù)器。它的值至少為1。收到連接請(qǐng)求后,這些請(qǐng)求需要排隊(duì),如果隊(duì)列滿(mǎn),就拒絕請(qǐng)求。
第四步是服務(wù)器套接字通過(guò)socket的accept方法等待客戶(hù)請(qǐng)求一個(gè)連接。
connection, address = socket.accept()
調(diào) 用accept方法時(shí),socket會(huì)時(shí)入“waiting”狀態(tài)。客戶(hù)請(qǐng)求連接時(shí),方法建立連接并返回服務(wù)器。accept方法返回一個(gè)含有兩個(gè)元素的 元組(connection,address)。第一個(gè)元素connection是新的socket對(duì)象,服務(wù)器必須通過(guò)它與客戶(hù)通信;第二個(gè)元素 address是客戶(hù)的Internet地址。
第五步是處理階段,服務(wù)器和客戶(hù)端通過(guò)send和recv方法通信(傳輸 數(shù)據(jù))。服務(wù)器調(diào)用send,并采用字符串形式向客戶(hù)發(fā)送信息。send方法返回已發(fā)送的字符個(gè)數(shù)。服務(wù)器使用recv方法從客戶(hù)接收信息。調(diào)用recv 時(shí),服務(wù)器必須指定一個(gè)整數(shù),它對(duì)應(yīng)于可通過(guò)本次方法調(diào)用來(lái)接收的最大數(shù)據(jù)量。recv方法在接收數(shù)據(jù)時(shí)會(huì)進(jìn)入“blocked”狀態(tài),最后返回一個(gè)字符 串,用它表示收到的數(shù)據(jù)。如果發(fā)送的數(shù)據(jù)量超過(guò)了recv所允許的,數(shù)據(jù)會(huì)被截短。多余的數(shù)據(jù)將緩沖于接收端。以后調(diào)用recv時(shí),多余的數(shù)據(jù)會(huì)從緩沖區(qū) 刪除(以及自上次調(diào)用recv以來(lái),客戶(hù)可能發(fā)送的其它任何數(shù)據(jù))。
傳輸結(jié)束,服務(wù)器調(diào)用socket的close方法關(guān)閉連接。
python編寫(xiě)client的步驟:
創(chuàng)建一個(gè)socket以連接服務(wù)器:socket = socket.socket( family, type )
使用socket的connect方法連接服務(wù)器。對(duì)于AF_INET家族,連接格式如下:
socket.connect( (host,port) )
host代表服務(wù)器主機(jī)名或IP,port代表服務(wù)器進(jìn)程所綁定的端口號(hào)。如連接成功,客戶(hù)就可通過(guò)套接字與服務(wù)器通信,如果連接失敗,會(huì)引發(fā)socket.error異常。
處理階段,客戶(hù)和服務(wù)器將通過(guò)send方法和recv方法通信。
傳輸結(jié)束,客戶(hù)通過(guò)調(diào)用socket的close方法關(guān)閉連接。
下面給個(gè)簡(jiǎn)單的例子:
server.py
if __name__ == '__main__':
import socket
sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
sock.bind(('localhost', 8001))
sock.listen(5)
while True:
connection,address = sock.accept()
try:
connection.settimeout(5)
buf = connection.recv(1024)
if buf == '1':
connection.send('welcome to server!')
else:
connection.send('please go out!')
except socket.timeout:
print 'time out'
connection.close()
client.py
if __name__ == '__main__':
import socket
sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
sock.connect(('localhost', 8001))
import time
time.sleep(2)
sock.send('1')
print sock.recv(1024)
sock.close()
在終端運(yùn)行server.py,然后運(yùn)行clien.py,會(huì)在終端打印“welcome to server!"。如果更改client.py的sock.send('1')為其它值在終端會(huì)打印”please go out!“,更改time.sleep(2)為大于5的數(shù)值, 服務(wù)器將會(huì)超時(shí)。
舉例:
服務(wù)端:
#socket server端
#獲取socket構(gòu)造及常量
from socket import *
#''代表服務(wù)器為localhost
myHost = ''
#在一個(gè)非保留端口號(hào)上進(jìn)行監(jiān)聽(tīng)
myPort = 50007
#設(shè)置一個(gè)TCP socket對(duì)象
sockobj = socket(AF_INET, SOCK_STREAM)
#綁定它至端口號(hào)
sockobj.bind((myHost, myPort))
#監(jiān)聽(tīng),允許5個(gè)連結(jié)
sockobj.listen(5)
#直到進(jìn)程結(jié)束時(shí)才結(jié)束循環(huán)
while True:
#等待下一個(gè)客戶(hù)端連結(jié)
connection, address = sockobj.accept( )
#連結(jié)是一個(gè)新的socket
print 'Server connected by', address
while True:
#讀取客戶(hù)端套接字的下一行
data = connection.recv(1024)
#如果沒(méi)有數(shù)量的話,那么跳出循環(huán)
if not data: break
#發(fā)送一個(gè)回復(fù)至客戶(hù)端
connection.send('Echo=>' + data)
#當(dāng)socket關(guān)閉時(shí)eof
connection.close( )
客戶(hù)端:
import sys
from socket import *
serverHost = 'localhost'
serverPort = 50007
#發(fā)送至服務(wù)端的默認(rèn)文本
message = ['Hello network world']
#如果參數(shù)大于1的話,連結(jié)的服務(wù)端為第一個(gè)參數(shù)
if len(sys.argv) > 1:
serverHost = sys.argv[1]
#如果參數(shù)大于2的話,連結(jié)的文字為第二個(gè)參數(shù)
if len(sys.argv) > 2:
message = sys.argv[2:]
#建立一個(gè)tcp/ip套接字對(duì)象
sockobj = socket(AF_INET, SOCK_STREAM)
#連結(jié)至服務(wù)器及端口
sockobj.connect((serverHost, serverPort))
for line in message:
#經(jīng)過(guò)套按字發(fā)送line至服務(wù)端
sockobj.send(line)
#從服務(wù)端接收到的數(shù)據(jù),上限為1k
data = sockobj.recv(1024)
#確認(rèn)他是引用的,是'x'
print 'Client received:', repr(data)
#關(guān)閉套接字
sockobj.close( )
相關(guān)文章
Django?ORM?事務(wù)和查詢(xún)優(yōu)化的操作方法
這篇文章主要介紹了Django?ORM?事務(wù)和查詢(xún)優(yōu)化,包括事務(wù)操作、ORM 惰性查詢(xún)及only與defer相關(guān)知識(shí),本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-09-09
基于Django框架的權(quán)限組件rbac實(shí)例講解
今天小編就為大家分享一篇基于Django框架的權(quán)限組件rbac實(shí)例講解,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2019-08-08
django之如何按日期查詢(xún)數(shù)據(jù)
這篇文章主要介紹了django之如何按日期查詢(xún)數(shù)據(jù)問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2023-08-08
Python?scipy利用快速傅里葉變換實(shí)現(xiàn)濾波
這篇文章主要為大家詳細(xì)介紹了Python?scipy如何利用快速傅里葉變換實(shí)現(xiàn)濾波,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2024-01-01
Django框架cookie和session方法及參數(shù)設(shè)置
這篇文章主要為大家介紹了Django框架cookie和session參數(shù)設(shè)置及介紹,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-03-03

