十五、网络编程

学习目标

  • 说出网络通信和网络编程的概念
  • 说出IP地址和子网掩码的作用
  • 说出网络通信的几种方式
  • 说出端口号的作用
  • 说出TCP/UDP协议的特点
  • 使用socket实现基于UDP的消息发送和接收
  • 使用socket实现基于TCP的服务器和客户端通信

一、网络通信的概念

简单来说,网络是用物理链路将各个孤立的工作站或主机相连在一起,组成数据链路,从而达到资源共享和通信的目的
使用网络的目的,就是为了联通多方然后进行通信,即把数据从一方传递给另一方

二、IP地址

https://blog.csdn.net/weixin_39453325/article/details/83656881 ---- 以太网帧、IP帧、UDP\TCP数据报、http报文结构

生活中的地址指的就是,找到某人或某机关或与其通信的指定地点。在网络编程中,如果一台主机想和另一台主机进行沟通和共享数据,首先要做的第一件事情就是要找到对方。在互联网通信中,我们使用IP地址来查询各个主机
在这里插入图片描述

在这里插入图片描述

三、端口

端口就像一个房子的门,是出入这间房子的必经之路。如果一个程序需要收发网络数据,那么就需要有这样的端口

3.1 端口号

断就是通过端口号来标记的,端口号只有整数,范围是从0到65535.端口号不是随意使用的,而是按照一定的规定进行分配。端口的分类标准有好几种,我们这里不作详细讲解,只介绍一下 知名端口动态端口

3.2 知名端口

知名端口是众所周知的端口号,范围从0到1023,可以理解为,一些常用的功能使用的号码是估计的,好比 电话110、10086、10010一样。一般情况下,如果一个程序需要使用知名端口的需要有root权限

3.3 动态端口号

动态端口号的范围是1024到65535
之所以成为动态端口,是因为它一般不固定分配某种服务,而是动态分配
动态分配是指当一个系统程序或应用程序需要网络通信时,它向主机申请一个端口,主机从可用的端口号中分配一个供它使用
当这个程序关闭时,同时也就释放了所占用的端口号

3.4 端口号的作用

我们知道,一台拥有IP地址的主机可以提供许多服务,比如HTTP(万维网)、FTP(文件传输)、SMTP(电子邮件)等,这些服务完全可以通过1个IP地址来实现。那么,主机是怎样区分不同的网络服务呢?显然不能只靠IP地址,因为IP地址与网络服务的关系是一对多的关系。实际上是通过"IP地址+端口号"来区分不同的服务的。需要注意的是,端口并不是一一对应的。比如你的电脑作为客户机访问一台WWW服务器时,WWW服务器使用 “80” 端口与你的电脑通信,但你的电脑则可能使用 “3457” 这样的端口

四、socket

4.1 不同电脑上的进程之间如何通信

首先解决的问题是如何唯一表示一个进程,否则通信无从谈起!在1台电脑上可以通过进程号(PID)来唯一标识一个进程,但是在网络中这是行不通的。其实 TCP/UDP 协议族已经帮我们解决了这个问题,网络层的"IP"地址可以唯一标识网络中的主机,而传输层的 “协议+端口” 可以唯一标识主机中的应用进程(进程)。这样利用 ip地址、协议、端口 就可以标识网络的进程了,网络中的进程通信就可以利用这个标志与其它进程进行交互
注意

所谓 **进程 ** 指的是:运行的程序以及运行时用到的资源这个整体称之为进程(在讲解多任务编程时进行详细讲解)
所谓 进程间通信指的是:运行的程序之间的数据共享

4.2 什么是socket

socket(简称 套接字)是进程间通信的一种方式,它与其他进程间通信的一个主要不同是:
它能实现不同主机的进程间通信,我们网络上各种各样的服务大都基于 Socket 来完成通信的

五、Python实现网络通信

5.1 udp发送数据

import socket
# 不同电脑之间的通信需要使用socket
# socket可以在不同的电脑间通信;还可以在同一个电脑的不同程序之间通信

# 1. 创建socket,并连接
# AF_INET:表示这个socket是用来进行网络连接
# SOCK_DGRAM:表示连接时一个 udp 连接
s = socket.socket(socket.AF_INET, socket.SPCK_DGRAM)

# 2. 发送数据
# data:要发送的数据,它是二进制的数据
# address:发送给谁,参数是一个元组,元组里有两个元素
# 第0个表示目标的ip地址;第1个表示程序的端口号

# 给 192.168.31.199 这台主机的 9000 端口号上发送 hello
# 端口号:0-65536   0-1024 不要用,系统一些重要的服务在使用
# 找一个空闲的端口号
s.sendto('hello'.encode('utf8'), '192.168.31.199', 9000)

# 3. 关闭socket
s.close()

5.2 udp接受数据

import socket
# 创建一个基于 udp 的网络socket
s = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)

# 绑定IP地址和端口号
s.bind(('192.168.31.199',9000))

# recvfrom 接受数据
content = s.recvfrom(1024)
# print(content)
# 接受到的数据是一个元组,元组里有两个元素
# 第 0 个元素是接受到的数据;第 1 个元素是发送方的 ip 地址 和 端口号组成的元组
data, (ip, pid) = content
print(f'从{ip}地址{pid}端口号接收到了消息,内容是:{data.decode('utf8')}')

s.close()

5.3 TCP客户端

import socket

# 基于tcp协议的socket连接
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)


# 在发送数据之前,必须要先和服务器建立连接
s.connect(('192.168.31.199', 9090)) # 调用 connect 方法连接到服务器
s.send('hello'.encode('utf8'))

# udp 直接使用sendto 发送数据
# s.sendto('hello'.encode('utf8'), ('192.168.31.199', 9090))

s.close()

5.4 TCP服务端

import socket
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)

s.bind(('192.168.31.199'), 9000)

# 参数是多出的监听队列长度,超出队列的连接会提示连接失败
s.listen(128) # 把socket变成一个被动监听的socket

# 接收到的结果是一个元组,元组里有两个元素
# 第 0 个元素是客户端的socket连接,第 1 个元素是客户端的 ip 地址和端口号
client_socket, client_addr = s.accept() # 接收客户端的请求

# udp 里接受数据,使用recvfrom
x = client_socket.recv(1024) # tcp里使用recv获取数据

print(f'接受到了{client_add[0]}客户端{client_add[1]}端口号发送的数据,内容是:{x.decode('utf8')}')

s.close()

5.5 文件下载客户端

import socket

client_socket = socket(socket.AF_NET, socket.SOCK_STREAM)

client_socket.connect(('192.168.31.199', 9000))

file_name = input("请输入您要下载的文件名")
s.send(file_name.encode('utf8'))

with open(file_name,'wb') as file:
    while True:
        content = s.recv(1024)
        if not content:
            break
        file.write(content)


s.close()

5.6 文件下载服务端

import socket, os

server_socket = socket.socket(AF_INET,socket.SOCK_STREAM)
server_socket.bind(('192.168.31.199', 9000))
server_socket.listen(128)

# 接受客户端的请求
client_socket, client_addr = server_socket.accept()
data = client_socket.recv(1024).decode('utf8')
print('接收到了来自{client_addr[0]}地址{client_addr[1]}端口的数据,内容是{data}')

if os.path.isfile(data):
    print('读取文件,返回文件')
    with open(data,mode='rb') as file:
        content = file.read()
        client_socket.send(content)
else:
    print('文件不存在')

s.close()
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

ModelBulider

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

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

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

打赏作者

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

抵扣说明:

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

余额充值