python标准库中内置了底层网络接口socket,以下代码均默认from socket import *
初步认识
socket 被翻译成套接字,尽管有些人诟病,但我觉得还挺贴切的。其功能是提供低级别的网络服务,最常用的就是根据IP来传输数据。
所谓传输,就要有两个“端”,首先做一个服务端
import socket
s = socket.socket()
host = socket.gethostname()
port = 12345
s.bind((host,port))
s.listen(3)
while True:
c, addr = s.accept()
print("Linked @ Addr",addr)
break
while True:
data = input("input data:")
c.send(data.encode("utf8"))
if data=="exit":
c.close()
break
需要注意的是,send 需要发送的内容是二进制码,故而通过encode 和decode 来编解码。最后如果输入exit 则退出。
然后写一个客户端
import socket
s = socket.socket()
host = socket.gethostname()
port = 12345
s.connect((host,port))
while True:
data = s.recv(1024).decode("utf8")
if data!=b'':
print("receive data:", data)
if data[:4]=="exit":
s.close()
break
运行之后,服务端和客户端的输入输出分别为
Linked @ Addr ('192.168.1.113', 9953)
input data:hello world
11
input data:who are you
11
input data:can u speak chinese?
20
input data:exit
4
receive data: hello world
receive data: who are you
receive data: can u speak chinese?
receive data: exit
socket对象
在上例中,通过socket.socket 创建了一个socket 对象,其完整的构造函数为
socket.socket(family=AF_INET, type=SOCK_STREAM,proto=0,fileno=None)
其中,family 表示套接字的地址族,主要包括三类
地址族 | AF_INET | AF_INET6 | AF_UNIX |
---|
协议来源 | IPv4 | IPv6 | UNIX |
在创建串口时,可以通过或运算| 同时选择多个地址族。
type 为套接字类型,比较常用的有两种
- SOCK_STREAM,为流式套接字,特点是传输与接收顺序相同,安全。
- SOCK_DGRAM,为数据报格式套接字,特点是快、无顺序、可能丢失
proto为协议号,一般为0,当协议族为AF_CAN 时,协议应为CAN_RAW , CAN_BCM , CAN_ISOTP 或 CAN_J1939 。
fileno代表一个创建socket 的文件。
尽管构造函数并无声明客户端与服务端的参数,但从功能来看,二者所能够调用的成员应该并不完全相同。
其中,bind ,listen ,accept 这三个方法为服务端的专有方法,其功能分别为
bind(address) :将其绑定到某个地址,其中地址address 一般为一个元组,包括IP和端口号listen(N) :启动一个服务器用于接受连接,N 为最多连接个数,不小于0。accept() :接受一个连接,无参数,返回值是(conn, address) 元组,conn 是一个新的socket 对象,用于收发数据。
相应地,客户端也有两个专用的方法
connect(address) :连接某个地址。connect_ex(address) :和connect 相比,当出现错误时返回错误码,而不报错。
接下来是客户端和服务端均可使用的方法,其中最关键的就是发送send 和接收recv 。
其中,与发送相关的函数有
send(bytes) :其中bytes 为发送字节,返回已发送的字节(有的时候可能并没有发完)。sendall(bytes) :和send 相比,会持续发送bytes ,直到所有数据都已发送或者报错。sendfile(file,offset=0,count=None) :在Unix下发送文件,在Windows下和send 相同,相当于是不可用的。sendto(bytes,addresss) :指定地址发送数据。
在Windows中可用的接收相关函数有两组,其中bufsize 表示接收数据的最大字节数。
| 返回数据 | 返回数据+接收端地址 |
---|
不写入缓冲区 | recv(bufsize) | recvfrom(bufsize) | 写入缓冲区buf | recv_into(buf,bufsize) | recvfrom_into(buf,bufsize) |
get-set是很多模块中都会出现的函数名称,前者代表获取某种参数,后者代表设置某些参数,一般后者的输入为前者的输出。
get | set | |
---|
get_inheritable() | set_inheritable(i) | 套接字文件描述符 | getblocking() | setblocking(flag) | flag 为false 是非阻塞,否则阻塞 | getpeername() | | 获取套接字连接到的远程地址 | getsockname() | | 获取套接字本机地址 | gettimeout() | settimeout(value) | 超时秒数 |
套接字中止或者关闭的方法有下面几种
close() 关闭套接字的文件描述符detach() 关闭套接字对象,但并不关闭文件描述符
shutdown(how) 可以部分关闭套接字的连接,其中how 为
SHUT_RD :后续不再允许接收SHUT_WR :后续不再允许发送SHUT_RDWR :后续的发送和接收都不允许
|