http://blog.csdn.net/karl_max/article/details/3977860

1. 并发原语:

(1) Pid = spawn(Fun) %% 创建一个新的并发进程,用于对Fun求值。

(2) Pid ! Message %% !是发送操作符,消息发送是异步的,返回结果是消息本身,所以Pid1!Pid2!...!M可以向多个进程发送消息M。

(3) receive ... end %% 接收一个发送给当前进程的消息,是同步的。语法:

receive

Pattern1 [when Guard1] -> Expressions1;

Pattern2 [when Guard1] -> Expressions2;

...

end

如果没有匹配任何模式也不会抛出异常,消息会留给后续过程来处理,然后等待下一个消息。

2. area_server_final.erl:一个计算面积的程序的客户机和服务器, 其中用start()封装了发启一个服务器进程,用area封装了发起一个远程调用(rpc),用loop实现的服务器循环,以及在发送和接收消息时判断进程ID等,这些方法都很有用。

-module(area_server_final).

-export([start/0, area/2]).

start() -> spawn(fun loop/0).

area(Pid, What) ->

rpc(Pid, What).

rpc(Pid, Request) ->

Pid ! {self(), Request},

receive

{Pid, Response} -> Response;

end.

loop() ->

receive

{From, {rectangle, W, H}} ->

From ! {self(), W*H},

loop();

{From, {circle, R}} ->

From ! {self(), 3.14159*R*R};

loop();

{From, Other} ->

From ! {self(), {error, Other}},

loop()

end.

在erlang shell中测试之

1> Pid = area_server_final:start().

2> area_server_final:area(Pid, {rectangle, 10, 8}).

3> area_server_final:area(Pid, {circle, 4}).

3. erlang:system_info(process_limit).可以获取虚拟机允许的进程上限(一般为32767),可以通过在启动erl时+P N来修改这个值,如:erl +P 500000

4. @spec statistics(runtime) -> {总CPU运行时间, 从上次调用以来的CPU运行时间}

@spec statistics(wall_clock) -> {总的真实消耗时间,从上调用以来的真实消耗时间}

真实时间包括运行过程中读写硬盘等的时间。

注意:返回的时间*1000后是微秒

5. receive的等待超时语法:

receive

Pattern1 [when Guard1] -> Expressions1;

Pattern2 [when Guard1] -> Expressions2;

...

after Time -> %% Time是毫秒

Expressions

end

6. 自定义的sleep,只有超时的receive

sleep(T) ->

receive

after T ->

true

end.

7. 超时为0的receive的妙用(还能想出别的用法吗?)

(1)清空进程邮箱里的所有消息:

flush_buffer() ->

receive

_Any -> flush_buffer()

after 0 ->

true

end.

(2)优先接收alarm消息,如果邮箱里有大量的消息,这一方法效率很低(有其它方法吗?)

priority_receive() ->

receive

{alarm, X} ->

{alarm, X}

after 0 ->

receive

Any -> Any

end

end.

8. 如果超时值是原子infinity,则永远不会超时。(这样有什么用呢?不太明白书上所说的情况。)

9. 用receive超时实现一个计时器:

-module(stimer).

-export([start/2, cancel/1]).

start(Time, Fun) -> spawn(fun() -> timer(Time, Fun) end).

cancel(Pid) -> Pid ! cancel.

timer(Time, Fun) ->

receive

cancel -> void

after Time -> Fun()

end.

10. receive的工作机制(选择性接收)

(1)启动一个计时器(在有after的情况下)。

(2)从进程邮箱中取出第一个消息,进行模式匹配,如果匹配成功,则从邮箱中删除之。

(3)如果没有匹配成功,则把取出的消息放入“保存队列”,然后继续取消息。重复这一步,直到匹配成功,或邮箱为空。

(4)如果没有匹配成功且此时邮箱为空,则挂起进程,等待新消息进入。注意,当有新消息进入时,只对新消息匹配。

(5)如果有一个消息匹配成功,则把保存队列中的消息按时间顺序放入到进程邮箱中。这时,将计时器清空。

(6)如果在等待消息时,计时器触发,则退出等待,进入after表达式,并将保存队列按时间顺序放回到进程邮箱中。

11. 注册进程:

register(AnAtom, Pid). %% 如果这个原子已经被注册了,这个调用会失败

unregister(AnAtom). %% 如果一个进程已经死亡,它会被自动取消注册

whereis(AnAtom) -> Pid | undefined

registered() -> [AnAtom:atom()] %% 返回系统中所有注册进程的列表。

进程注册的原子可以像进程Pid一样使用。

12. 并发程序模板:

-module(ctemplate).

-compile(export_all).

start() -> spawn(fun() -> loop([])) end).

rpc(Pid, Request) ->

Pid ! {self(), Request},

receive

{Pid, Response} -> Response

end.

loop(X) ->

receive

Any -> io:format("Received:~p~n", [Any]),

loop(X)

end.

13. 尾递归技术:

尾递归在编译时可以优化为一个跳转指令,所以可以无限循环下去。尾递归要确保函数F在调用自身之后,不再调用任何东西,也不要在列表或元组的构造器中使用F。

14. 想要代码在运行时能够动态更新,要用下面的spawn:

spawn(Mod, FuncName, Args).

其中Args是形如[A1, A2, ..., AN]的参数列表。

