使用TCP套接字编程可以实现基于TCP/IP协议的面向连接的通信,它分为服务器端和客户端两部分,其主要实现过程如下
服务端:
socket→bind→listen→accept→阻塞直到接收到客户链接请求→read→处理请求→write→close
客户端:
socket→connect→write→read→close
服务器端流程
1.创建socket
socket是一个结构体,被创建在内核中
sockfd=socket(AF_INET,SOCK_STREAM,0); //AF_INT:ipv4, SOCK_STREAM:tcp协议
2.调用bind函数
将socket和地址(包括ip、port)绑定。
需要定义一个结构体地址,以便于将port的主机字节序转化成网络字节序
struct sockaddr_in serveraddr; //地址结构体
bind函数:
bind(sockfd,(struct sockaddr*)&serveraddr,sizeof(serveraddr))
3.listen 监听 将接收到的客户端连接端放入队列
listen(sockfd,10) //第二个参数是队列长度
4.调用accept函数,从队列中获取请求,返回socket描述符
如果无请求将会阻塞,直到获得链接
5.调用io函数和客户端双向通信
6.关闭accept返回的socket
服务器端代码
from socket import *
from time import ctime
HOST = ‘’
PORT = 21567
BUFSIZ = 1024
ADDR = (HOST, PORT)
tcpSerSock = socket(AF_INET, SOCK_STREAM)
tcpSerSock.bind(ADDR)
tcpSerSock.listen(5)
while True:
print ‘waiting for connection…’
tcpCliSock, addr = tcpSerSock.accept()
print ‘…connected from:’, addr
while True:
data = tcpCliSock.recv(BUFSIZ)
if not data:
break
tcpCliSock.send(’[%s] %s’ %(ctime(), data))
tcpCliSock.close()
tcpSerSock.close()
客户端代码
from socket import *
HOST = ‘localhost’
PORT = 21567
BUFSIZ = 1024
ADDR = (HOST, PORT)
tcpCliSock = socket(AF_INET, SOCK_STREAM)
tcpCliSock.connect(ADDR)
while True:
data = raw_input(’> ')
if not data:
break
tcpCliSock.send(data)
data1 = tcpCliSock.recv(BUFSIZ)
if not data1:
break
print data1
tcpCliSock.close()
运行界面
服务器端:
D:\code\ex>python tsTserv.py waiting for connection… …connected from: (‘127.0.0.1’, 2883) waiting for connection… …connected from: (‘127.0.0.1’, 2885) waiting for connection… …connected from: (‘127.0.0.1’, 2889) waiting for connection… …connected from: (‘127.0.0.1’, 2891) waiting for connection… …connected from: (‘127.0.0.1’, 2892) waiting for connection… …connected from: (‘127.0.0.1’, 2893) waiting for connection…
客户端:
D:\code\ex>python tsTclnt.py > 1
[Thu Feb 02 15:52:21 2012] 1 >2
[Thu Feb 02 15:52:22 2012] 2 > 3
[Thu Feb 02 15:52:22 2012] 3 > 5
[Thu Feb 02 15:52:23 2012] 5 > 6
[Thu Feb 02 15:52:24 2012] 6 >
D:\code\ex>
11种状态
LISTEN -------------------- 等待从任何远端TCP 和端口的连接请求。
SYN_SENT --------------- 发送完一个连接请求后等待一个匹配的连接请求。
SYN_RECEIVED -------- 发送连接请求并且接收到匹配的连接请求以后等待连接请求确认。
ESTABLISHED ----------- 表示一个打开的连接,接收到的数据可以被投递给用户。连接的数据传输阶段的正常状态。
FIN_WAIT_1 -------------- 等待远端TCP 的连接终止请求,或者等待之前发送的连接终止请求的确认。
FIN_WAIT_2 -------------- 等待远端TCP 的连接终止请求。
CLOSE_WAIT ------------ 等待本地用户的连接终止请求。
CLOSING ------------------ 等待远端TCP 的连接终止请求确认。
LAST_ACK ---------------- 等待先前发送给远端TCP 的连接终止请求的确认(包括它字节的连接终止请求的确认)
TIME_WAIT --------------- 等待足够的时间过去以确保远端TCP 接收到它的连接终止请求的确认。
CLOSED ------------------- 不在连接状态(这是为方便描述假想的状态,实际不存在)
TIME_WAIT 两个存在的理由:
1.可靠的实现tcp全双工连接的终止;
2.允许老的重复分节在网络中消逝。
三次握手
置位概念: 根据TCP的包头字段, 存在3个重要的表示ACK, SYN, FIN
ACK: 表示验证字段
SYN: 位数置1, 表示建立TCP连接
FIN:位数置1,表示断开TCP连接
客------------SYN=1,SEQ=X-------->>>>>>服
户<<<<<<<-SYN=1,ACK=X+1,SEQ=Y----务
-------SYN=1,ACK=Y+1,SEQ=X+1>>>>>>>
端 <<<<<<<<<<----------->>>>>>>>>>>>> 端
简单说明
- 由客户端发送建立TCP连接的请求报文, 其中报文中包含seq序列号, 是由发送端随机生成的,并且将报文中的SYN字段置为1, 表示需要建立TCP连接 (SYN=1, seq=x, x为随机生成数值)
- 由服务端回复客户端发送的TCP连接请求报文, 其中包含seq序列号, 是由回复端随机生成的, 并且将SYN置为1,而且会产生ACK字段, ACK字段数值是在客户端发送过来的序列号seq的基础上加1进行回复,以便客户端收到信息时,知晓自己的TCP建立请求已得到验证 (SYN=1, ACK=x+1, seq=y, y为随机生成数值) 这里的ACK加1可以理解为是确认和谁建立连接
- 客户端收到服务端发送的TCP建立验证请求后, 会使自己的序列号加1表示, 并且再次回复ACK验证请求, 在服务端发过来的seq上加1进行回复(SYN=1, ACK=y+1, seq=x+1)3. 客户端收到服务端发送的TCP建立验证请求后, 会使自己的序列号加1表示, 并且再次回复ACK验证请求, 在服务端发过来的seq上加1进行回复(SYN=1, ACK=y+1, seq=x+1)