php mysql长连接聊天室_PHP之探索MySQL 长连接、连接池

PHP连接MysqL的方式,用的多的是MysqL扩展、MysqLi扩展、pdo_MysqL扩展,是官方提供的。PHP的运行机制是页面执行完会释放所有该PHP进程中的所有资源的,如果有多个并发访问本地的测试页面 http://127.0.0.1/1.php 根据PHP跟web服务器的不同,会开相应的线程或者进程去处理该请求,请求完了会释放结果的。也就是PHP无法从语言层面从页面到页面之间传递一些数据,但是MysqL_pconnect跟pdo中的ATTR,设置array(PDO::ATTR_PERSISTENT => true)如下是可以实现长连接的。

$conn = new PDO($dsn,DB_USER,DB_PASSWORD,array(PDO::ATTR_PERSISTENT => true)

);

长连接的作用我觉得是在高负载的情况下,通过复用长连接,减少了每个页面的建立数据库连接的时间,而这个建立MysqL connection的时间,在我的机器上

在数据库connnections < 10的情况下,MysqL pdo 建立connection time 为0.003ms,MysqLi建立connection time为0.14ms

在数据库connection接近满的时候,MysqL pdo建立connection time为0.13ms,MysqLi建立connection time为0.13ms

以上样本都是在大概估测时间,时间太小不好估计。其实建立连接的时间并不长,那这样为什么需要MysqL长连接、连接池这样的东西呢。那是在高负载下,比如server单机可以接受的MysqL并发在200左右,web server的单机并发在700左右,那么当大批量500并发连接压过来的时候,web server没到满负荷,MysqL提前到了满负荷,就会导致所有页面无法响应、或者已经建立好数据库连接的页面执行很慢。

PHP中的MysqL长连接由于PHP的运行方式有多种,因而长连接实现也有多种。需要web服务器支持才可以实现长连接,因为PHP是没有进程池跟连接池这种概念的,绝大多数情况下PHP应用本身不是一个应用服务器(后起之秀swoole,是一个优秀的PHP应用服务器,不过是在c层面做的)。因而PHP的长连接其实是搭载apache这样的带有mpm模块的webserver,linux 下apache会维护一个进程池,开启了apache mpm功能之后,apache会默认维持一个进程池,MysqL长连接之后的连接,并没有作为socet连接关闭,而是作为一个不释放的东西,放进了进程池/线程池里面去。等需要连接的时,apache从它维护的进程池/线程池里面取出MysqL socket connnection,然后就可以复用此连接了。

这里测试一下,首先本机环境是archlinux,后文所用MysqL httpd PHP都是自己编译的源代码,都在/home/dengpan/opt目录。httpd的mpm模型这里采用的是worker,httpd的mpm(apache用于并行方面功能的,俗称多路处理模块)其实有perfork、worker、event三种。mpm的好处是让apache随时有些备用的spare或者空闲的子进程(服务器线程池),随时等待新过来的请求,这样客户端不需要在请求服务之前等待子进程的产生。

使用什么mpm,需要单独指定编译进去apache里面去,比如编译work mpm到apache里面去,比如我的最简化httpd的编译参数是

./configure \

--with-apr=/home/dengpan/opt/apr-1.5.2 \

--with-apr-util=/home/dengpan/opt/apr-util-1.5.4 \

--prefix=/home/dengpan/opt/httpd-2.4.16 \

--with-mpm=worker

查看httpd加载的模块,

e3c6a18f691feac7c124b4f39ed5563f.png

看到worker.c 已经编译过去了,

mpm的配置参数为

StartServers 15

MinSpareThreads 75

MaxSpareThreads250

ThreadsPerChild 10

MaxRequestWorkers 400

MaxConnectionsPerChild 0

启动apache用pstree看到 |-httpd—15[httpd—11[{httpd}]],说明起了15个server进程,每个server起了10个子线程。整个mpm要维持的最小的闲置线程数量在75,最大的闲置线程在250。满载的最大的工作线程在400个。下面准备一个shell脚本,每1秒输出下当前MysqL的active连接数量,查看MysqL current连接数我用的较多的有2个方法

进MysqL shell,执行SHOW STATUS WHERE variable_name = ‘Threads_connected'; 不过这个方法得MysqL shell进的去才对,当connections很多的时候,MysqL shell进不去也就无法查询了

