pthreads v3下一些坑和需要注意的地方
一、子线程无法访问父线程的全局变量,但父线程可以访问子线程的变量
<?php class Task extends Thread
{
public $data; public function run()
{
global $num;
var_dump($num);
$this->data = 'abc'; //这是打印null
var_dump($GLOBALS);
$GLOBALS['test'] = 'def';
}
} //主线程中的全局变量,子线程中是无法访问的
//php中创建子线程,它会有一个单独的堆,运行在单独的地址空间中
//并不能像有些语言中,子线程是可以访问的到主线程中的变量的。
$num = 666; $GLOBALS['test'] = 'test'; $t = new Task();
$t->start() && $t->join(); //主线程可以访问子线程的变量
var_dump($t->data);
结果如下:

二、子线程无法修改父线程的变量
<?php class Task extends Thread
{
private $data; public function __construct(&$data)
{
$this->data = $data;
} public function run()
{
echo "task data : ", $this->data, "\n";
$this->data = 'def';
echo "task data : ", $this->data, "\n";
}
} $data = 'abc'; //我们这里传入的是引用
$t = new Task($data);
$t->start() && $t->join(); //但是$data变量数据并没改变
//这说明我们通过构造函数传入Task对象中的$data只是一个拷贝
//子线程中并不能够操作主线程中的变量
var_dump($data);
结果如下:

三、pthreads v3版本可以设置成员为匿名函数
<?php class Task extends Thread
{
private $call; public function __construct()
{
//pthreads v3版本好像可以设置成员为匿名函数
//在v2版本中好像是不可以的
$this->call = function ($param1, $param2) {
echo "task call param1 : {$param1} param2 : {$param2}\n";
};
} public function run()
{
//直接调用成员匿名函数
($this->call)("hello", "world");
}
} $t = new Task();
$t->start() && $t->join();
结果如下:

四、对于数据库连接资源,我们需要声明为静态成员
<?php class Task extends Thread
{
private $db; public function __construct()
{
//注意这里会报错,不能对PDO实例进行序列化或反序列化
$this->db = new PDO('mysql:dbname=test;host=192.168.33.226', 'root', '');
} public function run()
{
$result = $this->db->query("select id,name from tb_user");
while ($row = $result->fetch(PDO::FETCH_ASSOC)) {
echo "{$row['id']}\t{$row['name']}\n";
}
}
} $t = new Task();
$t->start() && $t->join();
结果如下:

代码修改如下:
<?php class Task extends Thread
{
//我们需要把数据库连接资源,声明为静态成员,然后调用静态方法进行创建
private static $db; //我们直接在__construct()构造函数里对$db进行实例化,好像会是null,有兴趣的可以试试
public static function getConn()
{
if (!is_resource(self::$db)) {
self::$db = new PDO('mysql:dbname=test;host=192.168.33.226', 'root', '');
}
return self::$db;
} public function run()
{
$result = self::getConn()->query("select id,name from tb_user");
while ($row = $result->fetch(PDO::FETCH_ASSOC)) {
echo "{$row['id']}\t{$row['name']}\n";
}
}
} $t = new Task();
$t->start() && $t->join();
结果如下:

最后说明,不排除pthreads以后的版本升级上面的有些问题不会出现。
我的php版本是7.2.4,pthreads的版本是3.1.7dev


