Linux服务——http协议及nginx服务

news2024/11/18 7:31:56

目录

一、HTTP协议

1、跨网络的主机间通讯方式

套接字相关的系统调用

2、HTTP协议访问网站的过程

3、http协议状态码分类

常见的http协议状态码

4、MIME

5、URL组成

6、HTTP协议版本

7、系统处理http请求的工作模式

8、apache与nginx的区别

 二、I/O模型

I/O模型相关概念

事件驱动模型

三、Nginx服务

1、nginx模块

2、安装nginx

yum安装nginx

编译安装nginx

3、nginx信号

4、nginx服务调优

1)根据cpu数量配置master有多少个worker子进程

2) 关闭或修改版本

3)cpu与work进程绑定

4)nginx进程的优先级(work进程的优先级) 

5)设置nginx可打开的文件个数 


一、HTTP协议

1、跨网络的主机间通讯方式

跨网络的主机间通讯方式:套接字

套接字Socket是进程间通信IPC的一种实现,允许位于不同主机(或同一主机)上不同进程之间进行通信和数据交换,SocketAPI出现于1983年 BSD 4.2实现在建立通信连接的每一端,进程间的传输要有两个标志:IP地址和端口号,合称为套接字地址 socket address

  • 客户机套接字地址定义了一个唯一的客户进程

  • 服务器套接字地址定义了一个唯一的服务器进程

套接字相关的系统调用

  • socket() 创建一个套接字

  • bind() 绑定IP和端口

  • listen() 监听

  • accept() 接收请求

  • connect() 请求连接建立

  • write() 发送

  • read() 接收

  • close() 关闭连接

2、HTTP协议访问网站的过程

建立连接---->接收请求---->处理请求---->访问资源---->构建响应报文---->发送响应报文---->记录日志

3、http协议状态码分类

1xx:100-101 信息提示
2xx:200-206 成功
3xx:300-307 重定向
4xx:400-415 错误类信息,客户端错误
5xx:500-505 错误类信息,服务器端错误

常见的http协议状态码

200: 成功,请求数据通过响应报文的entity-body部分发送;OK
301: 请求的URL指向的资源已经被删除;但在响应报文中通过首部Location指明了资源现在所处的新位置;Moved Permanently
302: 响应报文Location指明资源临时新位置 Moved Temporarily
304: 客户端发出了条件式请求,但服务器上的资源未曾发生改变,则通过响应此响应状态码通知客户端;Not Modified
307:  浏览器内部重定向
401: 需要输入账号和密码认证方能访问资源;Unauthorized
403: 请求被禁止;Forbidden
404: 服务器无法找到客户端请求的资源;Not Found(客户端的错误,如dns找不到网页,网址输入错误等)
500: 服务器内部错误;(大概率服务器宕机了)
502: 代理服务器从后端服务器收到了一条伪响应,如无法连接到网关;Bad Gateway
503: 服务不可用,临时服务器维护或过载,服务器无法处理请求,比如:php服务停止,无法处理php程序
504: 网关超时

4、MIME

MIME : Multipurpose Internet Mail Extensions 多用途互联网邮件扩展

文件 /etc/mime.types ,来自于mailcap包

MIME格式:type/subtype

如果你访问的文件类型在文件/etc/mime.type内,可以直接打开访问,若不在则是下载。 

5、URL组成

URL:Uniform Resorce Locator,统一资源定位符,用于描述某服务器某特定资源位置

URN:Uniform Resource Naming,统一资源命名

二者区别:URN如同一个人的名称,而URL代表一个人的住址。换言之,URN定义某事物的身份,而URL提供查找该事物的方法。URN仅用于命名,而不指定地址。

URL组成:

<scheme>://<user>:<password>@<host>:<port>/<path>;<params>?<query>#<frag>

scheme:方案,访问服务器以获取资源时要使用哪种协议
user:用户,某些方案访问资源时需要的用户名
password:密码,用户对应的密码,中间用:分隔
Host:主机,资源宿主服务器的主机名或IP地址
port:端口,资源宿主服务器正在监听的端口号,很多方案有默认端口号
path:路径,服务器资源的本地名,由一个/将其与前面的URL组件分隔
params:参数,指定输入的参数,参数为名/值对,多个参数,用;分隔
query:查询,传递参数给程序,如数据库,用?分隔,多个查询用&分隔
frag:片段,一小片或一部分资源的名字,此组件在客户端使用,用#分隔