shell直接查询,find /proc/pidof MysqLd/fd/ -follow -type s | wc -l,需要root权限,好处是即使MysqL因为too many connections无法进入shell的时候还是可以连接进去。

这里用方法2,因为后面回到机器的MysqL满载负荷的。写一个shell如下:

#!/bin/bash

while(true)

do

find /proc/`pidof MysqLd`/fd/ -follow -type s | wc -l

sleep 1

done

后面执行该shell不断的输出当前连接数,测试可得

cli下执行PHP,长连接无效,cli下脚本一退出,连接即释放

apche+mod_PHP不开启mpm模块的话,无论MysqL MysqL_pconnect、pdo_MysqL长连接,页面访问完毕,MysqL连接即释放。

apche+mod_PHP开启mpm模块(worker模式)的话,无论MysqL MysqL_pconnect、pdo_MysqL长连接,MysqL连接+1,直到达到最大的MysqL连接数,不在增加,但是访问页面还是可以复用连接查询到相应数据。

apache之所以能够复用MysqL连接,说明apache肯定为MysqL自己实现了一些功能函数、模块,否则不可能把一个未知类型的socket指针存下来的。用ldd查看,

➜ MysqL_persist ldd /home/dengpan/opt/httpd-2.4.16/bin/httpd

linux-vdso.so.1 (0x00007ffffcbde000)

libpcre.so.1 => /usr/lib/libpcre.so.1 (0x00007f8e8d17c000)

libaprutil-1.so.0 => /home/dengpan/opt/apr-util-1.5.4/lib/libaprutil-1.so.0 (0x00007f8e8cf57000)

libexpat.so.1 => /usr/lib/libexpat.so.1 (0x00007f8e8cd2d000)

libapr-1.so.0 => /home/dengpan/opt/apr-1.5.2/lib/libapr-1.so.0 (0x00007f8e8cafb000)

libuuid.so.1 => /usr/lib/libuuid.so.1 (0x00007f8e8c8f6000)

librt.so.1 => /usr/lib/librt.so.1 (0x00007f8e8c6ee000)

libcrypt.so.1 => /usr/lib/libcrypt.so.1 (0x00007f8e8c4b6000)

libpthread.so.0 => /usr/lib/libpthread.so.0 (0x00007f8e8c299000)

libdl.so.2 => /usr/lib/libdl.so.2 (0x00007f8e8c095000)

libc.so.6 => /usr/lib/libc.so.6 (0x00007f8e8bcf3000)

/lib64/ld-linux-x86-64.so.2 (0x00007f8e8d3ec000)

可以猜测出/home/dengpan/opt/apr-util-1.5.4/lib/libaprutil-1.so.0跟/home/dengpan/opt/apr-1.5.2/lib/libapr-1.so.0应实现了跟MysqL相关的代码段。由于我是本地编译的,很方便找到函数入口,/home/dengpan/github/apache-httpd/apr-util-1.5.4/dbd/apr_dbd_MysqL.c这个文件,其实是apache的mod_dbd对常见的数据库都做了长连接支持。而Nginx跟PHP-fpm的关系并不像PHP跟apache那样,所以Nginx+PHP-fpm无法实现对应的长连接。大概是PHP-fpm并没有做MysqL的进程、线程池。

最后给出我的建议,一般小型PHP应用是没有性能问题的,PHP自身连接MysqL很快,很多都处于性能过剩,随着apache慢慢被Nginx替代,PHP的MysqL长连接也只会越来越鸡肋。单机的话,其实要是怕MysqL创建connections有压力,最好把MysqL的创建使用单例模式,这样一个页面只会创建一个MysqL连接实例。诸如下面的单例例子代码,而这个更适合写在框架里去实现单例。

/**

* Created by PHPStorm.

* User: dengpan

* Date: 15-7-24

* Time: 下午2:56

*/

include "./db.PHP";

class DB {

private static $_instance;

private $db;

private function __construct()

{

$this->db = new MysqLi(DB_HOST,'my',3307);

}

public static function getInstance()

{

if (!(self::$_instance instanceof DB)) {

self::$_instance = new self();

}

return self::$_instance;

}

private function __clone()

{

}

public function getConn()

{

return $this->db;

}

}

$conn = DB::getInstance()->getConn();

点关注,不迷路

好了各位,以上就是这篇文章的全部内容了,能看到这里的人呀,都是人才。之前说过,PHP方面的技术点很多,也是因为太多了,实在是写不过来,写过来了大家也不会看的太多,所以我这里把它整理成了PDF和文档,如果有需要的可以

