先看一个小例子:

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

这是正常的调用方式。

再看如下代码:

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

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

再看一下例子:

sub A($name) {
#默认参数
say $name;
}
sub B() {
say 'BBBBBBBB';
}
say &A;
say &A();
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
in block <unit> at t.p6 line

这时调用会出错, 因为我们指定参数为 &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 .. {
say $_;
sleep();
}
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 .. {
say $_;
sleep();
}
say "线程退出!";
} my $t = Thread.new(code => &B, :!app_lifetime, :name<thread_B>);
$t.run;
say "主进程退出!";

结果:

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

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

总结:

Thread.new创建线程。

code 参数指定代码块。

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

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

  1. JAVA之旅(十二)——Thread,run和start的特点,线程运行状态,获取线程对象和名称,多线程实例演示,使用Runnable接口

    JAVA之旅(十二)--Thread,run和start的特点,线程运行状态,获取线程对象和名称,多线程实例演示,使用Runnable接口 开始挑战一些难度了,线程和I/O方面的操作了,继续坚持 一. ...

  2. 多线程-Thread的run()与start()的区别

    总结: 1) start: 用start方法来启动线程,真正实现了多线程运行,这时无需等待run方法体代码执行完毕而直接继续执行下面的代码.通过调用Thread类的start()方法来启动一个线程,这 ...

  3. Java Thread 的 run() 与 start() 的区别

    Java Thread 的使用 Java Thread 的 run() 与 start() 的区别 Java Thread 的 sleep() 和 wait() 的区别             1. ...

  4. Unity多线程(Thread)和主线程(MainThread)交互使用类——Loom工具分享

    Unity多线程(Thread)和主线程(MainThread)交互使用类——Loom工具分享 By D.S.Qiu 尊重他人的劳动,支持原创,转载请注明出处:http.dsqiu.iteye.com ...

  5. Java多线程01(Thread类、线程创建、线程池)

    Java多线程(Thread类.线程创建.线程池) 第一章 多线程 1.1 多线程介绍 1.1.1 基本概念 进程:进程指正在运行的程序.确切的来说,当一个程序进入内存运行,即变成一个进程,进程是处于 ...

  6. Java中实现多线程继承Thread类与实现Runnable接口的区别

    Java中线程的创建有两种方式: 1.  通过继承Thread类,重写Thread的run()方法,将线程运行的逻辑放在其中 2.  通过实现Runnable接口,实例化Thread类 在实际应用中, ...

  7. 多线程(Thread、线程创建、线程池)

      第1章 多线程 1.1 多线程介绍 学习多线程之前,我们先要了解几个关于多线程有关的概念. 进程:进程指正在运行的程序.确切的来说,当一个程序进入内存运行,即变成一个进程,进程是处于运行过程中的程 ...

  8. C#异步和多线程以及Thread、ThreadPool、Task区别和使用方法

    本文的目的是为了让大家了解什么是异步?什么是多线程?如何实现多线程?对于当前C#当中三种实现多线程的方法如何实现和使用?什么情景下选用哪一技术更好? 第一部分主要介绍在C#中异步(async/awai ...

  9. 二、多线程之Thread中run 和start 区别

    Thread使用run 和start 区别 结论:run()方法将作为当前调用线程本身的常规方法调用执行,并且不会发生多线程. System.out.println("开始测试多线程&quo ...

随机推荐

  1. 【Linux笔记】阿里云服务器被暴力破解

    一.关于暴力破解 前几天新购进了一台阿里云服务器,使用过程中时常会收到“主机被暴力破解”的警告,警告信息如下: 云盾用户您好!您的主机:... 正在被暴力破解,系统已自动启动破解保护.详情请登录htt ...

  2. FZU2127_养鸡场

    题目的意思为要你求出满足三边范围条件且周长为n的三角形的数目. 其实做法是直接枚举最短边,然后就可以知道第二条边的取值范围,同时根据给定的范围缩小范围. 同时根据第二条边的范围推出第三条边的范围,再次 ...

  3. UVA12583_Memory Overow

    题目是很简单的队列维护的题目. 每次加入之前判断该字母是否在队列以及队列的容量是否超过k即可. #include <iostream> #include <cstdio> #i ...

  4. 【loj6041】「雅礼集训 2017 Day7」事情的相似度 后缀自动机+STL-set+启发式合并+离线+扫描线+树状数组

    题目描述 给你一个长度为 $n$ 的01串,$m$ 次询问,每次询问给出 $l$ .$r$ ,求从 $[l,r]$ 中选出两个不同的前缀的最长公共后缀长度的最大值. $n,m\le 10^5$ 题解 ...

  5. Liunx 和 Win中的软链接详解

    用过Linux的朋友都知道linux中有软链接的概念,可以通过ln命令创建到目录或文件的软链接,软链接的好处就是可以让一个目录或文件有多个入口但保持单一物理位置,方便应用和管理.    1.命令格式: ...

  6. 【DP】CF859C Pie Rules

    https://www.luogu.org/problemnew/show/CF859C Description 有一个长度为\(n\)的序列,Alice和Bob在玩游戏.Bob先手掌握决策权. 他们 ...

  7. 【目标检测】Faster RCNN算法详解

    Ren, Shaoqing, et al. “Faster R-CNN: Towards real-time object detection with region proposal network ...

  8. UESTC--1300

    原题链接:http://acm.uestc.edu.cn/problem.php?pid=1300 分析:dp,最长公共上升子列.对于两个序列num1[maxn],num2[maxn]: 如果num1 ...

  9. ECONNRESET和WSAECONNRESET怎么产生的以及如何避免

    ECONNRESET是linux环境网络编程产生的错误,错误码为104, WSAECONNRESET是windows环境网络编程产生的错误,错误码为10054 两者产生的原因都一样,分以下几种情况: ...

  10. 使用uiautomator时遇到问题的处理方法

    本帖持续更新中… 一.使用adb devices无法连接到模拟器 这种情况可能是因为服务挂了之类的原因,重启一下服务 adb kill-server //关闭adb服务 adb start-serve ...