6、HTTP协议版本

主要介绍http/0.9版本、http/1.0版本以及http/1.1版本的区别

版本区别
http/0.9只有个GET命令,即下载命令
http/1.0不仅有GET命令,还有POST命令,即上传命令
http/1.1除了1.0版本有的,还支持长连接

长连接(持久连接):一次三次握手,多次四次挥手。简单理解就是,建立一个进程,可以有多次请求,用完继续使用,绿色又环保;但是可以设置默认超时时间,一般默认超时时间是15秒。

7、系统处理http请求的工作模式

工作模式有:prefork、worker、event

prefork:多进程I/O模型,每个进程响应一个请求。简单理解就是,来一个请求,开一个进程,来一个,开一个,很浪费系统资源。

worker:复用的多进程I/O模型,多进程多线程。有一个主进程,生成多个子进程,子进程生成多个线程,每个线程响应一个请求,并发送响应请求;每个请求完成后,不主动向父进程反馈自己已完成,得等到超时之后,才返回。

event:事件驱动模型(worker的变种),centeos8的默认模型。在worker模式的基础上,添加了一个监工线程,专门监视线程是否已完成响应,完成后,为其分配其他请求。

工作模式优点缺点
prefork稳定速度慢,占用资源,不适用于高并发场景
worker相对于prefork占用内存较少,可以同时处理更多请求使用keep-alive的长连接方式,某个线程会一直被占据,即使没有传输数据,也需要一直等待到超时才会被释放。如果过多的线程,被这样占据,也会导致在高并发场景下的无服务线程可用。
event单线程响应多请求,占据更少的内存,高并发下表现更优秀,会有一个专门的线程来管理keep-alive类型的线程,当有真实请求过来的时候,将请求传递给服务线程,执行完毕后,又允许它释放没有线程安全控制

8、apache与nginx的区别

nginx:

  • 轻量级,采用c进行编写,同样的web服务,会占用更少的内存及资源
  • 抗并发,处理请求是异步非阻塞的,负载能力比apache高很多,在高并发下能保持低资源低消耗高性能
  • 处理静态文件的性能好,静态处理性能比apache高三倍以上
  • 设计高度模块化,编写模块相对简单
  • 配置简洁,正则配置让很多事情变得简单,而且改完配置能使用-t测试配置是否有问题
  • 支持7层负载均衡,7层负载均衡可以有效防止ddos攻击
  • nginx本身就是一个反向代理服务器,也可以用来做邮件代理服务器使用,但基本上不用其做邮件代理服务器

apache:

  • apache的重写功能(rewrite)比nginx强大
  • 模块多
  • 更为成熟,bug少
  • 对php支持比较简单,nginx需要配合其他后端用
  • 处理动态请求有优势

两者最核心的区别:apache是同步多进程模型,一个连接对应一个进程;nginx是异步非阻塞模型,多个连接可以对应一个进程(万级别)

 二、I/O模型

Linux的 I/O

磁盘I/O:磁盘I/O是进程向内核发起系统调用,请求磁盘上的某个资源比如是html 文件或者图片,然后内核通过相应的驱动程序将目标文件加载到内核的内存空间,加载完成之后把数据从内核内存再复制给进程内存,如果是比较大的数据也需要等待时间。

网络I/O:一切皆文件,本质为对socket文件的读写。网络通信就是网络协议栈到用户空间进程的IO就是网络IO

网络请求的过程:

1、客户端发起请求,先发送到服务器的网卡,服务器网卡接收客户端的请求

2、服务器网卡将收到的报文复制到内核空间中,进行解析,然后调用用户空间

3、再将报文复制到用户空间,再次对报文进行解析,解析出客户端请求的内容

4、解析完之后,将需求反馈给内核空间,再通过内核去磁盘找所需文件,因为应用程序没有权限从磁盘直接读取文件

5、再磁盘找到文件之后,将文件加载到内核中,通过内核复制给用户空间

6、用户空间再将文件构建成响应报文,交给内核,通过内核复制给网卡

7、最后通过网卡发送响应报文给客户端

整体过程,来回在内核空间——用户空间——内核空间切换复制,很消耗系统资源,可以对其进行优化处理。

