Maison >développement back-end >Problème PHP >Les avantages et les inconvénients de l'encapsulation php des instances pdo et des connexions longues pdo

Les avantages et les inconvénients de l'encapsulation php des instances pdo et des connexions longues pdo

WBOY
WBOYavant
2021-12-20 17:18:482894parcourir

Cet article vous apporte des exemples d'encapsulation PHP de pdo en PHP, ainsi que des connaissances pertinentes sur les connexions longues pdo, comme leur nom l'indique, maintiennent toujours la connexion. Par rapport aux connexions courtes habituelles, le lien sera re-. créées à chaque requête. Les connexions longues peuvent réduire efficacement le processus de création et mieux économiser les performances. J'espère que cela aidera tout le monde !

Les avantages et les inconvénients de l'encapsulation php des instances pdo et des connexions longues pdo

1. Introduction

Récemment, j'ai besoin d'écrire un script pour réaliser le stockage des journaux de crash. Comme prévu, il est séparé du framework, donc d'accord, nous ne pouvons encapsuler que nous-mêmes les opérations liées à la base de données. Le blogueur a choisi ici d'encapsuler pdo pour faire fonctionner la base de données. pdo操作数据库相关。

二、为什么选择pdo

      众所周知的,php在早期的时候是带有mysql扩展的,但是后来由于过于古老缺失了mysql的新特性,因此主键没落。

      从php5开始,更建议大家使用mysqli扩展,这个是mysql扩展的增强版,是一个面向对象的MySQL接口,更容易使用。缺点是只能操作mysql,不够强大。

      还有就是pdo扩展了,这个是最丰富的的一个扩展,支持多种数据库,重要的是,在安全上是比其他两种扩展都要强的,通过使用prepared预处理更是有效的防止sql注入。因此,博主这里选择了封装pdo相关的操作。

三、pdo的长连接

1、什么是pdo的长连接

      长连接顾名思义就是一直保持连接,相对于平时的短连接,每次请求都会重新创建链接来说,长连接可以有效的减少创建的过程,可以更好的节省性能。

在操作上是在连接数据库的时候,多加一个参数:

$pdo = new PDO($dsn, $username, $passwd, [PDO::ATTR_PERSISTENT => true]);

后面的PDO::ATTR_PERSISTENT => true 就是开启长连接的方法。

2、长连接对nginx无效吗

      博主在搜索长连接相关知识的时候,看到一篇文章,结论是长连接仅适用于apache,不适用于nginx,这是真的吗?

参考博文地址:https://www.cnblogs.com/wpjamer/articles/7106389.html

      大致结论是:长连接更多的是针对于apache的,因为apache维护一个进程池,开启了apache mpm功能之后,apache会默认维持一个进程池,mysql长连接之后的连接,并没有作为socet连接关闭,而是作为一个不释放的东西,放进了进程池/线程池里面去。

      而对于nginx来说,长连接是无效的,脚本执行结束则释放资源?

3、php-fpm下的长连接测试

这里前辈已经测试过了,咱们给出前辈的地址,大家有兴趣的可以看看

参考博文地址:https://hacpai.com/article/1526490593632

结论:
      事实证明php-fpm是可以实现长连接的,只是如果该进程空闲的话,会造成资源浪费。

      php-fpm的配置文件可以考虑设置pm.max_requests = 1000,代表每一个子进程的最大请求服务数量,如果超过了这个值,该子进程会被自动重启。

      比如max_requests这个参数,如果设置很大的话,那这个子进程要运行很多次才会重启,假如这个请求发生了错误或者内存泄漏,那么这个值设置很大是不合适的。但如果请求没有问题,这个值设置小的话就会频繁的重启,这样也会碰到不少502的问题,所以要仁者见仁,智者见智的设置了,这里初始化设置1000,如果测试没有内存泄漏等问题,可以再大一些。

4、长连接对事务的影响

参考博文地址:https://www.zhihu.com/question/62603122

总结: 如果业务并发比较大且带有事务,不建议使用长连接的方式。

5、总结

      博主在不断的搜索中,发现长连接要发挥出最佳性能始终是避不开连接池这点的,而php恰恰又不能很好的实现连接池,这点确实是有点小遗憾。

      整体来说在php中是暂时无法配置和mysql

