❥不学不会系列❥-- 基于Python实现网络编程临门一弹

一、IP地址

1.IP地址的概念

  • IP 地址是标识网络中设备的一个地址,好比现实生活中的家庭地址
  • 网络中的设备效果图:
    请添加图片描述

2.IP地址的表现形式

在这里插入图片描述

  • IP 地址分为两类: IPv4 和 IPv6
  • IPv4 是目前使用的ip地址
  • IPv6 是未来使用的ip地址
  • IPv4 是由点分十进制组成
  • IPv6 是由冒号十六进制组成

3.IP地址的作用

  • IP 地址的作用是标识网络中唯一的一台设备,也就是说通过IP地址能够找到网络中某台设备
  • IP地址作用效果图:
    在这里插入图片描述

4.查看IP地址

  • Linux 和 mac OS 使用 ifconfig 这个命令
  • Windows 使用 ipconfig 这个命令
    在这里插入图片描述
  • ifconfig 和 ipconfig 都是查看网卡信息,网卡信息包括设备的IP地址
  • 192.168.1.107是设备在网络中的IP地址
  • 127.0.0.1表示本机地址,提示:如果和自己的电脑通信就可以使用该地址
  • 127.0.0.1该地址对应的域名是localhost,域名是 ip 地址的别名,通过域名能解析出一个对应的ip地址

5.检查网络是否正常

  • 检查网络是否正常使用 ping命令
  • 检查网络是否正常效果图:
    在这里插入图片描述
  • ping www.baidu.com 检查是否能上公网
  • ping 当前局域网的ip地址 检查是否在同一个局域网
  • ping 127.0.0.1 检查本地网卡是否正常

二、端口和端口号

1.什么是端口

  • 端口是传输数据的通道,好比教室的门,是数据传输必经之路
  • 那么如何准确找到对应的端口呢?
    其实,每一个端口都有一个对应的端口号,好比每个教室的门都有一个门牌号,想要找到端口通过端口号即可

2.什么是端口号

  • 操作系统为了统一管理这么多端口,就对端口进行了编号,这就是端口号,端口号其实就是一个数字,好比我们现实生活中的门牌号
  • 端口号有65536
  • 那么最终数据通信的流程是这样的,通过ip地址找到对应的设备通过端口号找到对应的端口,然后通过端口把数据传输给应用程序
  • 最终通信流程效果图:
    在这里插入图片描述

3.端口号的分类

  • 知名端口号
  • 动态端口号
  • 知名端口号
  • 知名端口号是指众所周知的端口号,范围从0到1023
  • 这些端口号一般固定分配给一些服务,比如21端口分配给FTP(文件传输协议)服务,25端口分配给SMTP(简单邮件传输协议)服务,80端口分配给HTTP服务
  • 动态端口号
  • 程序员开发应用程序使用的端口号称为动态端口号, 范围是从1024到65535
  • 如果程序员开发的程序没有设置端口号,操作系统会在动态端口号这个范围内随机生成一个给开发的应用程序使用
  • 运行一个程序默认会有一个端口号,当这个程序退出时,所占用的这个端口号就会被释放

三、TCP

1.TCP的概念

  • TCP 的英文全拼(Transmission Control Protocol)简称传输控制协议,它是一种面向连接的可靠的基于字节流的传输层通信协议
  • 面向连接的效果图:
    在这里插入图片描述

2.TCP的通信步骤

  • 1.创建连接
  • 2.传输数据
  • 3.关闭连接
  • TCP 通信模型相当于生活中的’打电话’,在通信开始之前,一定要先建立好连接,才能发送数据,通信结束要关闭连接

3.TCP的特点

  • 面向连接
    • 通信双方必须先建立好连接才能进行数据的传输,数据传输完成后,双方必须断开此连接,以释放系统资源
  • 可靠传输
    • TCP 采用发送应答机制
    • 超时重传
    • 错误校验
    • 流量控制和阻塞管理

四、socket

  • socket(简称 套接字) 是进程间网络数据通信的工具,好比现实生活中的插座,所有的家用电器要想工作都是基于插座进行,进程之间想要进行网络通信需要基于这个 socket
  • socket 效果图:
    在这里插入图片描述

五、TCP网络应用程序开发流程

  • TCP 网络应用程序开发分为:
    • TCP 客户端程序开发
    • TCP 服务端程序开发
  • 客户端程序是指运行在用户设备上的程序
  • 服务端程序是指运行在服务器设备上的程序,专门为客户端提供数据服务
  • TCP 客户端程序开发流程

在这里插入图片描述
步骤说明:

  • 创建客户端套接字对象
  • 和服务端套接字建立连接
  • 发送数据
  • 接收数据
  • 关闭客户端套接字
  • TCP 服务端程序开发流程