pthreads v3下一些坑和需要注意的地方的更多相关文章
- pthreads v3下的Volatile介绍与使用
由于pthreads v3中引入了Threaded对象自动不变性的概念,所以当我们在构造函数中给成员设置为数组时,在其他地方就无法对成员再次改写了. 例子如下: <?php //pthreads ...
- pthreads v3下的同步处理synchronized
之所以会用到同步,是因为如果多个线程中对同一个资源进行操作时,会发生混乱. 比如2个线程对变量进行加1操作,第1个线程还没来的及改写数据时,第2个线程就对变量进行操作了,那变量最终的结果就是未知的,这 ...
- pthreads v3下的worker和pool的使用
有些人会想,明明用thread已经可以很好的工作了,为什么还要搞个worker和pool? 之所以要用到worker和pool还是因为效率,因为系统创建一个新线程代价是比较昂贵,每个创建的线程会复制当 ...
- pthreads v3在centos7下的安装与配置
我的centos版本是7.4.1708,php的版本是7.2.4(注意要是线程安全版),如下图所示: 首先我们在如下网址下载好pthreads的源码: http://pecl.php.net/pack ...
- JDK动态代理给Spring事务埋下的坑!
一.场景分析 最近做项目遇到了一个很奇怪的问题,大致的业务场景是这样的:我们首先设定两个事务,事务parent和事务child,在Controller里边同时调用这两个方法,示例代码如下: 1.场景A ...
- (转)面试必备技能:JDK动态代理给Spring事务埋下的坑!
一.场景分析 最近做项目遇到了一个很奇怪的问题,大致的业务场景是这样的:我们首先设定两个事务,事务parent和事务child,在Controller里边同时调用这两个方法,示例代码如下: 1.场景A ...
- .NET Core 图片操作在 Linux/Docker 下的坑
一.前言 .NET Core 目前更新到2.2了,但是直到现在在 .NET Core 本身依然不包括和图片有关的 Image.Bitmap 等类型.对于图片的操作在我们开发中很常见,比如:生成验证码. ...
- HttpClient在多线程环境下踩坑总结
问题现场 在多线程环境下使用HttpClient组件对某个HTTP服务发起请求,运行一段时间之后发现客户端主机CPU利用率呈现出下降趋势,而不是一个稳定的状态. 而且,从程序日志中判断有线程处于han ...
- C# rtsp 转码rtmp nginx踩下的坑
最近有一个项目, 进行步态识别的时候,同时需要显示摄像的实时画面.对于 Winform 显示画面,之前针对 rtmp 流的时候,是先写一个HTML 网页加载视频流,然后在IIS上发布,将地址直接赋值给 ...
随机推荐
- mysql分表实战
本文主要讲述如何使用存储过程完成本表.并不讨论其他问题.首先我们得看看手册上关于meger引擎的说明: MERGE存储引擎,也被认识为MRG_MyISAM引擎,是一个相同的可以被当作一个来用的MyIS ...
- arcgis python arcpy add data script添加数据脚本
arcgis python arcpy add data script添加数据脚本mxd = arcpy.mapping.MapDocument("CURRENT")... df ...
- Swift 2.0学习笔记(Day 16)——字典集合
Swift字典表示一种非常复杂的集合,允许按照某个键来访问元素.字典是由两部分集合构成的,一个是键(key)集合,一个是值(value)集合.键集合是不能有重复元素的,而值集合是可以重复的,键和值是成 ...
- python 引用和对象理解(转)
引用和对象分离 从最开始的变量开始思考: 在python中,如果要使用一个变量,不需要提前进行声明,只需要在用的时候,给这个变量赋值即可 (这个和C语言等静态类型语言不同,和python为动态类型有关 ...
- python小数据池概念以及具体范围
= 赋值符号: == 比较值是否相等: is 比较,比较的是内存地址 ID(内容) 数字,字符串的小数据池 小数据池现象产生的原因,作用: 为了节省内存空间. &l ...
- 什么是webFlux
什么是webFlux 左侧是传统的基于Servlet的Spring Web MVC框架,右侧是5.0版本新引入的基于Reactive Streams的Spring WebFlux框架,从上到下依次是R ...
- node-rsa
[node-rsa] 引用 var NodeRSA = require('node-rsa') 生成一个私钥长度为512的key(同时生成公钥) var key = new NodeRSA({b: 5 ...
- VBox添加虚拟磁盘挂载
1. 关闭虚拟机,然后在设置里面选择添加虚拟硬盘 2.lsblk检查存在10G sdb虚拟磁盘 fdisk -l 检查 /dev/sdb 尚没有分区 3.磁盘分区 4.检查分区状况lsblk 5.格式 ...
- ELK 日志学习
一.Elasticsearch 安装(所有的版本均使用5.5.0 ,且版需要相同 logstash \ kibana \ filebeat ) 官网下载地址:https://www.elastic.c ...
- Python基础之字典操作
字典 字典的优点: dict key 必须是不可变数据类型,可哈希, value:任意数据类型. dict 优点:二分查找去查询 存储大量的关系型数据 特点:无序的(指的是不可人为的去改变顺序) 数据 ...