基本环境
python2.7
1 Socket的工作方式
Socket套接字在工作时将连接的对端分为服务器Server和客户端Client。服务器和客户端的通信模型如下:
服务器和客户端的通信过程如下:
2 socket的构造方法
- 我们用 socket()函数来创建套接字,语法格式如下:
socket.socket([family[, type[, proto]]])
参数
family: 套接字家族可以使AF_UNIX或者AF_INET
type: 套接字类型可以根据是面向连接的还是非连接分为SOCK_STREAM或SOCK_DGRAM
protocol: 一般不填默认为0.
- 不同的套接字类型有着不同的套接字地址:
1)AF_UNIX地址族中使用的是一个简单的字符串
2)AF_INET地址族中使用的是(host,port)地址对,其中host为主机名、IP地址或URL,port为一个整数;
3)AF_INET6地址族中用四元组(host,port,flowinfo,scopeid)表示。
3 Socket 对象(内建)方法
服务器端套接字
s.bind() 绑定地址(host,port)到套接字, 在AF_INET下,以元组(host,port)的形式表示地址。
s.listen() 开始TCP监听。backlog指定在拒绝连接之前,操作系统可以挂起的最大连接数量。该值至少为1,大部分应用程序设为5就可以了。
s.accept() 被动接受TCP客户端连接,(阻塞式)等待连接的到来客户端套接字
s.connect() 主动初始化TCP服务器连接,。一般address的格式为元组(hostname,port),如果连接出错,返回socket.error错误。
s.connect_ex() connect()函数的扩展版本,出错时返回出错码,而不是抛出异常公共用途的套接字函数
s.recv() 接收TCP数据,数据以字符串形式返回,bufsize指定要接收的最大数据量。flag提供有关消息的其他信息,通常可以忽略。
s.send() 发送TCP数据,将string中的数据发送到连接的套接字。返回值是要发送的字节数量,该数量可能小于string的字节大小。
s.sendall() 完整发送TCP数据,完整发送TCP数据。将string中的数据发送到连接的套接字,但在返回之前会尝试发送所有数据。成功返回None,失败则抛出异常。
s.recvform() 接收UDP数据,与recv()类似,但返回值是(data,address)。其中data是包含接收数据的字符串,address是发送数据的套接字地址。
s.sendto() 发送UDP数据,将数据发送到套接字,address是形式为(ipaddr,port)的元组,指定远程地址。返回值是发送的字节数。
s.close() 关闭套接字
s.getpeername() 返回连接套接字的远程地址。返回值通常是元组(ipaddr,port)。
s.getsockname() 返回套接字自己的地址。通常是一个元组(ipaddr,port)
s.setblocking(flag) 如果flag为0,则将套接字设为非阻塞模式,否则将套接字设为阻塞模式(默认值)。非阻塞模式下,如果调用recv()没有发现任何数据,或send()调用无法立即发送数据,那么将引起socket.error异常。
4 用例说明
# -*- coding:utf-8 -*-
import socket
__doc__ = """简单服务器"""
if __name__ == '__main__':
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM) #生成socket对象
host = socket.gethostname() #绑定服务器地址
port = 1234
s.bind((host, port))
s.listen(10) #开始侦听
while True:
c, addr = s.accept() #等待与客户端建立连接
print "get connection from %s"%str(addr)
c.send("服务端返回内容")
c.close() #关闭与当前客户端的连接
# -*- coding:utf-8 -*-
import socket
__doc__ = """客户端"""
if __name__ == '__main__':
s = socket.socket()
host = "panshan-PC"
port = 1234
s.connect((host, port))
content = ""
line = s.recv(8)
while line:
content += line
line = s.recv(8)
print content
s.close()