基于网络编程就是需要客户端和服务端,客户端和服务端都是工作于应用层,要是想要学会所有的协议是很困难的,于是就诞生了socket套接字: Socket是应用层与TCP/IP协议族通信的中间软件抽象层,它是一组接口。在设计模式中,Socket其实就是一个门面模式,它把复杂的TCP/IP协议族隐藏在Socket接口后面,对用户来说,一组简单的接口就是全部,让Socket去组织数据,以符合指定的协议。 我们无需深入理解tcp/udp协议,socket已经为我们封装好了,我们只需要遵循socket的规定去编程,写出的程序自然就是遵循tcp/udp标准的。
.png)
套接字的发展史:套接字分为两种类型1. AF_UNIX;2. AF_INET
基于文件类型的套接字家族
套接字家族的名字:AF_UNIX
unix一切皆文件,基于文件的套接字调用的就是底层的文件系统来取数据,两个套接字进程运行在同一机器,可以通过访问同一个文件系统间接完成通信
基于网络类型的套接字家族(使用这种类型)
套接字家族的名字:AF_INET
(还有AF_INET6被用于ipv6,还有一些其他的地址家族,不过,他们要么是只用于某个平台,要么就是已经被废弃,或者是很少被使用,或者是根本没有实现,所有地址家族中,AF_INET是使用最广泛的一个,python支持很多种地址家族,但是由于我们只关心网络编程,所以大部分时候我么只使用AF_INET)
套接字的工作流程
先从服务器端说起。服务器端先初始化Socket,然后与端口绑定(bind),对端口进行监听(listen),调用accept阻塞,等待客户端连接。在这时如果有个客户端初始化一个Socket,然后连接服务器(connect),如果连接成功,这时客户端与服务器端的连接就建立了。客户端发送数据请求,服务器端接收请求并处理请求,然后把回应数据发送给客户端,客户端读取数据,最后关闭连接,一次交互结束
6.基于socket实现简单套接字通信
服务端
import socket
# 类比打电话的流程
# 1.买手机
phone = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
# AF_INET是指定的套接字的类型是网络类型,SOCK_STREAM是指定套接字的传输方式是流式传输
# 2.绑定手机卡
phone.bind(('127.0.0.1',8080))
# 端口随便设置,需要注意的是端口是0-65535,但是其中的0-1024是给操作系统使用的
# 3.开机
phone.listen(5)
# listen后的是可挂起的连接数
# 4.等电话链接
# res = phone.accept()
coon, client_addr = phone.accept()
# (<socket.socket fd=240, family=AddressFamily.AF_INET, type=SocketKind.SOCK_STREAM, proto=0, laddr=('127.0.0.1', 8080), raddr=('127.0.0.1', 64565)>,
# ('127.0.0.1', 64565))
# res拿到的就是元祖的形式,第一个元素是套接字对象,第二个元素是客户端的IP和端口
# 5.收发消息
data = coon.recv(1024)
print('客户端的数据', data)
# 接收1024个字节,接收数据的最大数,1024是接收字节的最大限制(1.单位是bytes;2.最大接收1024)
coon.send(data.upper())
# 6.挂电话
coon.close()
# 7.关机
phone.close()
客户端
import socket
phone = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
# phone.bind(('127.0.1.1',8080))
phone.connect(('127.0.0.1',8080))
# 发、收消息
phone.send('hello'.encode('utf-8'))
data = phone.recv(1024)
print(data)
# 关闭
phone.close()
7.在简单套接字基础上加通信循环
服务端
import socket
# 类比打电话的流程
# 1.买手机
phone = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
# AF_INET是指定的套接字的类型是网络类型,SOCK_STREAM是指定套接字的传输方式是流式传输
# 2.绑定手机卡
phone.bind(('127.0.0.1',8080))
# 端口随便设置,需要注意的是端口是0-65535,但是其中的0-1024是给操作系统使用的
# 3.开机
phone.listen(5)
# listen后的是可挂起的连接数
# 4.等电话链接
# res = phone.accept()
# (<socket.socket fd=240, family=AddressFamily.AF_INET, type=SocketKind.SOCK_STREAM, proto=0, laddr=('127.0.0.1', 8080), raddr=('127.0.0.1', 64565)>,
# ('127.0.0.1', 64565))
# res拿到的就是元祖的形式,第一个元素是套接字对象,第二个元素是客户端的IP和端口
# 5.收发消息
while True:
coon, client_addr = phone.accept()
data = coon.recv(1024)
print('客户端的数据', data)
# 接收1024个字节,接收数据的最大数,1024是接收字节的最大限制(1.单位是bytes;2.最大接收1024)
coon.send(data.upper())
# 6.挂电话
coon.close()
# 7.关机
phone.close()
客户端
import socket
phone = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
# phone.bind(('127.0.1.1',8080))
phone.connect(('127.0.0.1',8080))
# 发、收消息
while True:
msg = input('>>>').strip()
phone.send(msg.encode('utf-8'))
data = phone.recv(1024)
print(data)
# 关闭
phone.close()
socket.send()实际上还是发给了自己的操作系统,操作系统再调用自己的网卡,应用程序来组织数据,操作系统操作TCP协议开始封装数据向下发包,软件的操作方式是应用级别的,操作系统完成应用层以下的4层封装打包
如果强行在client运行时关闭的话,在linux上服务端会一直循环,卡在recv接收处,windows则是直接报错
8.客户端与服务端代码bug修复
服务端可以并发的处理请求(没学到并发的情况下就是不断的循环),可以修改让服务端为客户端一个一个执行
服务端
import socket
# 类比打电话的流程
# 1.买手机
phone = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
phone.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)#可重复使用IP端口号
# AF_INET是指定的套接字的类型是网络类型,SOCK_STREAM是指定套接字的传输方式是流式传输
# 2.绑定手机卡
phone.bind(('127.0.0.1',8080))
# 端口随便设置,需要注意的是端口是0-65535,但是其中的0-1024是给操作系统使用的
# 3.开机
phone.listen(5)
# listen后的是可挂起的连接数
# 4.等电话链接
# res = phone.accept()
# (<socket.socket fd=240, family=AddressFamily.AF_INET, type=SocketKind.SOCK_STREAM, proto=0, laddr=('127.0.0.1', 8080), raddr=('127.0.0.1', 64565)>,
# ('127.0.0.1', 64565))
# res拿到的就是元祖的形式,第一个元素是套接字对象,第二个元素是客户端的IP和端口
# 5.收发消息
coon, client_addr = phone.accept()
while True:
try:
data = coon.recv(1024)
# if not data: break #linux上
print('客户端的数据', data)
# 接收1024个字节,接收数据的最大数,1024是接收字节的最大限制(1.单位是bytes;2.最大接收1024)
coon.send(data.upper())
except ConnectionResetError:
break
# 6.挂电话
coon.close()
# 7.关机
phone.close()
客户端
import socket
phone = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
# phone.bind(('127.0.1.1',8080))
phone.connect(('127.0.0.1',8080))
# 发、收消息
while True:
msg = input('>>>').strip()
if not msg:
continue
phone.send(msg.encode('utf-8'))
data = phone.recv(1024)
print(data.decode('utf-8'))
# 关闭
phone.close()
9.实现服务端可以对多个客户端提供服务
服务端
import socket
# 类比打电话的流程
# 1.买手机
phone = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
phone.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1)#可重复使用IP端口号
# AF_INET是指定的套接字的类型是网络类型,SOCK_STREAM是指定套接字的传输方式是流式传输
# 2.绑定手机卡
phone.bind(('127.0.0.1',8080))
# 端口随便设置,需要注意的是端口是0-65535,但是其中的0-1024是给操作系统使用的
# 3.开机
phone.listen(5)
# listen后的是可挂起的连接数
# 4.等电话链接
# res = phone.accept()
# (<socket.socket fd=240, family=AddressFamily.AF_INET, type=SocketKind.SOCK_STREAM, proto=0, laddr=('127.0.0.1', 8080), raddr=('127.0.0.1', 64565)>,
# ('127.0.0.1', 64565))
# res拿到的就是元祖的形式,第一个元素是套接字对象,第二个元素是客户端的IP和端口
# 5.收发消息
while True:
coon, client_addr = phone.accept()
print(client_addr)
while True:
try:
data = coon.recv(1024)
# if not data: break #linux上
print('客户端的数据', data)
# 接收1024个字节,接收数据的最大数,1024是接收字节的最大限制(1.单位是bytes;2.最大接收1024)
coon.send(data.upper())
except ConnectionResetError:
break
# 6.挂电话
coon.close()
# 7.关机
phone.close()
客户端
import socket
phone = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
# phone.bind(('127.0.1.1',8080))
phone.connect(('127.0.0.1',8080))
# 发、收消息
while True:
msg = input('>>>').strip()
if not msg:
continue
phone.send(msg.encode('utf-8'))
data = phone.recv(1024)
print(data.decode('utf-8'))
# 关闭
phone.close()
10.模拟ssh远程执行命令-项目分析
ssh就是一个基于套接字写的客户端、服务端架构的软件
公司通常都是使用linux,有钱的用unix,没钱的使用centOS这样的服务器,windows的操作系统只是针对用户的体验使用的

