[ Perl ] 多线程并发编程
https://www.cnblogs.com/yeungchie/
记录一些常用的 模块 / 方法 。
多线程
使用模块 threads
use 5.010;
use threads;
sub func {
my $id = shift;
sleep 1;
print "This is thread - $id\n";
}
创建线程
new
sub start {
my $id = shift;
my $t = new threads \&func, $id;
return $t;
}
create
sub start {
my $id = shift;
my $t = new threads \&func, $id;
return $t;
}
async
sub start {
my $id = shift;
my $t = async { &func( $id ) };
return $t;
}
线程收尸
- 阻塞
join
&start( 'join' )->join;
say 'Done';
This is thread - join
Done
# 父线程被子线程阻塞,成功收尸。
- 非阻塞
detach
&start( 'detach' )->detach;
say 'Done';
Done
# 由于非阻塞,父线程已经退出,子线程变成孤儿线程,无法收尸。
数据共享
使用模块 threads::shared
use threads::shared;
标记共享变量
有几种不同的写法
- 依次标记
:shared
my $scalar :shared;
my @array :shared;
my %hash :shared;
- 批量标记
:shared
my ( $scalar, @array, %hash ) :shared;
- 用函数标记
share()
my ( $scalar, @array, %hash );
share $scalar;
share @array;
share %hash;
克隆 shared_clone
向共享的变量中加入新的元素时,需要注意的地方。
my @newArray = qw( YEUNG CHIE 1 2 3 );
my $clone = shared_clone [@newArray];
push @array, $clone;
$hash{ keyName } = $clone;
锁 lock
多个线程同时编辑一个共享变量时,需要注意的地方。
经典的取钱问题:
1 - 输出额度$amount
= 500
2 - func() 函数模拟取钱,每次取 300
3 - 当$amount
< 300 时,则无法取钱
- 没加锁的情况
my $amount :shared = 500;
sub func {
unless ( $amount < 300 ) {
sleep 1; # 睡眠一秒模拟延迟
$amount -= 300;
}
}
# 这里两个线程模拟,两次取钱同时进行
my $t1 = new threads \&func;
my $t2 = new threads \&func;
$t1->join;
$t2->join;
say $amount;
-100
# 结果被取了两次,剩余额度为 -100
- 加了锁的情况
调整一下子函数 func()
, 加个锁。
...
sub func {
lock $amount;
unless ( $amount < 300 ) {
sleep 1;
$amount -= 300;
}
}
...
200
# 结果正确
线程队列
使用模块 Thread::Queue
use Thread::Queue;
创建队列
my $queue = new Thread::Queue;
入队 enqueue
my $var = 'YEUNG';
$queue->enqueue( $var );
$queue->enqueue( qw( CHIE 1 2 3 ) );
出队 dequeue
- 默认出队一个项目
say $queue->dequeue;
YEUNG
- 指定多个项目出队
say for $queue->dequeue( 3 );
CHIE
1
2
非阻塞出队 dequeue_nb
- 如果是阻塞出队
my $queue = new Thread::Queue qw( YEUNG CHIE );
say while $_ = $queue->dequeue;
YEUNG
CHIE
# 程序会卡在这里,等待队列中新的项目加入
- 使用非阻塞出队
my $queue = new Thread::Queue qw( YEUNG CHIE );
say while $_ = $queue->dequeue_nb;
YEUNG
CHIE
剩余 pending
pending
方法可以返回未出队的项目数量。
my $queue = new Thread::Queue qw( YEUNG CHIE );
say $queue->dequeue;
say $queue->pending;
say $queue->dequeue;
say $queue->pending;
YEUNG
1
CHIE
0
查看 peek
只是看看但是不出队。
my $queue = new Thread::Queue qw( YEUNG CHIE );
say $queue->peek;
say $queue->pending;
say $queue->peek( 2 );
say $queue->pending;
YEUNG
2
CHIE
2
入队结束 end
除了上面用 dequeue_nb
非阻塞出队,之外还可以用 end
方法来
my $queue = new Thread::Queue qw( YEUNG CHIE );
$queue->end;
say while $_ = $queue->dequeue;
# 这样虽然没有用
dequeue_nb
方法,程序也不会卡住了。
不过这个方法需要模块版本 >= 3.01
,一般系统自带 Perl 是不支持的,但是我们也可以自己来实现这个效果:
共享变量
共享一个全局变量标记入队结束。
my $endFlag :shared;
生产者线程
当入队结束时,
$endFlag
赋值为真。$endFlag = 1;
消费者线程
循环操作非阻塞出队。
while ( 1 ) {
my $item = $queue->dequeue_nb;
if ( defined $item ) {
say $item;
}
else {
# 当出队失败且入队结束时,退出循环
last if $endFlag;
}
}
线程信号量
使用模块 Thread::Semaphore
use Thread::Semaphore;
线程池
使用模块 Thread::Pool
use Thread::Pool;
参考资料/拓展
- threads - Perl interpreter-based threads - metacpan.org
- threads::shared - Perl extension for sharing data structures between threads - metacpan.org
- Thread::Queue - Thread-safe queues - metacpan.org
- Thread::Semaphore - Thread-safe semaphores - metacpan.org
- Thread::Pool - group of threads for performing similar jobs - metacpan.org
[ Perl ] 多线程并发编程的更多相关文章
- Java 多线程并发编程一览笔录
Java 多线程并发编程一览笔录 知识体系图: 1.线程是什么? 线程是进程中独立运行的子任务. 2.创建线程的方式 方式一:将类声明为 Thread 的子类.该子类应重写 Thread 类的 run ...
- 【收藏】Java多线程/并发编程大合集
(一).[Java并发编程]并发编程大合集-兰亭风雨 [Java并发编程]实现多线程的两种方法 [Java并发编程]线程的中断 [Java并发编程]正确挂起.恢复.终止线程 [ ...
- java多线程并发编程与CPU时钟分配小议
我们先来研究下JAVA的多线程的并发编程和CPU时钟振荡的关系吧 老规矩,先科普 我们的操作系统在DOS以前都是单任务的 什么是单任务呢?就是一次只能做一件事 你复制文件的时候,就不能重命名了 那么现 ...
- Java基础系列篇:JAVA多线程 并发编程
一:为什么要用多线程: 我相信所有的东西都是以实际使用价值而去学习的,没有实际价值的学习,学了没用,没用就不会学的好. 多线程也是一样,以前学习java并没有觉得多线程有多了不起,不用多线程我一样可以 ...
- Java 多线程并发编程
导读 创作不易,禁止转载! 并发编程简介 发展历程 早起计算机,从头到尾执行一个程序,这样就严重造成资源的浪费.然后操作系统就出现了,计算机能运行多个程序,不同的程序在不同的单独的进程中运行,一个进程 ...
- Java多线程并发编程/锁的理解
一.前言 最近项目遇到多线程并发的情景(并发抢单&恢复库存并行),代码在正常情况下运行没有什么问题,在高并发压测下会出现:库存超发/总库存与sku库存对不上等各种问题. 在运用了 限流/加锁等 ...
- java多线程并发编程
Executor框架 Executor框架是指java 5中引入的一系列并发库中与executor相关的一些功能类,其中包括线程池,Executor,Executors,ExecutorService ...
- Java多线程并发编程一览笔录
线程是什么? 线程是进程中独立运行的子任务. 创建线程的方式 方式一:将类声明为 Thread 的子类.该子类应重写 Thread 类的 run 方法 方式二:声明实现 Runnable 接口的类.该 ...
- Java 多线程并发编程面试笔录一览
知识体系图: 1.线程是什么? 线程是进程中独立运行的子任务. 2.创建线程的方式 方式一:将类声明为 Thread 的子类.该子类应重写 Thread 类的 run 方法 方式二:声明实现 Runn ...
随机推荐
- MySQL—分页查询
分页查询 应用场景:当要显示的数据,当一页显示不全,有很多的数据时,就需要分页提交sql请求 语法:select 查询列表 from表名 [ join type join 表2 on连接条件 whe ...
- Java基础 - 注解详解
What - 什么是注解? Annontation是Java5开始引入的新特征,中文名称叫注解.它提供了一种安全的类似注释的机制,用来将任何的信息或元数据(metadata)与程序元素( ...
- JavaScript 事件循环(1) —— 从 setTimeout 说起
转变认知 setTimeout 可能是很多前端工程师爱用的方法,它可以使得一段代码延迟执行,例如: setTimeout(() => console.log('A'), 1000); // 在1 ...
- Git常见错误整理
参考文章 git 排错 fatal: 'git status --porcelain' failed in submodule abi/cpp 1 fatal: 'git status --porce ...
- Tomcat配置Context.xml上下文遇到的坑
注意事项: 1. 在主机的 appBase 之外找到 WAR 和/或目录,并使用带有 docBase 属性的 context.xml 文件来定义它.避免双重部署导致出现不可预知的问题 {context ...
- Zookeeper 文件系统 ?
Zookeeper 提供一个多层级的节点命名空间(节点称为 znode).与文件系统不 同的是,这些节点都可以设置关联的数据,而文件系统中只有文件节点可以存放 数据而目录节点不行. Zookeeper ...
- 可以直接调用 Thread 类的 run ()方法么?
当然可以.但是如果我们调用了 Thread 的 run()方法,它的行为就会和普通的方 法一样,会在当前线程中执行.为了在新的线程中执行我们的代码,必须使用 Thread.start()方法.
- java后台解决上传图片翻转90的问题,有demo,经过测试可用
1.需要加入 依赖 metadata-extractor.jar 依赖如下 <dependencies> <!-- Extracts Exif, IPTC, XMP, ICC and ...
- (stm32f103学习总结)—GPIO结构
一.GPIO基本结构 二.GPIO工作模式 输入模式 输入浮空 输入上拉 输入下拉 模拟输入 输出模式 开漏输出 开漏复用功能 推挽式输出 推挽式复用功能 库函数中所对应的代码 1 typedef e ...
- mplab使用小知识
选择Debugger->Select Tool->MPLAB SIM可以使用MPALB中的软件调试 StopWatch可以观察程序运行时间 注意:在测试时需要注意红圈内晶振是不是和单片机上 ...