Perl6多线程1: new / run

先看一个小例子:

sub A($name = 3) {
  #默认参数
  say $name;
}
sub B(:name($name)) {
  #默认参数为 any
  say $name;
}
A();
A(100);
B();
B(name => ‘root‘);

这是正常的调用方式。

再看如下代码:

sub A($name) {
  #默认参数
  say $name;
}
sub B() {
  say ‘BBBBBBBB‘;
}
A(1);
B;
B();
&A(123);
&B();

可以看到, 当函数没参数时, 可以直接用 B() 来调用,如果有参数时, 可以 A() / &A() 调用。

再看一下例子:

sub A($name) {
  #默认参数
  say $name;
}
sub B() {
  say ‘BBBBBBBB‘;
}
say &A;
say &A(123);
say &B;
say &B();

从上例可以看出, &A / &B 并不是调用函数, 而是显示这个函数的地址(也就是这个函数的代码块形式)。

这个 &B 可以看做是函数的地址。

如下代码:

sub A($name) {
  #默认参数
  say $name;
  $name();
}
sub B() {
  say ‘BBBBBBBB‘;
}

A(&B);

把 &B 传送进去, 在 A 函数中调用 $name() 就相当于: &B(), 也就是调用函数B了。

对于 A 函数的定义, 可以 把 $name 改为 &name, 指定参数 name 为代码块形式的参数:

sub A(&name) {
  #默认参数
  say &name;
  #$name();
}
sub B() {
  say ‘BBBBBBBB‘;
}

A(‘B‘);

#输出:
C:\p6>perl6 t.p6
Type check failed in binding to parameter ‘&name‘; expected Callable but got Str
 ("B")
  in sub A at t.p6 line 1
  in block <unit> at t.p6 line 10

这时调用会出错, 因为我们指定参数为 &name, 也就是代码块的形式参数。

我们调用时, 发送代码块就行:

sub A(&name) {
  #默认参数
  say &name;
  &name();
}
sub B() {
  say ‘BBBBBBBB‘;
}

A(sub test {say ‘abc‘;});

这时程序可正常运行。

像下面这样:

sub A(&name) {
  #默认参数
  say &name;
  &name();
}
sub B() {
  say ‘BBBBBBBB‘;
}

