TCP/IP四层协议

TCP/IP协议 socket-冯金伟博客园

TCP/IP概念

tcp/ip协议是主机接入互联网以及接入互联网的两台主机通信的标准。

数据帧概念

数据帧
  |-- 包头
  |     |--源地址
  |     |--目标地址
  |     |--数据类型
  |
  |-- 数据

 socket在四层协议中的位置

TCP/IP协议 socket-冯金伟博客园

socket协议的交互流程

TCP/IP协议 socket-冯金伟博客园

socket对象(内建)方法

    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.setsockopt(level,optname,value) 	设置给定套接字选项的值。
s.getsockopt(level,optname[.buflen]) 	返回套接字选项的值。
s.settimeout(timeout) 	设置套接字操作的超时期,timeout是一个浮点数,单位是秒。值为None表示没有超时期。一般,超时期应该在刚创建套接字时设置,因为它们可能用于连接的操作(如connect())
s.gettimeout() 	返回当前超时期的值,单位是秒,如果没有设置超时期,则返回None。
s.fileno() 	返回套接字的文件描述符。
s.setblocking(flag) 	如果flag为0,则将套接字设为非阻塞模式,否则将套接字设为阻塞模式(默认值)。非阻塞模式下,如果调用recv()没有发现任何数据,或send()调用无法立即发送数据,那么将引起socket.error异常。
s.makefile() 	创建一个与该套接字相关连的文件

so基于tcp/ip 套接字通讯 server and client

服务端:

#1、 依照上socket流程图,实现一个功能,客户端输入什么,服务端就把输入的转为大写传送给客户端。
#2、客户端exit退出,不影响服务端的运行。
# soceet server

import socket
ip_port=('127.0.0.1',9999)

# 封装协议(对象)
s = socket.socket()

# 绑定ip,端口
s.bind(ip_port)

# 启动监听
s.listen(5)  # 挂起连接数,  允许最多处理5个请求
while True:
    # 等待连接
    conn, addr = s.accept()  # accept方法等待客户端连接,直到有客户端连接后,会返回连接线(conn)、连接地址(addr)

    while True:
        # 至此,当客户端连接时,conn即为连接客户端的连接线
        # 所以,当客户端主动断开连接时,conn就不存在了,也会抛出异常
        # 在这里定义一个异常跟踪
        try:
            # 接收消息
            recv_data=conn.recv(1024)  # 接收conn连接线路,并指定缓存该线路的1024
            print('客户端发来的消息是', data.decode('utf-8'))

            # 发送消息
            send_data=recv_data.upper()  # 将接收消息转换为大写
            conn.send(send_data)  # 使用conn线路,发送消息
        except Exception:  # 如果客户端主动断开,则server退出该循环等待下一条连接
            break

    # 结束进程
    conn.close() # 中断线路

 客户端:

# socket client

import socket
ip_port=('127.0.0.1',9999)

# 封装协议(对象)
s = socket.socket()

# 向服务端建立连接
s.connect(ip_port)

while True:
    msg = input('--->: ')

    if not msg:continue ###如果输入回车则执行continue返回上层继续执行。

    client.send(msg.encode('utf-8')) ###发送一个msg到服务端

    print('客户端已经发送消息')

    if msg == 'exit': break  ###如果msg=exit的时候执行break退出整个循环。

    data = client.recv(1024) ##client接收到的server消息,

    print('客户端收到server消息',data.decode('utf-8'))  ##打印出来收到的消息用decode解码。

client.close() ##结束连接

 注意:

TCP/IP协议 socket-冯金伟博客园

解决:

#1、加入一条socket配置,重用ip和端口

phone=socket(AF_INET,SOCK_STREAM)
phone.setsockopt(SOL_SOCKET,SO_REUSEADDR,1) #就是它,在bind前加
phone.bind(('127.0.0.1',8080))

#此条可以解决上述的问题
#2、发现系统存在大量TIME_WAIT状态的连接,通过调整linux内核参数解决,
vi /etc/sysctl.conf
编辑文件,加入以下内容:
net.ipv4.tcp_syncookies = 1
net.ipv4.tcp_tw_reuse = 1
net.ipv4.tcp_tw_recycle = 1
net.ipv4.tcp_fin_timeout = 30
 
然后执行 /sbin/sysctl -p 让参数生效。
 
net.ipv4.tcp_syncookies = 1 表示开启SYN Cookies。当出现SYN等待队列溢出时,启用cookies来处理,可防范少量SYN攻击,默认为0,表示关闭;

net.ipv4.tcp_tw_reuse = 1 表示开启重用。允许将TIME-WAIT sockets重新用于新的TCP连接,默认为0,表示关闭;

net.ipv4.tcp_tw_recycle = 1 表示开启TCP连接中TIME-WAIT sockets的快速回收,默认为0,表示关闭。

net.ipv4.tcp_fin_timeout 修改系統默认的 TIMEOUT 时间