显式指明模块,函数和参数列表的形式被称为MFA。

《Erlang程序设计》学习笔记-第2章 并发编程的更多相关文章

  1. 《erlang程序设计》学习笔记-第3章 分布式编程

    http://blog.csdn.net/karl_max/article/details/3985382 1. erlang分布式编程的基本模型 (1) 分布式erlang:这种模型可以让我们在一个 ...

  2. JavaScript高级程序设计学习笔记第三章--基本概念

    一.标识符: 1.区分大小写 2.命名规则: 第一个字符必须是一个字母.下划线(_)或一个美元符号($) 其他字符可以是字母.下划线.美元符号或数字 标识符中的字母也可以包含扩展的 ASCII 或 U ...

  3. JavaScript高级程序设计学习笔记第十三章--事件

    事件冒泡: IE 的事件流,事件开始时由最具体的元素(文档中嵌套层次最深的那个节点)接收,然后逐级向上传播到较为不具体的节点(文档).例如: <!DOCTYPE html> <htm ...

  4. JavaScript高级程序设计学习笔记第十一章--DOM扩展

    1.对 DOM 的两个主要的扩展是 Selectors API(选择符 API)和 HTML5 2.Selectors API Level 1 的核心是两个方法: querySelector()和 q ...

  5. JavaScript高级程序设计学习笔记第六章--面向对象程序设计

    1.ECMAScript没有类的概念,ECMA-262 把对象定义为:“无序属性的集合,其属性可以包含基本值.对象或者函数.”,有点类似于散列表 2.ECMAScript 中有两种属性:数据属性和访问 ...

  6. JavaScript高级程序设计学习笔记第五章--引用类型(函数部分)

    四.Function类型: 1.函数定义的方法: 函数声明:function sum (num1, num2) {return num1 + num2;} 函数表达式:var sum = functi ...

  7. JavaScript高级程序设计学习笔记第五章--引用类型

    一.object类型 1.创建object类型的两种方式: 第一种,使用构造函数 var person = new Object();或者是var person={};/与new Object()等价 ...

  8. JavaScript高级程序设计学习笔记第四章--变量、作用域和内存问题

    1.变量可能包含两种不同数据类型的值:基本类型值和引用类型值. 基本类型值指的是简单的数据段,而引用类型值指那些可能由多个值构成的对象. 2.变量复制 如果从一个变量向另一个变量复制基本类型的值,会在 ...

  9. Go语言学习笔记(4)——并发编程

    Golang在语言级别支持了协程,由runtime进行管理. 在Golang中并发执行某个函数非常简单: func Add(x, y int) { fmt.Println(x + y) } func ...

随机推荐

  1. 如何让Apache不显示服务器信息

    如何让Apache不显示服务器信息 Apache的默认配置是会显示服务器信息的,比如访问一个服务器上不存在的页面,Apache会返回"Not Found"的错误,这个错误页面的最下 ...

  2. 2lession-文件访问

    今天继续学习python,因为是根据网上的教程,里面用到了一些例子,包含有后面的知识点.但是,因为自己稍微有点c.java等语言基础,所以并没有严格按照教程来学习,反而是遇到知识点就记录下来. 代码如 ...

  3. java list 容器的ConcurrentModificationException

    java中的很多容器在遍历的同时进行修改里面的元素都会ConcurrentModificationException,包括多线程情况和单线程的情况.多线程的情况就用说了,单线程出现这个异常一般是遍历( ...

  4. Stack overflow 编译能通过,运行时出现Stack overflow

    Stack overflow 编译能通过,运行时出现Stack overflow 大家都知道,Windows程序的内存机制大概是这样的,全局变量(局部的静态变量本质也属于此范围)存储于堆内存,该段内存 ...

  5. c编程:僵尸吃大脑

    第一行输入一个正整数n 以下每一行输入僵尸已经吃了的大脑数量a,和需要生存必需要吃的大脑数量b.总共n行. 例子输入 3 4 5 3 3 4 3 例子输出 NO BRAINS MMM BRAINS M ...

  6. Qt5 UI信号、槽自动连接的控件重名大坑(UI生成的槽函数存在一个隐患,即控件重名。对很复杂的控件,不要在 designer 里做提升,而是等到程序启动后,再动态创建,可以避免很多问题)

    对Qt5稍有熟悉的童鞋都知道信号.槽的自动连接机制.该机制使得qt designer 设计的UI中包含的控件,可以不通过显式connect,直接和cpp中的相应槽相关联.该机制的详细文章见 http: ...

  7. docker安装及问题处理

    1.在Ubuntu的命令行中输入 sudo apt-get install docker.io 2.如果切换到了root用户下 apt-get install docker.io 3.对于新安装的Ub ...

  8. $_SERVER['DOCUMENT_ROOT']

    $_SERVER['DOCUMENT_ROOT'] 一.总结 $_SERVER 是一个包含了诸如头信息(header).路径(path).以及脚本位置(script locations)等等信息的数组 ...

  9. go初探 - 生成随机整数

    func RandInt64(min, max int64) int64 { if min >= max || min == 0 || max == 0 { return max } rand. ...

  10. POJ 1458 Common Subsequence (zoj 1733 ) LCS

    POJ:http://poj.org/problem?id=1458 ZOJ:http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemId=73 ...