热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

php线程视频教程,通俗易懂的php多线程解决方案

我们在做项目的时候,有些需求,特别是数据的响应处理需要花费大量的时间,由于php是一个短生命周期的脚本语言,到了默认的30秒

我们在做项目的时候,有些需求,特别是数据的响应处理需要花费大量的时间,由于php是一个短生命周期的脚本语言,到了默认的30秒,php的数据处理还没完成,php的生命周期就结束了。这时需要使用异步并发处理策略,也就是说,一次php调用可以发出的多个请求,这些请求不是按照顺序执行,而是可以异步并发执行的,一些请求用于在后台处理数据,一些请求用于接受后台响应状态,根据状态,与用户做一些简单的交互。但是问题来了,我们都知道php本身是不支持多线程的,那么应该怎么实现php的多线程呢?

一、php模拟实现多线程的三种方法

1、linux下的php多线程

下面所讲的东西是源自php的pcntl_fork函数.因为这个函数依赖操作系统fork的实现,所以本文所讲的东西只适用于linux/unix。那么先看看这个函数的用法吧.php手册上是这么说的:

$pid = pcntl_fork();

if ($pid == -1) {

die('could not fork');

} else if ($pid) {

// we are the parent

pcntl_wait($status); //Protect against Zombie children

} else {

// we are the child

}

?>

通过pcntl_fork创建一个子进程,如果返回值是-1的话,那么说明子进程创建失败.创建成功的进程id会返回给父进程,0返回给子进程.不好理解吧,所以应该这样写:

$pid = pcntl_fork();

if($pid == -1){

//创建失败咱就退出呗,没啥好说的

die('could not fork');

}

else{

if($pid){

//从这里开始写的代码是父进程的,因为写的是系统程序,记得退出的时候给个返回值

exit(0);

}

else{

//从这里开始写的代码都是在新的进程里执行的,同样正常退出的话,最好也给一个返回值

exit(0);

}

}

?>这样一改好理解多了,如果你父进程希望知道子进程正常退出的话,可以加上前面的pcntl_wait。

2.通过stream_socket_client 方式

function sendStream(){

$english_format_number = number_format($number, 4, '.', '');

echo $english_format_number;

exit();

$timeout = 10;

$result = array();

$sockets = array();

$convenient_read_block = 8192;

$host = "test.local.com";

$sql = "select waybill_id,order_id from xm_waybill where status>40 order by update_time desc limit 1 ";

$data = Yii::app()->db->createCommand($sql)->queryAll();

$id = 0;

foreach ($data as $k => $v) {

if ($k % 2 == 0) {

$send_data[$k]['body'] = NoticeOrder::getSendData($v['waybill_id']);

} else {

$send_data[$k]['body'] = array($v['order_id'] => array('extra' => 16));

}

$data = json_encode($send_data[$k]['body']);

$s = stream_socket_client($host . ":80", $errno, $errstr, $timeout, STREAM_CLIENT_ASYNC_CONNECT | STREAM_CLIENT_CONNECT);

if ($s) {

$sockets[$id++] = $s;

$http_message = "GET /php/test.php?data=" . $data . " HTTP/1.0\r\nHost:" . $host . "\r\n\r\n";

fwrite($s, $http_message);

} else {

echo "Stream " . $id . " failed to open correctly.";

}

}

while (count($sockets)) {

$read = $sockets;

stream_select($read, $w = null, $e = null, $timeout);

if (count($read)) {

/* stream_select generally shuffles $read, so we need to

compute from which socket(s) we're reading. */

foreach ($read as $r) {

$id = array_search($r, $sockets);

$data = fread($r, $convenient_read_block);

if (strlen($data) == 0) {

echo "Stream " . $id . " closes at " . date('h:i:s') . ".
";

fclose($r);

unset($sockets[$id]);

} else {

$result[$id] = $data;

}

}

} else {

/* A time-out means that *all* streams have failed

to receive a response. */

echo "Time-out!\n";

break;

}

}

print_r($result);

}

3、通过多进程代替多线程function daemon($func_name,$args,$number){

while(true){

$pid=pcntl_fork();

if($pid==-1){

echo "fork process fail";

exit();

}elseif($pid){//创建的子进程

static $num=0;

$num++;

if($num>=$number){

//当进程数量达到一定数量时候,就对子进程进行回收。

pcntl_wait($status);

$num--;

}

}else{ //为0 则代表是子进程创建的,则直接进入工作状态

if(function_exists($func_name)){

while (true) {

$ppid=posix_getpid();

var_dump($ppid);

call_user_func_array($func_name,$args);

sleep(2);

}

}else{

echo "function is not exists";

}

exit();

}

}

}

function worker($args){

//do something

}

daemon('worker',array(1),2);

二、真正实现php多线程的方法

php真正的多线程实现方式,通过安装php的扩展 pthread 可以做到。

但是这个下载的是 版本3 也就是php 7 才能用的,我们需要使的是 版本2

989861a86755a881b876e1d0b113e9c3.png

然后刷新的页面如下,拖到最底部:

da5256673f6881b7adc1b6d396c56c70.png

a43995f27ece1d9d87c88da6f47ea52c.png

下一页找到版本2的

下载下来,这个v2 才是php5才可以使用的

下载下来,安装:

或者,您直接这样下载:cd /tools

wget https://github.com/krakjoe/pthreads/archive/v2.0.10.zip

unzip v2.0.10.zip

cd pthreads-2.0.10

/usr/local/php/bin/phpize

./configure --with-php-config=/usr/local/php/bin/php-config