11.模拟ssh远程执行命令-代码实现
12.粘包现象
13.粘包底层原理分析
14.解决粘包问题-伪代码实现
15.解决粘包问题-简单版本
16.解决粘包问题-终极版本
17.文件传输功能实现
18.文件传输功能-函数版
19.文件传输功能-面向对象版
20.基于udp协议的套接字介绍
- 路飞学城—Python爬虫实战密训班 第三章
路飞学城—Python爬虫实战密训班 第三章 一.scrapy-redis插件实现简单分布式爬虫 scrapy-redis插件用于将scrapy和redis结合实现简单分布式爬虫: - 定义调度器 - ...
- 路飞学城—Python爬虫实战密训班 第二章
路飞学城—Python爬虫实战密训班 第二章 一.Selenium基础 Selenium是一个第三方模块,可以完全模拟用户在浏览器上操作(相当于在浏览器上点点点). 1.安装 - pip instal ...
- 路飞学城Python爬虫课第一章笔记
前言 原创文章,转载引用务必注明链接.水平有限,如有疏漏,欢迎指正. 之前看阮一峰的博客文章,介绍到路飞学城爬虫课程限免,看了眼内容还不错,就兴冲冲报了名,99块钱满足以下条件会返还并送书送视频. 缴 ...
- 路飞学城-Python开发集训-第3章
学习心得: 通过这一章的作业,使我对正则表达式的使用直接提升了一个level,虽然作业完成的不怎么样,重复代码有点多,但是收获还是非常大的,有点找到写代码的感觉了,遗憾的是,这次作业交过,这次集训就结 ...
- 路飞学城-Python开发集训-第1章
学习体会: 在参加这次集训之前我自己学过一段时间的Python,看过老男孩的免费视频,自我感觉还行,老师写的代码基本上都能看懂,但是实际呢?....今天是集训第一次交作业的时间,突然发现看似简单升级需 ...
- 路飞学城-Python开发集训-第4章
学习心得: 学习笔记: 在python中一个py文件就是一个模块 模块好处: 1.提高可维护性 2.可重用 3.避免函数名和变量名冲突 模块分为三种: 1.内置标准模块(标准库),查看所有自带和第三方 ...
- 路飞学城-Python开发集训-第2章
学习心得: 这章对编码的讲解超级赞,现在对于编码终于有一点认知了,但还没有大彻大悟,还需要更加细心的琢磨一下Alex博客和视频,以前真的是被编码折磨死了,因为编码的问题而浪费的时间很多很多,现在终于感 ...
- 路飞学城-Python开发-第二章
''' 数据结构: menu = { '北京':{ '海淀':{ '五道口':{ 'soho':{}, '网易':{}, 'google':{} }, '中关村':{ '爱奇艺':{}, '汽车之家' ...
- 路飞学城-Python开发-第三章
# 数据结构: # goods = [ # {"name": "电脑", "price": 1999}, # {"name&quo ...
- 路飞学城-Python开发-第一章
# 基础需求: # 让用户输入用户名密码 # 认证成功后显示欢迎信息 # 输错三次后退出程序 username = 'pandaboy' password = ' def Login(username ...
随机推荐
- Python笔记15------图像
主要三个库:Pilow(PIL).OpenCV.Skimage(针对scipy,用的少) 小例子:给一张图片的左上角粘贴一个相同的图片(缩略并旋转了45度) from PIL import Image ...
- 洛谷P1428 小鱼比可爱
题目描述 人比人,气死人:鱼比鱼,难死鱼.小鱼最近参加了一个"比可爱"比赛,比的是每只鱼的可爱程度.参赛的鱼被从左到右排成一排,头都朝向左边,然后每只鱼会得到一个整数数值,表示这只 ...
- [luogu2272 ZJOI2007] 最大半连通子图 (tarjan缩点 拓扑排序 dp)
传送门 题目描述 一个有向图G=(V,E)称为半连通的(Semi-Connected),如果满足:?u,v∈V,满足u→v或v→u,即对于图中任意两点u,v,存在一条u到v的有向路径或者从v到u的有向 ...
- echarts地图的基本使用配置
一.空气质量图 代码和配置如下: <template> <div class="box"> <div id="map">&l ...
- Elasticsearch Sliced Scroll分页检索案例分享
面试:你懂什么是分布式系统吗?Redis分布式锁都不会?>>> The best elasticsearch highlevel java rest api-----bboss ...
- ASP.NET-signalR学习笔记
- BA-siemens-insight_lenum点
lenum点特性 lenum点有如下特点 如果状态字是自定义的,只能在bacnet / ip的aln层使用 如果想在ms/tp层使用lenum的功能,就必须将system profile中bacnet ...
- Tom和Jerry来了,Tom和Jerry走了——北漂18年(38)
上次讲到跟我同一时候入职的女销售走了. 回忆起来,她的问题多半是技巧足够,脑子不足够,走了之后再没联系.不久之后,在老板的要求之下.LilyG又招聘了两位男销售,英文名字非常登对一个叫Tom,一个叫J ...
- do while 循环和while循环的差别
do while 循环和while循环的差别 1.do while循环是先运行循环体,然后推断循环条件,假设为真,则运行下一步循环,否则终止循环. while循环是先推断循环条件,假设条件为真则 ...
- Linux体验之旅(一)——制作U启,安装rhel-server-6.3
U启制作: 双击UltraISO: 点击文件→打开: 选择rhel-server6.3 点击启动→选择写入硬盘映像 最后选择格式化优盘→写入→完毕 注意:启动盘制作完毕后一定记得将rhel-serve ...