I/O模型相关概念

同步,异步,阻塞和非阻塞

同步:被调用者并不提供事件的处理结果相关的通知消息,需要调用者主动询问事情是否处理完成。就例如,公司领导让你做一件事,你做完之后,不给领导反馈你做完了,直到领导过来问你做没做完,你这才回答你做完了。

异步:被调用者通过状态、通知或回调机制主动通知调用者被调用者的运行状态。与同步正好相反,你做完了就会反馈做完了,不需要领导来问你。

阻塞:指IO操作需要彻底完成后才返回到用户空间,调用结果返回之前,调用者被挂起,干不了别的事情。

非阻塞:指IO操作被调用后立即返回给用户一个状态值,而无需等到IO操作彻底完成,在最终的调用结果返回之前,调用者不会被挂起,可以去做别的事情。

事件驱动模型

事件驱动模型
selectpollepoll
操作方式遍历遍历回调
底层实现数组链表哈希表
IO效率每次调用都进行线性遍历,时间复杂度O(n)每次调用都进行线性遍历,时间复杂度O(n)事件通知方式,每当fd就绪,系统注册的回调函数就会被调用,时间按复杂度O(1)
最大连接数

1024(x86)

2048(x64)

无上限无上限
fd拷贝每次调用select都需要把fd集合从用户拷贝到内核态每次调用poll都需要把fd集合从用户态拷贝到内核态调用epoll_ctl 时拷贝进内核并保存,之后每次epoll_wait不拷贝

三、Nginx服务

Nginx 功能介绍

  • 静态的web资源服务器html,图片,js,css,txt等静态资源

  • http/https协议的反向代理 7层

  • 结合FastCGI/uWSGI/SCGI等协议反向代理动态资源请求

  • tcp/udp协议的请求转发(反向代理) 4层

基础特性

  • 模块化设计,较好的扩展性

  • 高可靠性

  • 支持热部署:不停机更新配置文件,升级版本,更换日志文件

  • 低内存消耗:10000个keep-alive连接模式下的非活动连接,仅需2.5M内存

  • event-driven,aio,mmap,sendfile

Web 服务相关的功能

  • 虚拟主机(server)

  • 支持 keep-alive 和管道连接(利用一个连接做多次请求)

  • 访问日志(支持基于日志缓冲提高其性能)

  • url rewirte

  • 路径别名

  • 基于IP及用户的访问控制

  • 支持速率限制及并发数限制

  • 重新配置和在线升级而无须中断客户的工作进程

1、nginx模块

  • 核心模块:是 Nginx 服务器正常运行必不可少的模块,提供错误日志记录 、配置文件解析 、事件驱动机制 、进程管理等核心功能

  • 标准HTTP模块:提供 HTTP 协议解析相关的功能,比如: 端口配置 、 网页编码设置 、 HTTP响应头设置 等等

  • 可选HTTP模块:主要用于扩展标准的 HTTP 功能,让 Nginx 能处理一些特殊的服务,比如:Flash 多媒体传输 、解析 GeoIP 请求、 网络传输压缩 、 安全协议 SSL 支持等

  • 邮件服务模块:主要用于支持 Nginx 的 邮件服务 ,包括对 POP3 协议、 IMAP 协议和 SMTP协议的支持

  • Stream服务模块: 实现反向代理功能,包括TCP协议代理

  • 第三方模块:是为了扩展 Nginx 服务器应用,完成开发者自定义功能,比如: Json 支持、 Lua 支持等

核心模块:core module
标准模块:
 HTTP 模块: ngx_http_*
 HTTP Core modules   #默认功能
 HTTP Optional modules #需编译时指定
 Mail 模块: ngx_mail_*
 Stream 模块 ngx_stream_*
第三方模块

2、安装nginx

yum安装nginx

1)先安装epel-release源

yum install -y epel-release

2)安装nginx

yum install nginx -y

编译安装nginx

1)安装环境依赖包,并创建nginx用户管理nginx

yum -y install gcc pcre-devel openssl-devel zlib-devel openssl  openssl-devel

#安装环境依赖包

useradd -M -s /sbin/nologin nginx

#新建nginx用户

2)下载nginx源码包

cd /opt/                #切换到/opt目录下

wget http://nginx.org/download/nginx-1.18.0.tar.gz                #官网下载安装包

 3)解压安装包,创建一个安装nginx的安装目录

