Erlang 函数式编程之模式匹配(下)

模式匹配是 Erlang 的根基,它被用于从数据结构中提取值,控制函数内部的流程,在并行程序中给进程发消息,选择处理那些消息。

上篇只是简单描述了一点 Erlang 语言中模式匹配的基本用法——从数据结构中提取值,我们将在本篇深入模式匹配的一些根植于 Erlang 编程语言内部的用法。

case 表达式

case Expression of
      Pattern1 [when Cuard1] -> Seq1;
      Pattern2 [when Cuard2] -> Seq2;
...
end

case 的执行过程:首先执行 Experssion 表达式,假设其值为 Value,然后将 Value轮流与 Pattern1(可能会带有可选的关卡 Guard1)、Pattern2 等进行模式匹配,知道成功。一旦成功匹配,对应的表达式 Seq 就会执行,而 Seq 表达式的执行结果就是这个 case 表达式的值,若无 Pattern 匹配成功,将会发生异常错误。 此外 Erlang 进行异常处理的 try...catch 表达式基本相当于 case 表达式的加强版,在 case 之后加上 catch 和 after 就相当于 try...catch 表达式。

函数

% func.erl
-module(func).
-export([area/1]).

area({rectangle, Width, Height}) ->
    Width * Height;
area({square, Side}) ->
    Side * Side;
area({triangle, Bottom, Hight}) ->
    1/2 * Bottom * Hight;
area({circle, Radius}) ->
    3.1415 * Radius * Radius.

模块声明和导出声明在最前面,不是要讨论的内容所以这里略过。

如上面代码所示,在文件中声明了一个函数 area 用来求一些平面几何体的面积,他有以分号分隔的四个子句,最后的子句以句号加空格结束。每个子句都有一个头部和一个主体,两者用箭头(->)分隔。头部包含一个函数名,后接一个或多个模式,主体包含一个表达式。这些子句会在头部里的模式与调用函数的参数成功匹配时执行,而这些子句会根据他们在函数定义里出现的顺序进行匹配。

下面我将实际操作一下,首先新建 func.erl 文件如下:


func

写好代码,并保存之后,执行 $ erl 进入 Erlang 的 Shell 环境,执行 > c(func). 编译 Erlang 代码,如下图所示:

Result

我们看到,第 2、3、4、5 条命令,分别调用了 area/1 函数,并成功执行,但第 6、7 条却调用失败,并抛出了不匹配的异常错误,这说明函数调用也是一种模式匹配。

简单并发

客户端-服务器架构是 Erlang 的中心,这里的所谓的 CS 架构和我们平常理解的 CS 架构不太一样,在这里客户端和服务器这两个词是指这两种进程所扮演的角色:客户端总是通过向服务器发送一个请求来发起计算,而服务器计算后生成回复,然后发送一个响应给客户端。

% area_server
-module(area_server).
-export([init/0, area/2, loop/0]).

init() -> spawn(area_server, loop, []).

area (Pid, What) -> rpc(Pid, What).

% 封装了向服务器发送请求和等待响应的代码
rpc(Pid, Request) ->
    Pid ! {self(), Request},
    receive
        {Pid, Response} ->
            Response
    end.

loop() ->
    receive
        {From, {rectangle, Width, Height}} ->
            From ! {self(), Width * Height},
            loop();
        {From, {square, Side}} ->
            From ! {self(), Side * Side},
            loop();
        {From, {triangle, Bottom, Hight}} ->
            From ! {self(), 1/2 * Bottom * Hight},
            loop();
        {From, {circle, Radius}} ->
            From ! {self(), 3.1415 * Radius * Radius},
            loop()
    end.
基本的并发函数
  • Pid = spawn(Mod, Fun, Args)

创建一个新的并发进程来执行 apply(Mod, Fun, Args),这个新进程和调用进程(父进程)并列运行。spawn 返回一个 Pid(进程标识符)。

  • Pid ! Message

向进程标识符为 Pid 的进程发送消息 Message,消息的发送是异步的,发送方并不等待,而是会继续之前的工作。

  • receive...end

接收发送给某个进程的消息。
当某个消息到达该进程之后,系统会尝试将它与 receive 表达式中的各种模式分别匹配,若没有匹配的模式,消息就会被保存起来供以后处理,进程则开始等待下一条消息。

在这里将不会介绍并发编程的细节,因为主题是模式匹配,以后会详细解释。

那么运行一下上面的代码:

RUN
最后编辑于
?著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,029评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,238评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事?!?“怎么了?”我有些...
    开封第一讲书人阅读 159,576评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,214评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,324评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,392评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,416评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,196评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,631评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,919评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,090评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,767评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,410评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,090评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,328评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,952评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,979评论 2 351

推荐阅读更多精彩内容

  • 世界是并行的,Erlang程序反应了我们思考和交流的方式,人作为个体通过发送消息进行交流,如果有人死亡,其他人会注...
    abel_cao阅读 2,757评论 1 4
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,644评论 18 139
  • //Clojure入门教程: Clojure – Functional Programming for the J...
    葡萄喃喃呓语阅读 3,642评论 0 7
  • “=” 操作符 上篇文章说到 Erlang 的变量是一次性赋值变量,可以使用 “=” 操作符给变量绑定一个值(这个...
    Helperhaps阅读 3,016评论 0 3
  • 我是一个不知不扣的懒家伙,上学的时候就总上课睡觉,所幸学习还凑合,开始上班也总是分秒必争,不到最后几分钟不到岗,现...
    别哭有我阅读 221评论 0 0