A(sub test {say ‘abc‘;});
A(&B);
#输出:
C:\p6>perl6 t.p6
sub test () { #`(Sub|89260968) ... }
abc
sub B () { #`(Sub|89261120) ... }
BBBBBBBB

C:\p6>

说了这么多, 关键问题只是说明函数调用时, 参数可以为代码块形式。 可以用普通形式的参数($name)接收代码块, 或指定参数为代码块形式(&name)来接收代码块。

最后回到文章正题:perl6 多线程 。

Thread 类内置, 不用另外安装。

线程创建方法为:

method new(:&code!, Bool :$app_lifetime = False, Str :$name = ‘<anon>‘ --> Thread:D)

new函数有个参数: :&code! 就是上面所说的例子的形式, 它是个字典形式的代码块, 调用时这样:

code => 代码块
code => &B
code => {say ‘Thread‘;}

对于第二个 app_lifetime参数,是用来设置线程用的。 当设置为 true 时, 主进程退出后线程跟着退出。当设置为 false 时, 线程只有它运行结束时才自动退出。

name 是指定一个标识些线程的字符串。

创建一个线程后, 线程不会自动运行, 我们可以用 run 方法运行线程。

看如下代码, app_lifetime 设置为 True:

sub B() {
  for 1..5 {
    say $_;
    sleep(2);
  }
  say "线程退出!";
}

my $t = Thread.new(code => &B, :app_lifetime, :name<thread_B>);
$t.run;
say "主进程退出!";

结果:

C:\p6>perl6 t.p6
主进程退出!

C:\p6>

可以看到, 主进程退出后线程也跟着退出了。

我们把 app_lifetime 设置为 false再看看:

sub B() {
  for 1..5 {
    say $_;
    sleep(2);
  }
  say "线程退出!";
}

my $t = Thread.new(code => &B, :!app_lifetime, :name<thread_B>);
$t.run;
say "主进程退出!";

结果:

C:\p6>perl6 t.p6
主进程退出!
1
2
3
4
5
线程退出!

C:\p6>

可以看到主进程结束后线程还是会运行。

总结:

Thread.new创建线程。

code 参数指定代码块。

app_lifetime 设置线程是否与主进程一同退出。

时间: 2024-10-06 17:20:25

Perl6多线程1: new / run的相关文章

多线程 start 和 run 方法到底有什么区别?

昨天栈长介绍了<Java多线程可以分组,还能这样玩!>线程分组的妙用.今天,栈长会详细介绍 Java 中的多线程 start() 和 run() 两个方法,Java 老司机请跳过,新手或者对这两个不是很理解的可以继续往下看. 首先要知道实现多线程最基本的两种方式: 1.继承 java.lang.Thread 类: 2.实现 java.lang.Runnable接口: 其中 Thread 类也是实现了 Runnable 接口,而 Runnable 接口定义了唯一的一个 run() 方法,所以基于

Perl6多线程3: Promise start / in / await

创建一个Promise 并自动运行: my $p = Promise.start({say 'Hello, Promise!'}); 如果把代码改成如下, 我们会发现什么也没打印: my $p = Promise.start({sleep 2;say 'Hello, Promise!'}); 匿名函数 sleep 2 秒, 这时, 它还没运行完, 主程序就退出了, 这里 promise也跟着退出, 所以什么也没打印. 我们可以改写成这样: my $p = Promise.start({sleep

Perl6多线程2: Promise new/keep/bread/status/result

来源于个人理解的翻译. 创建一个 promise: my $p = Promise.new; 可以打印运行 的Promise 状态: my $p = Promise.new(); $p.then({say 'hello, world'}); say $p.status; 上面的promise创建好后, 当 $p 状态为 kept或broken 时, 会执行 then 里面的 匿名函数. 但是, 上面的$p状态总是为: Plannd. 所以, 那个 hello, world 总是不能打印. 那怎么

多线程start()与run()的区别

概要 1.start()与run()介绍 2.start()与run()源码查看 3.start()与run()测试 start()与run()介绍 1.通过我们在启动线程的时候使用的start,为什么不用run呢? 因为start()会新开一个线程来执行:而run只是一个普通想法,相当于当前线程来调用,不会启动新线程: 2.start()只能调用一次,run()可以调用多次 start()与run()源码查看 1.先来看下start()的源码,start()启动线程其实是通过start0()启

多线程01

进程: 是一个正在执行中的程序 每一个进程都有一个执行顺序.该顺序是一个执行路径,或者叫一个控制线程 就是进程中的一个独立的控制单元,线程在控制着进程的执行jvm vm 启动的时候有一个进程java.exe 该进程中至少一个线程负责java程序的执行.而且这个西拿出一些的代码存在于 线程的定义 定义一个类 extend Threed 重写 three的润()方法,调用线程的start()方法 多线程的随机性 run()的覆盖 Thread类用于描述线程.该类就定义了一个功能,用于存存储要运行的代

黑马程序员 - 多线程

------Java培训.Android培训.iOS培训..Net培训.期待与您交流! ------- 多线程进程:一个正在执行中的程序,每一个进行执行,都有一个执行顺序,该顺序就是一个执行路径,或者加一个执行单元线程:就是进程中的一个独立的执行路径,一个进程中至少有一个线程.java vm启动的时候会有一个进程java.exe.该进程中至少一个线程负责执行java程序的执行,而且这个线程运行的代码存在于main方法中.该线程称之为主线程.扩展:其中更细节说明java vm启动不止一个线程,还有

java多线程详解

转自:线程间通信.等待唤醒机制.生产者消费者问题(Lock,Condition).停止线程和守护线程.线程优先级 1  线程间通信 1.1  线程间通信 其实就是多个线程在操作同一个资源,但是操作的动作不同. 比如一个线程给一个变量赋值,而另一个线程打印这个变量. 1.2  等待唤醒机制 wait():将线程等待,释放了CPU执行权,同时将线程对象存储到线程池中. notify():唤醒线程池中一个等待的线程,若线程池有多个等待的线程,则任意唤醒一个. notifyAll():唤醒线程池中,所有

Java的多线程 简单入门

Java的多线程 简单入门 首先可以先搞清楚什么是程序.进程.线程,以及它们之间的关系: 定义: 一 程序只是一组指令的有序集合,它是静态的 二 进程是具有一定独立功能的程序关于某个数据集合上的一次运行活动,是系统进行资源分配和调度的一个独立单位: 三 线程是进程的一个实体,是CPU调度和分派的基本单位,它是比进程更小的能独立运行的基本单位.线程自己基本上不拥有系统资源,只拥有一点在运行中必不可少的资源(如程序计数器,一组寄存器和栈),一个线程可以创建和撤销另一个线程: 进程与线程区别与联系 (

java 多线程例子

java 多线程例子 编写具有多线程能力的程序经常会用到的方法有: run(), start(), wait(), notify(), notifyAll(), sleep(), yield(), join() 还有一个重要的关键字:synchronized 本文将对以上内容进行讲解. 一:run() 和start() 示例1: public class ThreadTest extends Thread {public void run() {for (int i = 0; i < 10; i