匹配和作用域
模式匹配
在某些场景下,可能需要找到最高温度或最低温度。所以查找温度值列表中最大值或最小值是非常有用的
在扩展程序实现该功能之前,先看一下寻找列表中的最大值的方法:
-module(match). -export([list_max/1]). list_max([Head|Rest]) -> list_max(Rest, Head). list_max([], Res) -> Res; list_max([Head|Rest], Result_so_far) when Head > Result_so_far -> list_max(Rest, Head); list_max([Head|Rest], Result_so_far) -> list_max(Rest, Result_so_far).
8> match:list_max([1,2,3,4,5,7,4,3,2,1]).
7
首先注意有两个函数的名称是完全相同的。但是,由于它们接受不同数目的参数,所以在 Erlang 中它们被当作两个完全不相同的函数
在需要使用它们的时候,使用名称/参数数量的方式就可以了,这里名称就是函数的名称,参数数量是指函数的参数的个数 这个例子中为 list_max/1 与 list_max/2
遍历列表的中元素过程中 “携带” 了一个值(最大值),即 Result_so_far。list_max/1 函数:
- 把列表中的第一个元素当作最大值元素
- 使用剩余的元素作参数调用函数 list_max/2
在上面的例子中为 list_max([2,3,4,5,6,7,4,3,2,1],1)
如果使用空列表或者非列表类型的数据作为实参调用 list_max/1,则会产生一个错误
注意:Erlang 的哲学是不要在错误产生的地方处理错误,而应该在专门处理错误的地方来处理错误
Guard 谓词
在 list_max/2 中,当 Head > Result_so_far 时,则使用 Head 代替 Result_so_far 并继续调用函数
- when 用在函数的 -> 前时是一个特别的的单词,它表示 只有测试条件为真时才会用到函数的这一部分 。这种类型的测试被称这为 guard
- 如果 guard 为假 ,则 跳过此部分 而尝试使用函数的后面一部分
这个例子中,如果 Head 不大于 Result_so_far 则必小于或等于。所以在函数的下一部分中不需要 guard 测试
可以用在 guard 中的操作符还包括:
- < 小于
- > 大于
- == 等于
- >= 大于或等于
- =< 小于或等于
- /= 不等于
要将上面找最大值的程序修改为查找最小值元素非常容易,只需要将 > 变成 < 就可以了 但是,最好将函数名同时也修改为 list_min
词法作用域
前面提到过,每个变量在其作用域内只能被赋值一次 但上面的例子中也可以看到,Result_so_far 却被赋值多次,为什么可以呢?
这是因为,每次 调用 一次 list_max/2 函数都会 创建 一个 新的作用域 。在每个不同的作用域中,Result_so_far 都被当作完全不同的变量。下面再看一些其他的例子:
1> M = 5. 5 2> M = 6. ** exception error: no match of right hand side value 6 3> M = M + 1. ** exception error: no match of right hand side value 6 4> N = M + 1. 6
使用 匹配操作符 = 创建 一个 变量 并给这个 变量赋值
M = 5 创建了一个变量 M,并给其赋值为 5,如果在相同的作用域中,再写 M = 6, 则会导致错误 同样的道理 M = M + 1 也会报错,但 N = M + 1 就不会报错,因为再次创建了一个新的变量
除了创建新变量外,匹配操作符另一个用处就是将 Erlang 项分开 :
5> {X, Y} = {paris, {f, 28}}.
{paris,{f,28}}
6> X.
paris
7> Y.
{f,28}
如果同样用 X 和 Y 再使用一次,则会产生一个错误:
8> {X, Y} = {london, {f, 36}}.
** exception error: no match of right hand side value {london,{f,36}}
变量用来 提高程序的可读性 。例如,在 list_max/2 函数中,可以这样写:
list_max([Head|Rest], Result_so_far) when Head > Result_so_far -> New_result_far = Head, list_max(Rest, New_result_far);