元组(tuple)类似c/c++的结构体,但是还不够灵活。例如和业务相关的一个由5个元素组成的元组,由于业务变更,需要增加一个字段,于是你不得不跑遍所有的代码并修改所有涉及这些元组的地方。如果修改的地方很多,非常容易犯错误。于是我们引入记录(record)概念。
以下以一个简单的例子阐述如何使用记录:
-module(record).
-export([test/0]).
-record( person, {name, addr="Shanghai"} ). %% 声明一个person record
test() ->
test( #person{name="Zhangsan",addr="Beijing"} ). %% 创建一个person record,并传给test/1
test( R = #person{name=_Name, addr=_Addr} ) -> %% record的模式匹配
io:format("{person,~p,~p}~n", [R#person.name,R#person.addr ]), %% 获取record某个字段
R#person{addr=R#person.addr ++ "-China"}. %% 更新record字段(实际上创建新的record)
1.record声明
record语法可以让你使用记录,它们本质上就是标记元组(第一元素是原子的元组)
-record(customer, {name="<anonymous>",address, phone}).
相当于{customer, name, address, phone}的元组。但注意以后使用record,我们是不需要关心record字段顺序的。
在定义时,record字段可以使用缺省的值
2.创建record
#customer{}
#customer{phone="12345"}
#customer{phone="12345", name="John", address="L.A."}
再一次强调,record不用关心字段顺序
3. 获取record的字段
假定你已经用R绑定了上述3个例子上的任何一个。现在你可以用#和点分法来访问record的各个字段:
R#customer.name
R#customer.address
R#customer.phone
4.record字段的模式匹配
print_contact( #customer{name=Name, address=_Addr, phone=Phone} )
when Phone =/= undefined ->
io:format("Contact: ~p at ~p.~n",[Name, Phone]).
说明:不用的变量前加下划线可以屏蔽编译时的变量unused告警
5.更新记录
根据一个已有的记录创建新的记录
A = R,
B = R#customer{name="Jane"}.
6.使用'_'匹配
Customer = #{customer,name="Alan",_='_'}.
_匹配剩余的元组里的其他元素,所以结果是#customer{name="Alan", address='_', phone='_' }。