2. Pourquoi choisir pdo🎜🎜 Comme nous le savons tous, php avait l'extension mysql au début, mais plus tard, elle était trop ancienne et il manquait mysql. Les nouvelles fonctionnalités de , donc la clé primaire a disparu. 🎜🎜 À partir de <code>php5, il est recommandé d'utiliser l'extension mysqli. Il s'agit d'une version améliorée de l'extension mysql et est un objet. Interface MySQL orientée, plus facile à utiliser. L'inconvénient est qu'il ne peut faire fonctionner que mysql et n'est pas assez puissant. 🎜🎜 Il existe également l'extension pdo. C'est l'extension la plus riche et prend en charge une variété de bases de données. L'important est qu'elle soit plus solide que les deux autres extensions en termes de sécurité. Le prétraitement >préparé est encore plus efficace pour empêcher l'injection sql. Par conséquent, le blogueur a choisi d'encapsuler les opérations liées au pdo. 🎜🎜3. Connexion longue de pdo🎜

1. Qu'est-ce qu'une connexion longue de pdo

🎜 Comme son nom l'indique, une connexion longue signifie maintenir la connexion tout le temps. Par rapport à la connexion courte habituelle, le lien. seront recréées à chaque demande. Les connexions longues peuvent réduire efficacement le processus de création et mieux économiser les performances. 🎜🎜En fonctionnement, lors de la connexion à la base de données, ajoutez un paramètre supplémentaire : 🎜
https://github.com/nadirvishun/php-pdo-class
🎜Le PDO::ATTR_PERSISTENT => true derrière se trouve la méthode pour activer les connexions longues. 🎜

2. Les connexions longues sont-elles invalides pour nginx ?

🎜 Lorsque le blogueur cherchait des informations sur les connexions longues, il a vu un article que les connexions longues ne s'appliquent qu'à Apache et non à . nginx. , est-ce vrai ? 🎜
    /**
     * @params:重连函数,上限3次
     * @date:2020/3/18
     * @time:17:03
     */
    public function customConnect()
    {
        try {
            $this->pdo = new PDO($this->config['dsn'], $this->config['username'], $this->config['password'], $this->config['params']);
            $this->pdo->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION); //需要将错误处理模式变成异常模式
            return true;
        } catch (Exception $e) {
            if (stripos($e->getMessage(), 'MySQL server has gone away') !== false || stripos($e->getMessage(),' bytes failed with errno=10053') !==false) {
                $this->close();
                $this->tryNums++;
                if ($this->tryNums > 3) {
                    return false;
                }
                self::customConnect();
            } else {
                $this->throw_exception($e->getMessage());
                return false;
            }
        }
    }
🎜 La conclusion générale est la suivante : les connexions longues sont plus ciblées sur apache, car apache maintient un pool de processus et active apache mpm après la fonction , apache maintiendra un pool de processus par défaut. La connexion après la longue connexion mysql n'est pas fermée en tant que connexion socet, mais en tant que Things. qui ne sont pas publiés sont placés dans le pool de processus/pool de threads. 🎜🎜       Et pour nginx, les connexions longues ne sont pas valides et les ressources sont libérées à la fin de l'exécution du script ? 🎜

3. Test de connexion long sous php-fpm

🎜Les seniors l'ont déjà testé ici Nous donnerons l'adresse des seniors Si vous êtes intéressé, vous pouvez jeter un oeil🎜
//自定义warnings处理函数set_error_handler('customException');//拿到warnngs错误之后,转化为error错误抛出,这样就可以被try..catch捕获function customException( $error_no, $error_msg, $error_file, $error_line){
    throw new \Exception($error_msg,0,null);
    //throw new \Exception($error_msg);}
🎜Conclusion : strong>
Les faits ont prouvé que php-fpm peut réaliser de longues connexions, mais si le processus est inactif, cela entraînera un gaspillage de ressources. Le fichier de configuration de 🎜🎜 php-fpm peut envisager de définir pm.max_requests = 1000, qui représente le nombre maximum de services de requête pour chaque processus enfant si cette valeur est dépassée. , l'enfant Le processus sera automatiquement redémarré. 🎜🎜 Par exemple, si le paramètre max_requests est défini sur une valeur très grande, le processus enfant devra s'exécuter plusieurs fois avant de redémarrer. Si une erreur se produit ou une fuite de mémoire se produit dans cette requête, alors. il n'est pas approprié de fixer cette valeur à une valeur très élevée. Mais s'il n'y a pas de problème avec la requête, si cette valeur est définie sur une petite valeur, elle redémarrera fréquemment, ce qui rencontrera également beaucoup de problèmes 502, c'est donc une question de jugement et sagesse. Voici le paramètre d'initialisation1000, si le test ne présente pas de fuites de mémoire et d'autres problèmes, il peut être plus important. 🎜