make

make install注意:您的php 在编译的时候需要开启 –enable-maintainer-zts./configure --prefix=/usr/local/php --disable-fileinfo --enable-fpm --with-config-file-path=/etc --with-config-file-scan-dir=/etc/php.d --with-openssl --with-zlib --with-curl --enable-ftp --with-gd --with-xmlrpc --with-jpeg-dir --with-png-dir --with-freetype-dir --enable-gd-native-ttf --enable-mbstring --with-mcrypt=/usr/local/libmcrypt --enable-zip --with-mysql=/usr/local/mysql --without-pear --enable-maintainer-ztsvim /etc/php.ini

添加

extension=pthreads.so重启php

/etc/init.d/php-fpm restart



推荐阅读
  • Python 数据可视化实战指南
    本文详细介绍如何使用 Python 进行数据可视化,涵盖从环境搭建到具体实例的全过程。 ... [详细]
  • 在多线程并发环境中,普通变量的操作往往是线程不安全的。本文通过一个简单的例子,展示了如何使用 AtomicInteger 类及其核心的 CAS 无锁算法来保证线程安全。 ... [详细]
  • Java高并发与多线程(二):线程的实现方式详解
    本文将深入探讨Java中线程的三种主要实现方式,包括继承Thread类、实现Runnable接口和实现Callable接口,并分析它们之间的异同及其应用场景。 ... [详细]
  • 本文总结了在SQL Server数据库中编写和优化存储过程的经验和技巧,旨在帮助数据库开发人员提升存储过程的性能和可维护性。 ... [详细]
  • 在 Ubuntu 中遇到 Samba 服务器故障时,尝试卸载并重新安装 Samba 发现配置文件未重新生成。本文介绍了解决该问题的方法。 ... [详细]
  • 本文详细介绍了数据库并发控制的基本概念、重要性和具体实现方法。并发控制是确保多个事务在同时操作数据库时保持数据一致性的关键机制。文章涵盖了锁机制、多版本并发控制(MVCC)、乐观并发控制和悲观并发控制等内容。 ... [详细]
  • MySQL Decimal 类型的最大值解析及其在数据处理中的应用艺术
    在关系型数据库中,表的设计与SQL语句的编写对性能的影响至关重要,甚至可占到90%以上。本文将重点探讨MySQL中Decimal类型的最大值及其在数据处理中的应用技巧,通过实例分析和优化建议,帮助读者深入理解并掌握这一重要知识点。 ... [详细]
  • 本文介绍了如何使用Python的Paramiko库批量更新多台服务器的登录密码。通过示例代码展示了具体实现方法,确保了操作的高效性和安全性。Paramiko库提供了强大的SSH2协议支持,使得远程服务器管理变得更加便捷。此外,文章还详细说明了代码的各个部分,帮助读者更好地理解和应用这一技术。 ... [详细]
  • 利用REM实现移动端布局的高效适配技巧
    在移动设备上实现高效布局适配时,使用rem单位已成为一种流行且有效的技术。本文将分享过去一年中使用rem进行布局适配的经验和心得。rem作为一种相对单位,能够根据根元素的字体大小动态调整,从而确保不同屏幕尺寸下的布局一致性。通过合理设置根元素的字体大小,开发者可以轻松实现响应式设计,提高用户体验。此外,文章还将探讨一些常见的问题和解决方案,帮助开发者更好地掌握这一技术。 ... [详细]
  • 您的数据库配置是否安全?DBSAT工具助您一臂之力!
    本文探讨了Oracle提供的免费工具DBSAT,该工具能够有效协助用户检测和优化数据库配置的安全性。通过全面的分析和报告,DBSAT帮助用户识别潜在的安全漏洞,并提供针对性的改进建议,确保数据库系统的稳定性和安全性。 ... [详细]
  • 本文探讨了在PHP中实现MySQL分页查询功能的优化方法与实际应用。通过详细分析分页查询的常见问题,提出了多种优化策略,包括使用索引、减少查询字段、合理设置缓存等。文章还提供了一个具体的示例,展示了如何通过优化模型加载和分页参数设置,显著提升查询性能和用户体验。 ... [详细]
  • 如何使用 `org.eclipse.rdf4j.query.impl.MapBindingSet.getValue()` 方法及其代码示例详解 ... [详细]
  • 性能测试中的关键监控指标与深入分析
    在软件性能测试中,关键监控指标的选取至关重要。主要目的包括:1. 评估系统的当前性能,确保其符合预期的性能标准;2. 发现软件性能瓶颈,定位潜在问题;3. 优化系统性能,提高用户体验。通过综合分析这些指标,可以全面了解系统的运行状态,为后续的性能改进提供科学依据。 ... [详细]
  • 本文深入解析了通过JDBC实现ActiveMQ消息持久化的机制。JDBC能够将消息可靠地存储在多种关系型数据库中,如MySQL、SQL Server、Oracle和DB2等。采用JDBC持久化方式时,数据库会自动生成三个关键表:`activemq_msgs`、`activemq_lock`和`activemq_ACKS`,分别用于存储消息数据、锁定信息和确认状态。这种机制不仅提高了消息的可靠性,还增强了系统的可扩展性和容错能力。 ... [详细]
  • 如何撰写数据分析师(包括转行者)的面试简历?
    CDA数据分析师团队出品,作者:徐杨老师,编辑:Mika。本文将帮助您了解如何撰写一份高质量的数据分析师简历,特别是对于转行者。 ... [详细]
author-avatar
一二三
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有