tar xf nginx-1.18.0.tar.gz         #解压安装包
cd nginx-1.18.0/                #进入解压的安装包中

mkdir /apps/nginx -p        #创建安装目录

4)检测安装环境,编译安装

./configure 
--prefix=/apps/nginx \        #指定安装目录
--user=nginx \        #指定用户
--group=nginx \        #指定用户组
#以下全是安装的功能模块
--with-http_ssl_module \    
--with-http_v2_module \
--with-http_realip_module \
--with-http_stub_status_module \
--with-http_gzip_static_module \
--with-pcre \
--with-stream \
--with-stream_ssl_module \
--with-stream_realip_module


5)修改权限

chown -R nginx.nginx /apps/nginx        #将nginx的安装目录修改属主和属组

 安装好后的,安装目录中的配置文件功能

 启动nginx服务

方法一:使用绝对路径启动
/apps/nginx/sbin/nginx

方法二:创建软连接后,直接nginx启动
ln -s /apps/nginx/sbin/nginx  /usr/sbin/
nginx    #直接输入nginx启动

方法三:创建nginx的service文件,使用systemd管理
vim /usr/lib/systemd/system/nginx.service

#建立文件
[Unit]
Description=nginx - high performance web server
Documentation=http://nginx.org/en/docs/
After=network-online.target remote-fs.target nss-lookup.target
Wants=network-online.target
[Service]
Type=forking
PIDFile=/apps/nginx/logs/nginx.pid
#注意文件位置,如果不对 启动不了
ExecStart=/apps/nginx/sbin/nginx -c /apps/nginx/conf/nginx.conf
#注意启动文件位置
ExecReload=/bin/kill -s HUP $MAINPID
ExecStop=/bin/kill -s TERM $MAINPID
[Install]
WantedBy=multi-user.target

 使用第三种方法创建service文件为例

vim /usr/lib/systemd/system/nginx.service

 重新加载配置

systemctl   daemon-reload        #重新加载system配置文件

systemctl     start    nginx        #启动nginx服务

3、nginx信号

nginx 命令支持向其发送信号,实现不同功能

nginx 当做单独命令使用有以下选项

编译安装nginx,使用nginx命令前,需要将安装包中的nginx创建软连接至PATH环境下,或者将安装包中的/sbin/nginx拷贝到/usr/sbin/下

ln -s /apps/nginx/sbin/nginx  /usr/sbin/

 nginx  -v        显示nginx版本

 nginx   -V        显示编译安装的详细情况

 可以使用man手册来查看详细的信号 如果没安装,去源码包里找到man文件

 nginx  -s   信号(stop、quit、reload、reopen) 

nginx -s   stop   #立即关闭nginx
nginx -s   quit   #优雅退出   不影响业务的状态下退出
nginx -s   reload #重新加载
nginx -s   reopen #重新加载配置文件

nginx  -t      检查语法格式 

nginx -g 指定配置 不已配置文件中的为准

nginx -g 'user zhangsan;'   已张三身份运行,默认是以nginx身份
nginx -g 'daemon off;'    前台运行命令

4、nginx服务调优

1)根据cpu数量配置master有多少个worker子进程

全局配置文件/apps/nginx/conf/nginx.conf中修改:work_processes auto(表示根据cpu数量决定个数)。

2) 关闭或修改版本

关闭版本

修改版本,需要修改安装目录下的/src/core/nginx.h文件

vim /opt/nginx-1.18.0/src/core/nginx.h        #在源码包中

vim /opt/nginx-1.18.0/src/http/ngx_http_header_filter_module.c        #在源码包中


 然后重新编译安装,才算修改完成
注意:重新编译安装,如果nginx服务没有关,重新编译安装了,需要重启nginx服务,才能生效

3)cpu与work进程绑定

 将Nginx工作进程绑定到指定的CPU核心,默认Nginx是不进行进程绑定的,绑定并不是意味着当前nginx进程独占以一核心CPU,但是可以保证此进程不会运行在其他核心上,这就极大减少了nginx的工作进程在不同的cpu核心上的来回跳转,减少了CPU对进程的资源分配与回收以及内存管理等,因此可以有效的提升nginx服务器的性能。

修改安装目录文件下的nginx的配置文件

