tcp服务端和客户端

使用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>>>>>>>

端 <<<<<<<<<<----------->>>>>>>>>>>>> 端

简单说明

  1. 由客户端发送建立TCP连接的请求报文, 其中报文中包含seq序列号, 是由发送端随机生成的,并且将报文中的SYN字段置为1, 表示需要建立TCP连接 (SYN=1, seq=x, x为随机生成数值)
  2. 由服务端回复客户端发送的TCP连接请求报文, 其中包含seq序列号, 是由回复端随机生成的, 并且将SYN置为1,而且会产生ACK字段, ACK字段数值是在客户端发送过来的序列号seq的基础上加1进行回复,以便客户端收到信息时,知晓自己的TCP建立请求已得到验证 (SYN=1, ACK=x+1, seq=y, y为随机生成数值) 这里的ACK加1可以理解为是确认和谁建立连接
  3. 客户端收到服务端发送的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)
  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值