f9b64ae5502d25c92381f6c90ffc0648.png

07a63a31ebb230dda4fd1458ef8fb05f.png

以上内容希望帮助到大家,很多PHPer在进阶的时候总会遇到一些问题和瓶颈,业务代码写多了没有方向感,不知道该从那里入手去提升,对此我整理了一些资料,包括但不限于:分布式架构、高可扩展、高性能、高并发、服务器性能调优、TP6,laravel,YII2,Redis,Swoole、Swoft、Kafka、MysqL优化、shell脚本、Docker、微服务、Nginx等多个知识点高级进阶干货需要的可以免费分享给大家,需要的可以加入我的

总结

如果觉得编程之家网站内容还不错,欢迎将编程之家网站推荐给程序员好友。

本图文内容来源于网友网络收集整理提供,作为学习参考使用,版权属于原作者。

如您喜欢交流学习经验,点击链接加入交流1群:1065694478(已满)交流2群:163560250

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

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

相关文章

python 读取地震道头数据_python地震数据可视化详解

本文实例为大家分享了python地震数据可视化的具体代码&#xff0c;供大家参考&#xff0c;具体内容如下准备工作&#xff1a;在windows10下安装python3.7&#xff0c;下载参考源码到本地。1. demo绘图测试demo绘图指令cmd> python seisplot.py --demo问题1)缺少依赖包File &…

在MySQL查询山东省男生信息_MySQL-查询

来一波英语单词解释(意思)create 创建show 显示database 数据库use 使用select 选择table 表from 来自…distinct 消除重复行as 同样地(用于其别名)where 范围like 模糊查询rlike 正则查询In 范围查询not in 不非连续的范围之内between ... and …表示…

java 导入world数据_java读取world文件,把world文件中的内容,原样输出到页面上。...

POI,处理可以。样式在Java代码中添加就可以。给了一个例子这个是Excel的。package cn.com.my.common;import java.io.File;import java.io.FileInputStream;import java.io.FileOutputStream;import java.io.IOException;import java.sql.Connection;import java.sql.ResultSet…

java程序员 css_Java程序员从笨鸟到菜鸟之(十七)CSS基础积累总结(下)

七.组织元素(span和div)span和div元素用于组织和结构化文档&#xff0c;并经常联合class和id属性一起使用。在这一课中&#xff0c;我们将进一步探究span和div的用法&#xff0c;因为这两个HTML元素对于CSS是很重要的。用span组织元素用div组织元素用span组织元素span元素可以说…

redlock java_Redlock分布式锁

这篇文章主要是对 Redis 官方网站刊登的 Distributed locks with Redis 部分内容的总结和翻译。什么是 RedLockRedis 官方站这篇文章提出了一种权威的基于 Redis 实现分布式锁的方式名叫 Redlock&#xff0c;此种方式比原先的单节点的方法更安全。它可以保证以下特性&#xff1…

java 两个数组交叉_java – 如何交叉两个没有重复的排序整数数组?

