一. 什么是TCP协议
TCP协议,传输控制协议(Transmission Control Protocol,缩写为TCP)是一种面向连接的、可靠的、基于字节流的传输层通信协议,它能提供高可靠性通信(即数据无误、数据无丢失、数据无失序、数据无重复到达的通信)
主要的适用场景:
1.适合于对传输质量要求较高,以及传输大量数据的通信。
2.在需要可靠数据传输的场合,通常使用TCP协议。
3.HTTP/HTTPS等网络服务都采用TCP协议。
TCP通信需要经过创建连接、数据传送、终止连接三个步骤。在python中,提供了一个socket模块,他里面封装了系统底层socket接口的python格式API,对于Python开发者来说,只需要导入该模块就可以实现socket编程了。
from socket import *
TCP客户端的核心接口API是connect,由于TCP要求的是稳定传输,所以在后面数据传输前,需要借助三次握手的协议特点去实现连接。而connect函数则是三次握手协议的发起者。
二.Socket简介
socket通常也称作"套接字",用于描述IP地址和端口,是一个通信链的句柄,应用程序通常通过"套接字"向网络发出请求或者应答网络请求。socket起源于Unix,而Unix/Linux基本哲学之一就是“一切皆文件”,对于文件用【打开】【读写】【关闭】模式来操作。socket就是该模式的一个实现,socket是一种特殊的文件,一些socket函数就是对其进行的操作(读/写IO、打开、关闭)
socket和file的区别:
*file模块是针对某个指定文件进行【打开】【读写】【关闭】
*socket模块是针对服务器端和客户端Socket 进行【打开】【读写】【关闭】
三.服务端和客户端的代码
TCP-服务端
1、创建一个socket,用函数socket();
2、绑定IP地址、端口等信息到socket上,用函数bind();
3、开启监听,用函数listen();
4、接收客户端传来的连接,用函数accept();
5、收发数据,用函数send()和recv(),或者read()和write();
6、关闭网络连接;
7、关闭监听;
举例:
#!/usr/bin/python3
# -*- coding: UTF-8 -*-
#导入模块
from socket import *
from time import ctime`在这里插入代码片`
#获取本地主机名
HOST = 'localhost'
#端口自己取,最好取几万以上
PORT = 21567
#缓存设置为1024
BUFSIZ =1024
ADDR = (HOST,PORT)
#创建TCP连接
tcpserSock = socket(AF_INET,SOCK_STREAM)#1
#绑定地址(包括ip地址和端口号)
tcpserSock.bind(ADDR)#2
#创建监听,设置最大允许连接数,各连接和server的通信遵循FIFO原则
tcpserSock.listen(5)#3
#while循环是为了能让对话一直进行,直到客户端输入quit
while True:
print ('waiting for connection...')
#等待客户端的连接
#注意:accept()函数会返回一个元组
#元素1为客户端的socket对象,元素2为客户端的地址(ip地址,端口号)
tcpCliSock, addr = tcpserSock.accept()#4
print ('...connected from:',addr)
while True:
#接受客户端的请求
data = tcpCliSock.recv(BUFSIZ)#5
if data.lower() == ‘quit’:
break
if not data:
break
tcpCliSock.send('[%s]%s' %(ctime(),data))
tcpCliSock.close()#6
tcpserSock.close()#7
TCP-客户端
1、创建一个socket,用函数socket();
2、绑定IP地址、端口等信息到socket上,用函数bind();* 可选
3、设置要连接的对方的IP地址和端口等属性;
4、连接服务器,用函数connect();
5、收发数据,用函数send()和recv(),或者read()和write();
6、关闭网络连接;
from socket import *
from time import ctime
HOST = '192.168.0.104'#目的IP地址
PORT = 21567
BUFSIZ = 1024
ADDR = (HOST, PORT)
#创建一个客户端的socket对象
tcpCliSock = socket(AF_INET,SOCK_STREAM)
#连接服务器
tcpCliSock.connect(ADDR)
#while循环是为了保证能持续进行对话
while True:
#输入发送的消息
data = raw_input('>')
if data.lower() == ‘quit’:
break
if not data:
break
#将信息发送给服务器
tcpCliSock.send(data)
#输入发送的消息
#接收服务器传来的消息
data = tcpCliSock.recv(BUFSIZ)
if data.lower() == ‘quit’:
break
if not data:
break
print (data)
tcpCliSock.close()
四.socket对象
sk = socket.socket(socket.AF_INET,socket.SOCK_STREAM,0)
参数一:地址簇
参数 | 描述 |
---|---|
socket.AF_INET | IPv4(默认) |
socket.AF_INET6 | IPv6 |
ocket.AF_UNIX | 只能够用于单一的Unix系统进程间通信 |
参数二:类型
参数 | 描述 |
---|---|
socket.SOCK_STREAM | 流式socket , for TCP (默认) |
socket.SOCK_DGRAM | 数据报式socket , for UDP |
socket.SOCK_RAW | 原始套接字,普通的套接字无法处理ICMP、IGMP等网络报文,而SOCK_RAW可以;其次,SOCK_RAW也可以处理特殊的IPv4报文;此外,利用原始套接字,可以通过IP_HDRINCL套接字选项由用户构造IP头。 |
socket.SOCK_RDM | 是一种可靠的UDP形式,即保证交付数据报但不保证顺序。SOCK_RAM用来提供对原始协议的低级访问,在需要执行某些特殊操作时使用,如发送ICMP报文。SOCK_RAM通常仅限于高级用户或管理员运行的程序使用。 |
socket.SOCK_SEQPACKET | 可靠的连续数据包服务 |
Socket类方法
方法 | 描述 |
s.bind(address) | 将套接字绑定到地址。address地址的格式取决于地址族。在AF_INET下,以元组(host,port)的形式表示地址。 |
sk.listen(backlog) | 开始监听传入连接。backlog指定在拒绝连接之前,可以挂起的最大连接数量。 |
sk.setblocking(bool) | 是否阻塞(默认True),如果设置False,那么accept和recv时一旦无数据,则报错。 |
sk.accept() | 接受连接并返回(conn,address),其中conn是新的套接字对象,可以用来接收和发送数据。address是连接客户端的地址。 |
sk.connect(address) | 连接到address处的套接字。一般,address的格式为元组(hostname,port),如果连接出错,返回socket.error错误。 |
sk.connect_ex(address) | 同上,只不过会有返回值,连接成功时返回 0 ,连接失败时候返回编码,例如:10061 |
sk.close() | 关闭套接字连接 |
sk.recv(bufsize[,flag]) | 接受套接字的数据。数据以字符串形式返回,bufsize指定最多可以接收的数量。flag提供有关消息的其他信息,通常可以忽略。 |
sk.recvfrom(bufsize[.flag]) | 与recv()类似,但返回值是(data,address)。其中data是包含接收数据的字符串,address是发送数据的套接字地址。 |
sk.send(string[,flag]) | 将string中的数据发送到连接的套接字。返回值是要发送的字节数量,该数量可能小于string的字节大小。即:可能未将指定内容全部发送。 |
sk.sendall(string[,flag]) | 将string中的数据发送到连接的套接字,但在返回之前会尝试发送所有数据。成功返回None,失败则抛出异常。内部通过递归调用send,将所有内容发送出去。 |
sk.sendto(string[,flag],address) | 将数据发送到套接字,address是形式为(ipaddr,port)的元组,指定远程地址。返回值是发送的字节数。该函数主要用于UDP协议。 |
sk.settimeout(timeout) | 设置套接字操作的超时期,timeout是一个浮点数,单位是秒。值为None表示没有超时期。 |
sk.getpeername() | 返回连接套接字的远程地址。返回值通常是元组(ipaddr,port)。 |
sk.getsockname() | 返回套接字自己的地址。通常是一个元组(ipaddr,port) |
sk.fileno() | 套接字的文件描述符 |