在这里插入图片描述
步骤说明:

  • 创建服务端套接字对象
  • 绑定端口号
  • 设置监听
  • 等待接受客户端的连接请求
  • 接收数据
  • 发送数据
  • 关闭套接字

六、TCP客户端程序开发

1.socket类的介绍

  • 导入 socket 模块 import socket
  • 创建客户端 socket 对象 socket.socket(AddressFamily, Type)
  • 参数说明:
    • AddressFamily 表示IP地址类型, 分为IPv4和IPv6
    • Type 表示传输协议类型
  • 方法说明:
    • connect((host, port)) 表示和服务端套接字建立连接, host是服务器ip地址,port是应用程序的端口号
    • send(data) 表示发送数据,data是二进制数据
    • recv(buffersize) 表示接收数据, buffersize是每次接收数据的长度

2.TCP客户端程序示例代码

import socket


if __name__ == '__main__':
    # 创建tcp客户端套接字
    # 1. AF_INET:表示ipv4
    # 2. SOCK_STREAM: tcp传输协议
    tcp_client_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
    # 和服务端应用程序建立连接
    tcp_client_socket.connect(("192.168.131.62", 8080))
    # 代码执行到此,说明连接建立成功
    # 准备发送的数据
    send_data = "你好服务端,我是客户端小黑!".encode("gbk")
    # 发送数据
    tcp_client_socket.send(send_data)
    # 接收数据, 这次接收的数据最大字节数是1024
    recv_data = tcp_client_socket.recv(1024)
    # 返回的直接是服务端程序发送的二进制数据
    print(recv_data)
    # 对数据进行解码
    recv_content = recv_data.decode("gbk")
    print("接收服务端的数据为:", recv_content)
    # 关闭套接字
    tcp_client_socket.close()

# 执行结果
b'hello'
接收服务端的数据为: hello

七、TCP服务端程序开发

1.socket类的介绍

  • 导入 socket 模块 import socket
  • 创建服务端 socket 对象 socket.socket(AddressFamily, Type)
  • 参数说明:
    • AddressFamily 表示IP地址类型, 分为IPv4和IPv6
    • Type 表示传输协议类型
  • 方法说明:
    • bind((host, port)) 表示绑定端口号, host 是 ip 地址,port 是端口号,ip 地址一般不指定,表示本机的任何一个ip地址都可以
    • listen (backlog) 表示设置监听,backlog参数表示最大等待建立连接的个数
    • accept() 表示等待接受客户端的连接请求
    • send(data) 表示发送数据,data是二进制数据
    • recv(buffersize) 表示接收数据, buffersize是每次接收数据的长度

2.TCP服务端程序示例代码

import socket

if __name__ == '__main__':
    # 创建tcp服务端套接字
    tcp_server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
    # 设置端口号复用,让程序退出端口号立即释放
    tcp_server_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, True) 
    # 给程序绑定端口号
    tcp_server_socket.bind(("", 8989))
    # 设置监听
    # 128:最大等待建立连接的个数, 提示: 目前是单任务的服务端,同一时刻只能服务一个客户端,后续使用多任务能够让服务端同时服务多个客户端,
    # 不需要让客户端进行等待建立连接
    # listen后的这个套接字只负责接收客户端连接请求,不能收发消息,收发消息使用返回的这个新套接字来完成
    tcp_server_socket.listen(128)
    # 等待客户端建立连接的请求, 只有客户端和服务端建立连接成功代码才会解阻塞,代码才能继续往下执行
    # 1. 专门和客户端通信的套接字: service_client_socket
    # 2. 客户端的ip地址和端口号: ip_port
    service_client_socket, ip_port = tcp_server_socket.accept()
    # 代码执行到此说明连接建立成功
    print("客户端的ip地址和端口号:", ip_port)
    # 接收客户端发送的数据, 这次接收数据的最大字节数是1024
    recv_data = service_client_socket.recv(1024)
    # 获取数据的长度
    recv_data_length = len(recv_data)
    print("接收数据的长度为:", recv_data_length)
    # 对二进制数据进行解码
    recv_content = recv_data.decode("gbk")
    print("接收客户端的数据为:", recv_content)
    # 准备发送的数据
    send_data = "ok, 问题正在处理中...".encode("gbk")
    # 发送数据给客户端
    service_client_socket.send(send_data)
    # 关闭服务与客户端的套接字, 终止和客户端通信的服务
    service_client_socket.close()
    # 关闭服务端的套接字, 终止和客户端提供建立连接请求的服务
    tcp_server_socket.close()

# 执行结果
客户端的ip地址和端口号: ('172.16.47.209', 52472)
接收数据的长度为: 5
接收客户端的数据为: hello