vim /apps/nginx/conf/nginx.conf


4)nginx进程的优先级(work进程的优先级) 

 当你想将nginx的work进程的优先级调高 可以使用nice设置,优先级范围是 -20到19,数值越小,优先级越高


5)设置nginx可打开的文件个数 

所有worker进程能打开的文件数量上限,包括:Nginx的所有连接(例如与代理服务器的连接等),而不仅仅是与客户端的连接,另一个考虑因素是实际的并发连接数不能超过系统级别的最大打开文件数的限制.最好与ulimit -n 或者limits.conf的值保持一致,所以需要联合ulimit.conf文件一起设置。


 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/920223.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

春秋云镜 CVE-2019-16113

春秋云镜 CVE-2019-16113 Bludit目录穿越漏洞 靶标介绍 在Bludit<3.9.2的版本中&#xff0c;攻击者可以通过定制uuid值将文件上传到指定的路径&#xff0c;然后通过bl-kernel/ajax/upload-images.php远程执行任意代码。 启动场景 漏洞利用 exp https://github.com/Kenun…

Matlab绘制灰度图像

灰度图介绍 灰度图像是指每个像素的信息由一个量化的灰度级来描述的图像。如果每个像素的灰度值用一个字节表示&#xff0c;灰度值级数就等于256级&#xff0c;每个像素可以是0-255之间的任何一个数。其特点是&#xff1a;它只有亮度信息&#xff0c;没有颜色信息。占据存储空…

Java IO流(四)Netty理论[模型|核心组件]

概述 Netty是由JBOSS提供的一个Java开源框架,可从Github获取独立项目Netty是一个异步的、基于事件驱动的网络应用框架,用于快速开发可维护、高性能的网络服务器和客户端(摘录官网)Netty所谓的异步是针对用户使用Channel进行IO操作,会立即返回ChannelFuture。但IO操作的任务是提…

AutoSAR配置与实践(基础篇)3.5 BSW 的模式管理

传送门 -> AUTOSAR配置与实践总目录 AutoSAR配置与实践(基础篇)3.5 BSW 的模式管理 一、模式管理的组成二、模式项内容简介一、模式管理的组成 AUTOSAR为ECU的运行时软件的状态处理提供了模式管理组件,如下 • BswM模式管理器 • NM网络管理 • EcuM状态管理器 • ComM通…

Kaggle分类问题Titanic——Machine Learning from Disaster

目录 前言1 题目介绍2 数据清洗3 数据可视化分析4 模型训练5 源码 前言 这是我在大三选修课的课程设计&#xff0c;内容参考了Kaggle上高赞的代码&#xff0c;有详细批注&#xff0c;整体比较基础&#xff0c;结构相对完整&#xff0c;便于初学者学习。这个是一个分类问题&am…

vue 简单实验 v-if 条件判定

1.代码 <script src"https://unpkg.com/vuenext" rel"external nofollow" ></script> <div id"conditional-rendering"><span v-if"seen">现在你看到我了</span> </div> <script> const C…

纷享销客入选“寻找创新的「踏脚石」|36氪数字创新标杆案例”

近日&#xff0c;36氪重磅发布“数字创新标杆案例&先进团队名册”&#xff0c;本期围绕寻找创新的「踏脚石」的主题&#xff0c;共遴选出36个数字创新标杆案例与10个先进团队&#xff0c;纷享销客连接型CRM凭借过硬的产品及自主科研创新实力和服务实践有幸入选。 此次评选历…

分布式定时任务框架Quartz总结和实践(2)—持久化到Mysql数据库

本文主要介绍分布式定时任务框架Quartz集成SpringBoot持久化数据到Mysql数据库的操作&#xff0c;上一篇文章使用Quartz创建定时任务都是保存在内存中&#xff0c;如果服务重启定时任务就会失效&#xff0c;所以Quartz官方也提供将定时任务等信息持久化到Mysql数据库的功能&…

[docker][WARNING]: Empty continuation line found in:

报警内容&#xff1a; 下面展示一些 内联代码片。 //执行 sudo docker build ubuntu:v1.00 . [WARNING]: Empty continuation line found in:出现上述错误原因为18行多了一个 " \" 符号&#xff0c;去除即可

Hexo添加相册

