Erlang读音/ˈɜːrlæŋ/
。第一次见到的时候总感受怎么读都读不对,后来在维基上看到Erlang标注了音标,才能准确的读出来,并且也没那么怪异。由于工做才有机会接触这门语言,也所以只有三天的时间能够看《Erlang程序设计》这本书。学习这门语言的时候带着一个工做目标:把一个Erlang日志收集分析统计的代码转换成Python的。而Erlang的风格是尽可能不写注释,尽可能在写函数名和变量名的时候表达清楚代码的含义。这样一来学习Erlang就成了必要的,很庆幸,领导给了三天时间学习,三天时间基本也足够了。除了这一片基础语法的入门篇以外,后续还有一篇或者两篇并发编程和分布式编程的,毕竟这个才是Erlang擅长的领域。话很少说,show me your article
html
<!--more-->python
依赖工具:程序员
下载连接:shell
IDEA配置Erlang插件:编程
全部的变量都必须以大写字母开头,变量只可一次赋值,赋值以后不可在变。 f()函数释放shell绑定变量。segmentfault
case
、if
、try..catch
、receive
表达式中的模式匹配时,用分号“;”分界恒等测试符号 =:=以及不等测试符号 =/=数据结构
当程序中某处的语法要求只能使用单个表达式可是逻辑上又须要在此使用多个表达式时,就可使用begin...end快表达式并发
begin Expr1, ... ExprN end
1> Point = {point, 20, 43}. {point,20,43} 2> {point, x, y} = Point. ** exception error: no match of right hand side value {point,20,43} 3> {point, X, Y} = Point. {point,20,43} 4> X. 20 5> Y. 43 6> Person = {person, {name, {first, joe}, {last, armstrong}}, {footsize, 42}}. {person,{name,{first,joe},{last,armstrong}},{footsize,42}} 7> {_, {_, {_, Who}, {_, _}}, {_, Size}} = Person. {person,{name,{first,joe},{last,armstrong}},{footsize,42}} 8> Who. joe 9> Size. 42
竖线符号|app
列表操做演示代码分布式
1> L = [1+7, hello, 2-2, {cost, apple, 30-20}, 3]. [8,hello,0,{cost,apple,10},3] 2> L1 = [123, {oranges, 4} | L]. [123,{oranges,4},8,hello,0,{cost,apple,10},3] 3> [E1 | L2] = L1. [123,{oranges,4},8,hello,0,{cost,apple,10},3] 4> E1. 123 5> L2. [{oranges,4},8,hello,0,{cost,apple,10},3] 6> [E2, E3 | L3] = L2. [{oranges,4},8,hello,0,{cost,apple,10},3] 7> E3. 8
列表表达式
形式:[F(X) || X <- L]
1> L = [1, 2, 3, 4, 5]. [1,2,3,4,5] 2> [2 * X || X <- L]. [2,4,6,8,10] 3> [X || {a, X} <- [{a, 1}, {b, 2}, {c, 3}, {a, 4}, hello, "wow"]]. [1,4]
Erlang的字符串是一个整数列表。整数列表的内容由每个字符对应的ascii码构成
1> I = $s. 115 2> [I-32, $u, $r, $p, $r, $i, $s, $e]. "Surprise" 3> $r. 114 4> [I-32, $u, $r, $p, 114, $i, $s, $e]. "Surprise"
映射组是一个由多个Key-Vaule结构组成的符合数据类型,相似于Python的字典。具体使用以下
1> M1 = #{"name" => "alicdn", "percentage" => 80}. #{"name" => "alicdn","percentage" => 80} 2> maps:get("name", M1). "alicdn" 3> M2 = maps:update("percentage", 50, M1). #{"name" => "alicdn","percentage" => 50} 4> map_size(M1). 2 5> #{"name" := X, "percentage" := Y} = M2. #{"name" => "alicdn","percentage" => 50} 6> X. "alicdn" 7> Y. 50
构造映射组和模式匹配时的符号不同,=>
和:=
的区别。常见的put方法参见erlang maps库的使用。
-import(lists, [map/2, sum/1]).
导出模块中的函数:
-export([start/0, area/2]).
-compile(export_all).
,避免在开发阶段常常会向export中添加函数或者删除函数-module(learn_test). -author("ChenLiang"). %% API -export([area/1]). area({rectangle, Width, Height}) -> Width * Height; area({circle, R}) -> 3.14159 * R * R; area({square, X}) -> X * X.
编译模块,调用函数
1> c(learn_test). {ok,learn_test} 2> learn_test:area({circle, 2.0}). 12.56636 3>
同名同目(参数数量,arity)的才是同一个函数。所以函数名相同,目不相同的函数是彻底不一样的两个函数。同名不一样目的函数一般做为辅助函数。
示例代码:计算列表元素的和
-module(learn_test). -author("ChenLiang"). %% API -export([sum/1]). sum(L) -> sum(L, 0). sum([], N) -> N; sum([H|T], N) -> sum(T, H + N).
erlang中的匿名函数就是fun。fun也能够有若干个不一样的字句。
1> Z = fun(X) -> 2*X end. #Fun<erl_eval.6.50752066> 2> Double = Z. #Fun<erl_eval.6.50752066> 3> Double(4). 8 4> TempConvert = fun({c, C}) -> {f, 32 + C * 9 / 5}; 5> ({f, F}) -> {c, (F - 32) * 5 / 9} 6> end. #Fun<erl_eval.6.50752066> 7> TempConvert({c, 100}). {f,212.0} 8> TempConvert({f, 212}). {c,100.0}
返回fun或者接受fun做为参数的函数都称为高阶函数。
以fun为参数的函数
常见的是lists模块中的map(Fun, List1) -> List2,filter(Pred, List1) -> List2函数。
lists模块的具体使用参见:https://www.erlang.org/doc/ma...
1> Even = fun(X) -> X rem 2 =:= 0 end. #Fun<erl_eval.6.50752066> 2> lists:map(Even, [1, 2, 3, 4, 5, 6]). [false,true,false,true,false,true] 3> lists:filter(Even, [1, 2, 3, 4, 5, 6]). [2,4,6]
返回fun的函数
通常在返回的函数内部封装了一些变量和逻辑。一般状况下不写返回fun的函数。
1> Mult = fun(Times) -> (fun(X) -> X * Times end ) end. #Fun<erl_eval.6.50752066> 2> Triple = Mult(3). #Fun<erl_eval.6.50752066> 3> Triple(4). 12
强化模式匹配的功能,给模式匹配增长一些变量测试和比较的能力
max(X, Y) when X > Y -> X; max(_, Y) -> Y.
记录是Erlang中基于元组的key-value数据定义,使用示例以下:
-module(learn_test). -author("ChenLiang"). %% API -export([record_test1/0, record_test2/0]). -record(person, {name, age=18, hobby=["erlang"]}). %% record定义能够存放于hrl和erl中 record_test1() -> Person = #person{name="hahaha"}, %% 为record中字段赋值 Person#person.hobby. %% 经过.操做符访问record中字段 record_test2() -> Person = #person{}, #person{name = Name} = Person, %% 经过模式匹配获取record字段 Name. %% 输出undefined
某些文件的扩展名为 .hrl
。这些.hrl
是在 .erl
文件中会用到的头文件,使用方法以下:
-include("File_Name").
例如:
-include("mess_interface.hrl").
.hrl 文件中能够包含任何合法的 Erlang 代码,可是一般里面只包含一些记录和宏的定义。
case语句语法
case Experssion of Pattern1 [when Guard1] -> Expr_seq1; Pattern2 [when Guard2] -> Expr_seq2; ... end
将Expression的结果和各个Pattern逐个匹配,匹配成功,则计算表达式序列的值,并返回。所有匹配不到,则直接报错。
case语句使用示例:
-module(learn_test). -author("ChenLiang"). %% API -export([filter/2]). filter(P, [H|T]) -> case P(H) of true -> [H|filter(P, T)]; false -> filter(P, T) end ; filter(_, []) -> [].
在erl shell中运行结果以下:
1> c(learn_test). {ok,learn_test} 2> learn_test:filter(fun(X) -> X rem 2 =:= 0 end, [1, 2, 3, 4, 5]). [2,4]
if语句使用示例
-module(learn_test). -author("ChenLiang"). %% API -export([bigger/2]). bigger(X, Y) -> if X > Y -> X; X < Y -> Y; true -> -1 end.
若是没有匹配的断言,则会抛出异常。所以最后一个断言一般是true断言。
Erlang中一切都是表达式,都有返回值,所以异常捕获语句也有返回值。
捕获全部的异常_:_
-module(learn_test). -author("ChenLiang"). %% API -export([catch_exc1/0,catch_exc2/0]). exception() -> exit({system, "123123"}). catch_exc1() -> try exception() catch _:_ -> 111 end. catch_exc2() -> try exception() catch _ -> 222 end.
erl shell输出结果
1> learn_test:catch_exc1(). 111 2> learn_test:catch_exc2(). ** exception exit: {system,"123123"} in function learn_test:exception/0 (learn_test.erl, line 17) in call from learn_test:catch_exc2/0 (learn_test.erl, line 28)
多种错误的检测能够 使用try catch风格。
参考stackoverflow-How do I elegantly check many conditions in Erlang?
记得帮我点赞哦!
精心整理了计算机各个方向的从入门、进阶、实战的视频课程和电子书,按照目录合理分类,总能找到你须要的学习资料,还在等什么?快去关注下载吧!!!
念念不忘,必有回响,小伙伴们帮我点个赞吧,很是感谢。
我是职场亮哥,YY高级软件工程师、四年工做经验,拒绝咸鱼争当龙头的斜杠程序员。听我说,进步多,程序人生一把梭
若是有幸能帮到你,请帮我点个【赞】,给个关注,若是能顺带评论给个鼓励,将不胜感激。
职场亮哥文章列表:更多文章
本人全部文章、回答都与版权保护平台有合做,著做权归职场亮哥全部,未经受权,转载必究!