八、TCP网络应用程序的注意点

  • 当 TCP 客户端程序想要和 TCP 服务端程序进行通信的时候必须要先建立连接
  • TCP 客户端程序一般不需要绑定端口号,因为客户端是主动发起建立连接的
  • TCP 服务端程序必须绑定端口号,否则客户端找不到这个 TCP 服务端程序
  • listen 后的套接字是被动套接字,只负责接收新的客户端的连接请求,不能收发消息
  • 当 TCP 客户端程序和 TCP 服务端程序连接成功后, TCP 服务器端程序会产生一个新的套接字,收发客户端消息使用该套接字
  • 关闭 accept 返回的套接字意味着和这个客户端已经通信完毕
  • 关闭 listen 后的套接字意味着服务端的套接字关闭了,会导致新的客户端不能连接服务端,但是之前已经接成功的客户端还能正常通信
  • 当客户端的套接字调用 close 后,服务器端的 recv 会解阻塞,返回的数据长度为0,服务端可以通过返回数据的长度来判断客户端是否已经下线,反之服务端关闭套接字,客户端的 recv 也会解阻塞,返回的数据长度也为0

九、案列-多任务版TCP服务端程序

1.需求

  • 前面的TCP服务端程序只能服务于一个客户端,如何开发一个多任务版的TCP服务端程序能够服务于多个客户端呢?
  • 完成多任务,可以使用线程,比进程更加节省内存资源

2.实现步骤

  • 编写一个TCP服务端程序,循环等待接受客户端的连接请求
  • 当客户端和服务端建立连接成功,创建子线程,使用子线程专门处理客户端的请求,防止主线程阻塞
  • 把创建的子线程设置成为守护主线程,防止主线程无法退出

3.多任务版TCP服务端程序的示例代码

import socket
import threading


# 处理客户端的请求操作
def handle_client_request(service_client_socket, ip_port):
    # 循环接收客户端发送的数据
    while True:
        # 接收客户端发送的数据
        recv_data = service_client_socket.recv(1024)
        # 容器类型判断是否有数据,可以直接使用if语句进行判断,如果容器类型里面有数据表示条件成立,否则条件失败
        # 容器类型: 列表、字典、元组、字符串、set、range、二进制数据
        if recv_data:
            print(recv_data.decode("gbk"), ip_port)
            # 回复
            service_client_socket.send("ok,问题正在处理中...".encode("gbk"))

        else:
            print("客户端下线了:", ip_port)
            break
    # 终止和客户端进行通信
    service_client_socket.close()


if __name__ == '__main__':
    # 创建tcp服务端套接字
    tcp_server_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
    # 设置端口号复用,让程序退出端口号立即释放
    tcp_server_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, True)
    # 绑定端口号
    tcp_server_socket.bind(("", 9090))
    # 设置监听, listen后的套接字是被动套接字,只负责接收客户端的连接请求
    tcp_server_socket.listen(128)
    # 循环等待接收客户端的连接请求
    while True:
        # 等待接收客户端的连接请求
        service_client_socket, ip_port = tcp_server_socket.accept()
        print("客户端连接成功:", ip_port)
        # 当客户端和服务端建立连接成功以后,需要创建一个子线程,不同子线程负责接收不同客户端的消息
        sub_thread = threading.Thread(target=handle_client_request, args=(service_client_socket, ip_port))
        # 设置守护主线程
        sub_thread.setDaemon(True)
        # 启动子线程
        sub_thread.start()
    # tcp服务端套接字可以不需要关闭,因为服务端程序需要一直运行
    # tcp_server_socket.close()

# 执行结果
客户端连接成功: ('172.16.47.209', 51528)
客户端连接成功: ('172.16.47.209', 51714)
hello1 ('172.16.47.209', 51528)
hello2 ('172.16.47.209', 51714)

十、socket之send和recv原理剖析

1.socket的发送和接收缓冲区

当创建一个TCP socket对象的时候会有一个发送缓冲区和一个接收缓冲区,这个发送和接收缓冲区指的就是内存中的一片空间

2.send原理剖析

  • send是不是直接把数据发给服务端?
  • 不是,要想发数据,必须得通过网卡发送数据,应用程序是无法直接通过网卡发送数据的,它需要调用操作系统接口,也就是说,应用程序把发送的数据先写入到发送缓冲区(内存中的一片空间),再由操作系统控制网卡把发送缓冲区的数据发送给服务端网卡

3.recv原理剖析

  • recv是不是直接从客户端接收数据?
  • 不是,应用软件是无法直接通过网卡接收数据的,它需要调用操作系统接口,由操作系统通过网卡接收数据,把接收的数据写入到接收缓冲区(内存中的一片空间),应用程序再从接收缓存区获取客户端发送的数据

4.send和recv原理剖析图

在这里插入图片描述
不管是recv还是send,都不是直接接收到对方的数据和发送数据到对方,发送数据会写入到发送缓冲区,接收数据是从接收缓冲区来读取,发送数据和接收数据最终是由操作系统控制网卡来完成

  • 2
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 3
    评论
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

「已注销」

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值