首发博客地址 https://blog.zysicyj.top/ 视频教学地址 https://www.bilibili.com/video/BV1Fw411Q7pS/ 打开butterfly配置文件&#xff0c;在menu处新增一行 menu: 主页: / || fas fa-home 时间轴: /archives/ || fas fa-archive 标签: /tags/ || fas fa-tags 分类: /cate…

函数模版c++

泛型编程 如何实现一个通用的交换函数呢&#xff0c;利用c函数重载的方式 void Swap(int& left, int& right) { int temp left; left right; right temp; } void Swap(double& left, double& right) { double temp left; left right; right temp; } vo…

AMBA总线协议(9)——AHB(七):终章

一、前言 在之前的文章中我们讲述了AHB协议的分割传输机制&#xff0c;它使得从机可以决定一次传输是否继续进行&#xff0c;以防止 传输的执行将占据大量的时钟周期&#xff0c;有效提高了总线的公平性与效率问题&#xff0c;本文中我们将一次性学习完AHB最后的内容&#xff0…

蚂蚁 SOFAServerless 微服务新架构的探索与实践

赵真灵&#xff08;有济&#xff09; 蚂蚁集团技术专家 Serverless 和微服务领域专家曾负责基于 K8s Deployment 的应用发布运维平台建设、K8s 集群的 Node/pod 多级弹性伸缩与产品建设。当前主要负责应用架构演进和 Serverless 相关工作。同时也是 SOFAArk 社区的开发和维护者…

【CP2K学习】-在Ubuntu上安装CP2K的全过程(包括gcc,gfortran,MKL等配置)

在Ubuntu中安装CP2K CP2K的安装检查系统是否安装gcc,gfortranMKL数学库的安装CP2K安装包下载CP2K的编译CP2K的测试ssmp版本测试popt版本测试 CP2K是第一性原理计算程序中发展迅速的程序之一&#xff0c;因其开源性、速度性等优点&#xff0c;是广大计算化学研究者的选择。 本文…

2023.8各大浏览器11家对比:Edge/Chrome/Opera/Firefox/Tor/Vivaldi/Brave,安全性,速度,体积,内存占用

测试环境&#xff1a;全默认设置的情况下&#xff0c;均在全新的系统上进行测试&#xff0c;系统并未进行任何改动&#xff0c;没有杀毒软件&#xff0c;浏览器进程全部在后台&#xff0c;且为小窗模式&#xff0c;小窗分辨率均为浏览器厂商默认缩放大小(变量不唯一)&#xff0…

C#|如何调试进依赖动态库中

第一步&#xff1a;打开项目属性 第二步 打开debug的本地调试可用 第三步 把要调试的代码拖进主界面打断点就可以进断点了

测试分类

测试分类&#xff08;全是概念&#xff1b;非常抽象&#xff09;按对象划分界面测试可靠性测试容错性测试文档测试兼容性测试易用性安装卸载测试安全测试性能测试内存泄漏测试 按是否查看代码划分黑盒测试白盒测试灰盒测试 按开发阶段划分单元测试集成测试系统测试回归测试冒烟…

js判断类型:typeof Object.prototype.toString instanceof constructor有什么区别?一文讲清楚

相信很多小伙伴在使用js的过程中&#xff0c;经常会需要对js的数据类型进行判断&#xff0c;而js中可以对数据类型进行判断的方法有很多种&#xff0c;最常见的有typeof、Object.prototype.toString、instanceof、constructor这四种&#xff0c;那么他们有什么区别呢&#xff1…

ssm+vue游戏攻略网站源码和论文

ssmvue游戏攻略网站源码和论文052 开发工具&#xff1a;idea 数据库mysql5.7 数据库链接工具&#xff1a;navcat,小海豚等 技术&#xff1a;ssm 一、主要内容和基本要求 游戏攻略网站分为管理员与用户两种角色。 管理员的功能包括登录&#xff0c;用户管理&#xff0c;游…

Laravel 框架构造器的查询表达式构造器的 Where 派生查询 ⑥

作者 : SYFStrive 博客首页 : HomePage &#x1f4dc;&#xff1a; THINK PHP &#x1f4cc;&#xff1a;个人社区&#xff08;欢迎大佬们加入&#xff09; &#x1f449;&#xff1a;社区链接&#x1f517; &#x1f4cc;&#xff1a;觉得文章不错可以点点关注 &#x1f44…