4. L'impact des connexions longues sur les transactions

    /**
     * destruct 关闭数据库连接
     */
    public function destruct()
    {
        $this->pdo = null;
    }
🎜Résumé : Si la concurrence commerciale est relativement importante et qu'il y a des transactions, il n'est pas recommandé d'utiliser des connexions longues. 🎜

5. Résumé

🎜 Au cours de la recherche continue, le blogueur a découvert que pour obtenir les meilleures performances des connexions longues, le pool de connexions ne peut pas être évité et PHP ne peut pas bien réaliser la connexion Chi, cela. c'est effectivement un peu dommage. 🎜🎜             De manière générale, il est temporairement impossible de configurer le pool de connexions parfait avec mysql en php. Dans les endroits où le métier est plus complexe, il est préférable d'utiliser des connexions longues. avec prudence. Chaque connexion est constituée d'un seul thread, ce qui entraînera un gaspillage important de ressources. 🎜

      如果是某些业务需要持续的数据库操作,比如提交日志接口等,那么是可以考虑打开长连接的,记得设置max_requests来定量关闭php-fpm连接,fpm关闭之后也会自动释放mysql的连接。

      还有pm.max_spare_servers设置服务器空闲时最大php-fpm进程数量。

例如: pm.max_spare_servers = 25 如果空闲时,会检查进程数,多于25个了,就会关闭几个,达到25个的状态。

      擅长swoole的同学,可以参考这篇文章:
基于swoole扩展实现真正的PHP数据库连接池

四、pdo部分demo的封装

      首先这部分博主是参考了一个网友的封装,github地址如下:

https://github.com/nadirvishun/php-pdo-class

      这个网友基本的增删改查都封装好了,而且都有参数预处理,安全性还是可以的。不过既然作为一个基准的类,还是缺少一些东西。

1、断线重连机制

例如重连函数:

    /**
     * @params:重连函数,上限3次
     * @date:2020/3/18
     * @time:17:03
     */
    public function customConnect()
    {
        try {
            $this->pdo = new PDO($this->config['dsn'], $this->config['username'], $this->config['password'], $this->config['params']);
            $this->pdo->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION); //需要将错误处理模式变成异常模式
            return true;
        } catch (Exception $e) {
            if (stripos($e->getMessage(), 'MySQL server has gone away') !== false || stripos($e->getMessage(),' bytes failed with errno=10053') !==false) {
                $this->close();
                $this->tryNums++;
                if ($this->tryNums > 3) {
                    return false;
                }
                self::customConnect();
            } else {
                $this->throw_exception($e->getMessage());
                return false;
            }
        }
    }

2、转化php warnings为try…catch可捕获的错误

      这步原因是长连接会频繁的造成mysql gone away错误,而这个错误是phpwarnings级别错误,try..catch根本就捕获不到,所以博主这里自定义错误处理函数来处理。

      这部分是转化phpwarnings错误为try..catch可以捕获的error错误,关于php的报错机制以及错误处理这块,咱们下篇再讨论。

//自定义warnings处理函数set_error_handler('customException');//拿到warnngs错误之后,转化为error错误抛出,这样就可以被try..catch捕获function customException( $error_no, $error_msg, $error_file, $error_line){
    throw new \Exception($error_msg,0,null);
    //throw new \Exception($error_msg);}

3、析构方法回收资源

    /**
     * destruct 关闭数据库连接
     */
    public function destruct()
    {
        $this->pdo = null;
    }

4、query的时候ping一下

  public function query($sql = null, $param = null)
    {
        //检测连接是否活跃
        $this->pdo_ping();
        //判断之前是否有结果集
        if (!empty($this->PDOStatement)) {
            $this->free();
        }
        xxxxxxxxxx        }

5、下载地址

      这四步完善之后,这个pdo的类还是可以用的,大家需要的话可以去百度云上下载。

链接: https://pan.baidu.com/s/1Siz_bKlhEIVNV99Y0zTzqw 提取码: ebqx

大家如果感兴趣的话,可以点击《PHP视频教程》进行更多关于PHP知识的学习。

Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!

Déclaration:
Cet article est reproduit dans:. en cas de violation, veuillez contacter admin@php.cn Supprimer