这个问题本质上减少到一个连接操作,然后是一个过滤器操作(删除重复,只保留内部匹配).由于输入都已经排序,所以可以通过O(O(size(a)size(b))的merge join来有效地实现连接.过滤器操作将为O(n),因为连接的输出被排序,并且要删除重复项,所有您需要做的是检查每个元素是否与之​​前…

java retentionpolicy_Java注解之如何利用RetentionPolicy.SOURCE生存周期

上一篇文章简单讲了下Java注解的学习之元注解说明&#xff0c;学习了Java注解是如何定义的&#xff0c;怎么使用的&#xff0c;但是并没有介绍Java的注解是怎么起作用的&#xff0c;像Spring Boot里面的那些注解&#xff0c;到底是怎么让程序这样子运行起来的&#xff1f;特别是…

在java程序中定义的类有两种成员_java试题 急需答案 谢谢!!!

三、填空(每小题2分&#xff0c;共10分)1&#xff0e;在Applet中&#xff0c;创建一个具有10行45列的多行文本区对象ta的语句为&#xff1a;2&#xff0e;创建一个标识有“关闭”字样的标签对象gb的语句为。3&#xff0e;方法是一种仅有方法头&#xff0c;没...三、填空(每小题…

java 同步 变量,在java中的对象上同步,然后更改同步的变量的值

I came across a code like thissynchronized(obj) {obj new Object();}Something does not feel right about this , I am unable to explain, Is this piece of code OK or there is something really wrong in it, please point it out.Thanks解决方案Its probably not wha…

java set泛型_Java 集合二 泛型、Set相关

泛型1、在定义一个类的方法时&#xff0c;因为不确定返回值类型&#xff0c;所以用一个符号代替&#xff0c;这个符号就是泛型eg:ArrayList list new ArrayList();2、泛型的好处&#xff1a;1、提高了数据的安全性&#xff0c;将运行时的问题提前暴露在编译阶段2、避免了强转的…

java annotation 实现_在Java中如何实现自己的annotation

1. 先定义annotation2. 使用annotation例子&#xff1a;import java.lang.annotation.*;import java.lang.reflect.Method;Target(ElementType.METHOD)Retention(RetentionPolicy.RUNTIME)interface Test {String info() default "";}class Annotated {Test(info &q…

登录界面拦截java_java拦截通过url访问页面,必须通过登录页面访问目标页面

在web.xml中配置过滤&#xff1a;LoginFiltercom.verification.action.LoginFilterLoginFiltery/form/dealParse.do/* 拦截所有请求/.do 拦截以“.do”结尾的请求/index.jsp 拦截指定的jsp/artery/form/* 拦截该目录下的所有请求等等拦截器&#xff0c;拦截请求类&#xf…

python textwrap_[Python标准库]textwrap——格式化文本段落

textwrap——格式化文本段落作用&#xff1a;通过调整换行符在段落中出现的位置来格式化文本。 Python 版本&#xff1a;2.5 及以后版本 需要美观打印时&#xff0c;可以用 textwrap 模块来格式化要输出的文本。这个模块允许通过编程提供类似段落自动换行或填充…

java 字符串 1_java 字符串操作大全1

1、length() 字符串的长度例&#xff1a;char chars[]{a,b.c};String snew String(chars);int lens.length();2、charAt() 截取一个字符例&#xff1a;char ch;ch"abc".charAt(1); 返回b3、getChars() 截取多个字符void getChars(int sourceStart,int sourceEnd,char…

java实现权限_Java实现权限管理的两种方式

编辑特别推荐:种方式&#xff1a;利用filter、xml文件和用户信息表配合使用来实现权限管理。1.过滤器filterpackage cn.com.aaa.bbb.filter;import java.io.IOException;import java.io.InputStream;import java.util.HashMap;import java.util.Iterator;import java.util.List…

java 输入16进制_尝试使用十六进制输入来使用小端和大端

我试图用这两个原型编写C函数&#xff1a;int extract_little (char* str, int ofset, int n);int extract_big(char* str, int ofset, int n);现在一般的想法是我需要从地址str ofset开始以两种格式返回一个n字节整数 . 附&#xff1a; Ofset还没有做任何事情&#xff0c;我计…

java gson_Java 中 Gson的使用

JSON 是一种文本形式的数据交换格式&#xff0c;它比XML更轻量、比二进制容易阅读和编写&#xff0c;调式也更加方便;解析和生成的方式很多&#xff0c;Java中最常用的类库有&#xff1a;JSON-Java、Gson、Jackson、FastJson等一、Gson的基本用法Gson提供了fromJson() 和toJson…

spring注入普通java类_普通java类如何取得注入spring Ioc容器的对象

[除了使用XML配置外&#xff0c;还可以选择使用基于注解(annotation)的配置方式&#xff0c;其依赖于字节码来织入组件。注解注入在XML注入之前完成&#xff0c;因此在XML配置中可以重载注解注入的属性。一、建一个SpringUtil类package com.ceopen.eoss.spring; import org.spr…

java web 集成dom4j_[JavaWeb基础] 031.dom4j写入xml的方法

上一篇我们讲述了dom4j读取xml的4种方法&#xff0c;甚是精彩&#xff0c;那么怎么样写入xml呢&#xff1f;我们直接看下源码实现。public static void main(String[] args) throws Exception {// 创建文档Document document DocumentHelper.createDocument();// 设置编码docu…

java servlet 调试日志 logger sae_java servlet 调试日志 lo

java servlet 调试日志 lo[2021-02-10 08:32:08] 简介:php去除nbsp的方法&#xff1a;首先创建一个PHP代码示例文件&#xff1b;然后通过“preg_replace("/(\s|\&nbsp\;| |\xc2\xa0)/", " ", strip_tags($val));”方法去除所有nbsp即可。推荐&#x…