| 编程范型 | 多范式:函數式、并发 |
|---|---|
| 設計者 | 喬·阿姆斯特朗、Robert Virding、Mike Williams |
| 實作者 | 愛立信 |
| 发行时间 | 1986年,40年前(1986) |
| 当前版本 |
|
| 型態系統 | 動態、強 |
| 許可證 | Apache许可证2.0(从OTP 18.0开始) Erlang公共许可协议1.1(英语:Erlang Public License)(早期版本) |
| 文件扩展名 | .erl .hrl |
| 網站 | www |
| 主要實作產品 | |
| Erlang | |
| 受影响于 | |
| Prolog,Smalltalk,PLEX(英语:PLEX (programming language)),[2]LISP | |
| 影響語言 | |
| Akka,Clojure,Dart,Elixir,F♯,Opa(英语:Opa (programming language)),Oz, Reia,Rust,Scala | |
| |

Erlang(/ˈɜːrlæŋ/)是一種通用的并发函数式程序设计语言。Erlang也可以指Erlang/OTP的通稱,開源電信平台(OTP)是Erlang的常用執行環境及一系列標準元件。
Erlang 執行環境為專有以下要求的系統設計:
Erlang是運作於虛擬機的解释型语言,但是現在也包含有烏普薩拉大學高性能Erlang計劃(HiPE)[3]開發的原生程式碼編譯器,自R11B-4版本開始,Erlang也支持脚本方式执行。在編程範型上,Erlang屬於多重典範程式語言,涵蓋函數式、并行及分布式。循序執行的Erlang是一个及早求值,單次賦值和动态类型的函數式程式語言。
它由喬·阿姆斯特朗(Joe Armstrong)在瑞典電信設備製造商愛立信所轄的電腦科學研究室開發,目的是創造一種可以應付大規模并發活動的程序设计语言和執行環境。Erlang於1987年釋出正式版本,最早是愛立信擁有的私有軟體,經過十年的發展,於1998年發表開放源碼版本。
Erlang得名於丹麥數學家及統計學家Agner KrarupErlang,同時Erlang還可以表示EricssonLanguage。Erlang語言由瑞典愛立信電信公司的喬·阿姆斯特朗開始設計,開始於公元一九八零年代。最初是以Prolog程序设计语言為基礎,幾度改版之後,改成以Joe's Abstract Machine為基礎的獨立語言執行環境。雖然語言風格仍與Prolog相近,不過因Erlang語言設計的走向,Erlang成為具备函數語言特色的程序设计语言[4]。
1998年起,Erlang發布開放源碼版本,稱為開源電信平台。開源電信平台採用修改過的Mozilla公共許可證協議發放,同時爱立信仍然提供商業版本的技術支持。目前,Erlang最大的商業用户是爱立信,其他知名用户有北電網路、亚马逊以及T-Mobile等[5]。
Erlang程序結構以函數定義為主。函數是一組將輸入分別對應到輸出的規則,對應方式遵守數學函數的慣例。此外,Erlang語言由幾項構句要素所組成,包括文字(或稱原子)、數字、列表、值組、字元、字串、二進位資料、模組、與特定用途的關鍵字如fun ... end, if ... end, case ... of ... end, spawn, !, receive ... end等等。以下段落分別列示並舉例說明Erlang程式的基本構成部份,涵蓋資料格式、表達式格式與內建函數。
| 類型 | 意義與構詞規則 | 例子 |
|---|---|---|
| 原子 | 原子是基本資料單元,以一般文字構成。構詞規則有:
|
|
| 數字 | 數字是基本資料單元,可以是整數或實數。
|
|
| 列表 | 列表是與鏈接序列相同的資料結構。任一列表大致區分為頭部與尾部,頭部是列表的第一項,尾部是列表除第一項之外的其他部份。
|
|
| 值組 | 值組是將二個、三個或多個資料放在一起的資料結構。
|
|
| 字元 | Erlang將字元存為32位元的整數。
|
|
| 字串 | Erlang將字串視同一列整數列表。
|
|
| 二進位資料 | 以左邊 << 、右邊 >> 符號,包含由位元語法(页面存档备份,存于互联网档案馆)表示的資料。 |
|
| 函數識別項 | Erlang容許用文字表示函數識別項,使程式中可以對指定函數做函數呼叫,或者當做資料傳遞。函數識別項格式為:
|
用途見以下「函數式程式設計」小節。 |
| 程序代號 | Erlang容許以內建函數erlang:spawn/3、erlang:spawn/4、erlang:spawn/1、erlang:spawn/2等等,將指定函數啟動為一個程序。程序啟動之後,Erlang以左邊 < 、右邊 > ,包含一個數字和點號組成的編號,表示此程序代號。 | 見以下「平行式程式設計」小節。 |
| 模組 | Erlang容許將一些程式整理為一個模組。模組的設定,是在源碼檔案開頭書寫模組標記,格式為:-module(模組名稱).-export( [ 函數名稱/參數數目 , 函數名稱/參數數目 , ... ] ).-import( 模組名稱, [ 函數名稱/參數數目 , 函數名稱/參數數目 , ... ] ). 模組名稱和函數名稱都是原子。 -module(模組名稱) 定義模組的名字,要與檔名相同。 -export( ... ) 定義模組釋出的函數,模組內的任何函數必須要釋出才能讓外部透過模組呼叫該函數。 -import( ... ) 定義本模組要從其他模組匯入哪些函數,以便本模組自己使用。另外,為了方便程式的撰寫並測試,還容許 -compile(export_all) 定義本模組的所有函數全部對外釋出。 -compile(export_all). | (略) |
| 巨集 | 巨集是將一項資料以另一個文字做為代名。
-define ( 代名 , 資料 ).
? 代名
?MODULE |
|
| 類型 | 構詞規則 | 例子 | ||||||||||||||||||||||||||||
|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
| 變數 | 變數是一種提供與資料綁定、賦值的詞彙。Erlang的變數是單一賦值,一個變數只能賦值一次。
|
| ||||||||||||||||||||||||||||
| 樣式匹配 |
|
| ||||||||||||||||||||||||||||
| 函數 | 函數是由一或多項對應規則組成。每一項規則是將一部份匹配樣式的輸入映射到相對的輸出。
原子 ( 變數 , 變數 , ... ) -> 表達式 , 表達式 , ...在 -> 左邊是函數名稱及搭配的參數列,右邊為函數本體。
規則 ; 規則 ; ... ; 規則 .以分號分隔一或多項規則,並最後以句號結束。同一函數的每一規則必須以相同的原子開頭,並接受相同數量的參數列。 函數被呼叫時,會讓呼叫方依序對被呼叫方的每一條函數規則做樣式匹配,比對函數名稱、參數數目、參數樣式等等。首先完成匹配的函數規則會被執行,並且後面的函數規則會被忽略。 | 見以下「函數式程式設計」小節 | ||||||||||||||||||||||||||||
| 函數呼叫 | 格式為原子 ( 資料 , 資料 , ... )表示函數名稱及搭配的參數列。呼叫符合函數名稱及相同參數數目的函數。 函數呼叫時,所給予的參數可能是已賦值的變數。並且,如果參數是變數,必須是已賦值的變數。 | 見以下「函數式程式設計」小節 | ||||||||||||||||||||||||||||
| 真值比較 |
真值比較的結果,如果成功則傳回true原子,失敗則傳回false原子。 請記得,Erlang是以true和false表示布林資料類型。 | (略) | ||||||||||||||||||||||||||||
| 運算子 | Erlang提供常用的運算子方便基本運算。運算子是用在中序的表達式裏,包含 + - * / div(商) rem(餘) 等。位元算算有 bnot, band, bor, bxor, bsl(算術左移), bsr(算術右移) 等。用於列表有 ++(列表銜接) --(列表剔除) 等。各種運算式皆可用 ( ) 調整運算優先順序。 | (略) | ||||||||||||||||||||||||||||
| 防衛式 | 防衛式是接在when關鍵字之後的一組表達式,藉由防衛式的真偽值做程式控制處理。防衛式的原則如下方所述:
|
| ||||||||||||||||||||||||||||
| 受防衛式限制的函數 | 函數對應規則格式為: 原子 ( 變數 , 變數 , ... ) -> 表達式 , 表達式 , ... 若一條函數規則加上防衛式,此規則的處理範圍會多一些限制。受防衛式限制的函數對應規則格式為: 原子 ( 變數 , 變數 , ... ) when 防衛式 -> 表達式 , 表達式 , ... |
| ||||||||||||||||||||||||||||
| 行後註解 | 任何 % 符號開頭,往後到行尾的文字皆為註解文字。 | 'H.W.'. % Hello, World! | ||||||||||||||||||||||||||||
| λ演算式 | λ演算式是匿名函數,在Erlang以 fun ... end 關鍵字敘述。格式為: fun ( 變數 , 變數 , ... ) -> 表達式 , 表達式 , ... end 使用無參數的λ演算式,可以做出惰性求值的效果。 |
| ||||||||||||||||||||||||||||
| 因果式 | 使用 if ... end 關鍵字敘述條件判斷原則。格式為: if 防衛式 -> 表達式, 表達式, ... ; 防衛式 -> 表達式, 表達式, ... ; ...... 防衛式 -> 表達式, 表達式, ...end |
| ||||||||||||||||||||||||||||
| 案例式 | 使用 case ... of ... end 關鍵字,根據一個變數的案例,帶往相對的處理程序。格式為: case 表達式 of 樣式 -> 表達式, 表達式, ... ; 樣式 -> 表達式, 表達式, ... ; ...... 樣式 -> 表達式, 表達式, ...end
|
| ||||||||||||||||||||||||||||
| 試誤 | 使用 try ... catch ... end 關鍵字敘述試誤的情況與結果。格式為: try 表達式 of 樣式 -> 表達式, 表達式, ... ; 樣式 -> 表達式, 表達式, ... ; ...... 樣式 -> 表達式, 表達式, ...catch 樣式(例外) -> 表達式, 表達式, ... ; 樣式(例外) -> 表達式, 表達式, ... ; ...... 樣式(例外) -> 表達式, 表達式, ...after 表達式, 表達式, ...end
| (略) | ||||||||||||||||||||||||||||
| 接收訊息 | 每個Erlang程式執行時,都可以從自己程序的郵箱中取得由其他程序送到的訊息。可以使用 receive ... end 關鍵字接收訊息,格式為: receive 樣式 -> 表達式, 表達式, ... ; 樣式 -> 表達式, 表達式, ... ; ...... 樣式 -> 表達式, 表達式, ...end
|
| ||||||||||||||||||||||||||||
| 發送訊息 | Erlang容許向程序傳送訊息。使用 ! 關鍵字,格式為: 程序代號 ! 訊息
|
| ||||||||||||||||||||||||||||
| 列表解析 | 列表解析,是提供快速建立列表的語法。語法等同於集合建構式。格式為:
|
|
開源電信平台包括一個Erlang直譯器、一個Erlang編譯器、程序節點通訊協定、CORBA、一個分散式資料庫Mnesia、以及許多程式庫[6]。內建函數涵蓋了各種方面的功能,涵蓋了系統命令、資料存取、格式轉換、網路通訊、圖形介面、 ... 等。以下列表介紹幾項常用的Erlang內建函數。(參閱文件(页面存档备份,存于互联网档案馆)或索引 (页面存档备份,存于互联网档案馆))
| 模組:函數名稱 / 參數數目 | 用途 |
|---|---|
| c:cd / 1 | 切換到指定目錄位置。 > c:cd("D:\\code").D:/code/ok當指定目錄不正確時,則保持在原目錄位置。 |
| c:c / 1 | 編譯指定的程式碼,之後載入新編譯好的程式。 > c:c(test). % test.erl 必須存在於目錄位置{ok, test}> c:c(test1)../test1.erl:none: ...error |
| io:format / 2 | 按照指定的格式文字將資料印在標準輸出埠。 > io:format("~.8B, ~c, ~s, ~.2f~n", [32, $a, "hello", 3.1416]).40, a, hello, 3.14ok |
| lists:sublist / 3 | 由列表中擷取子列表。Erlang字串是整數列表,於是本函數視同擷取子字串。 > lists:sublist("Hello, World!", 2, 2). "el" |
這是輸出 Hello World 的一種方式:[7]
-module(hello).-export([hello_world/0]).hello_world()->io:fwrite("hello, world\n").
若要編譯這個程式,將它存為一個名為 hello.erl 的文字檔,然後從 Erlang終端 進行編譯。不要忘了在每個命令的最後加上一個句號(.)。例如:
Erlang (BEAM) emulator version 4.9.1 [source]Eshell V4.9.1 (abort with ^G)1> c(hello).{ok,hello}(在 Unix系統 上,你可以通過在命令列裡輸入 "erl" 來進入 Erlang終端。在 Windows系統 上,你需要打開一個 命令提示符 視窗,然後輸入 "werl"來進入 Erlang終端,或者在程式功能表中找到 Erlang 的圖示。)從 Erlang終端 上運行這個程式:
2> hello:hello_world().hello, worldok
Erlang支持函數式程式設計的一般特色,特色包括單次賦值、遞迴定義、λ演算與高階函數等等。Erlang函数大致寫法如下,以整數階乘模組为例:
-module(fact).-export([fac/1]).fac(N)when N > 1 -> N * fac(N-1);fac(1) -> 1.
以下是快速排序演算法的Erlang實作:
%% quicksort:qsort(List)%% Sort a list of items-module(quicksort).-export([qsort/1]).qsort([]) -> [];qsort([Pivot|Rest]) -> qsort([ X || X <- Rest, X =< Pivot]) ++ [Pivot] ++ qsort([ Y || Y <- Rest, Y > Pivot]).
以下是費氏數列求解函數:
-module(example).-export([fibo/1]).fibo(N)when N > 1 -> fibo(N-1) + fibo(N-2);fibo(1) -> 1;fibo(0) -> 0.
> c(example).{ok,example}> lists:map(fun(X)->example:fibo(X) end, lists:seq(1,10)).[1,1,2,3,5,8,13,21,34,55]函數式程式設計難免以遞迴計算,而消耗了大量遞迴堆疊空間。為了克服這個問題,一般使用累積參數與尾端遞迴等技巧節省遞迴數目:如以下例子。
-module(test).-export([fibo_accu/1]).fibo_accu(N) -> fibo(N, 0, 1).fibo(N, C1, C2)when N > 2 -> fibo(N-1, C2, C1+C2);fibo(0, _, _) -> 0;fibo(1, _, _) -> 1;fibo(_, C1, C2) -> C1+C2.
> c(example).{ok,test}> lists:map(fun(X)->test:fibo_accu(X) end, lists:seq(1,10)).[1,1,2,3,5,8,13,21,34,55]函數式程式設計容許使用高階函數求解。以下例子說明Erlang實做複合函數。 ( f o g ,唸作 fafter g 。)
'After'(F, G) -> fun(X) -> erlang:apply(F, [erlang:apply(G, [X])]) end.
> (example:'After'(fun test:show/1, fun test:parse/1))(3.1416).Real number 3.141600 is met.ok
Erlang最主要的特色是平行導向程式設計,強調多程序平行運作,並且以訊息對彼此溝通[8]。Erlang提供了spawn函數和! 、receive ...end 等關鍵字,可以描述在Erlang/開源電信平台中的如何啟動一些程序、並且如何讓程序傳遞訊息。此外,平行導向程式設計的精神還強調程序的容錯處理,藉由程序發生錯誤時的訊息傳遞,使其他程序可以得知錯誤的發生,使方便於後續處理。以下分別介紹平行導向程式設計的一般程式撰寫方式,以及錯誤處理的使用方式。
基本的平行程式示範如下:
% create process and call the function web:start_server(Port, MaxConnections)ServerProcess =spawn(web, start_server, [Port, MaxConnections]),
% send the {pause, 10} message (a tuple with an atom "pause" and a number "10")% to ServerProcess (asynchronously)ServerProcess ! {pause, 10},% receive messages sent to this processreceive a_message -> do_something; {data, DataContent} -> handle(DataContent); {hello, Text} -> io:format("Got hello message: ~s", [Text]); {goodbye, Text} -> io:format("Got goodbye message: ~s", [Text])end.收到a_message 結果就是do_something ;收到 {data,DataContent} 結果會呼叫 handle(DataContent) ;收到 {hello,Text} 結果教是印出 "Got hello message: ..." ,收到 {goodbye,Text} 結果是印出"Got goodbye message: ..." 。以下程式,示範產生一組環狀傳遞訊息的程序。
ring_proc(Funs) -> Ns = lists:seq(1, length(Funs)), [P|Pids] = [spawn(?MODULE, lists:nth(Nth,Funs),[]) || Nth <- Ns ], [ Pid ! ToPid || {Pid, ToPid} <- lists:zip([P|Pids], Pids++[P]) ]. func() ->receiveToPid -> func_msg_(ToPid)end.func_msg_(ToPid) ->receivestop -> io:format("Stop process ~w~n", [self()]), ToPid ! stop;Message -> io:format("~w: transmit message to ~w~n", [self(), ToPid]), ToPid ! Message, func_msg_(ToPid)end.接收stop訊息,就對下一個程序送stop訊息;接收到其他任何訊息,就對下一個程序送同樣的訊息。如果傳送任何其他訊息,就會讓所有的程序不斷對下一個程序傳遞訊息。而以下是測試傳送stop訊息的執行結果。
> [P|_] =example:ring_proc([func,func,func]).[<0.233.0>,<0.234.0>,<0.232.0>]>P ! stop.Stop process <0.233.0>stopStop process <0.234.0>> Stop process <0.232.0>>
Erlang容錯處理機制,由二個步驟實現:一是將二個程序連接起來,二者之間存在一道通訊管道,可提供錯誤訊息的傳遞 ── 在此使用link/1函數;二是將程序回報錯誤的機制打開 ── 在此使用process_flag/2函數。
-module(example).-compile(export_all).hello() ->Pid = spawn(?MODULE, world, []), link(Pid), ... .執行時,以 Pid =spawn(example,hello, []) 啟動程序,此程序將啟動另一個程序,並且與它連接。
以上 hello/0 函數前段使用process_flag/2函數,將trap_exit標籤打開,即可開啟程序回報錯誤機制。
hello() ->process_flag(trap_exit,true), Pid = spawn(?MODULE, world, []), link(Pid), ... .
於是,當程序結束時,會送出{'EXIT',From,Reason}資料。程序正常結束時,Reason為normal。
另外,spawn函數另外有程序連接版本,spawn_link函數,同時啟動並連接到新程序。
Erlang提供分散式機制,能在另一台電腦啟動一些Erlang程序,並由本機電腦對其他電腦的Erlang程序傳遞訊息。
$>erl -name node_1
$>erl -sname node_1
啟動新的網路節點時,Erlang使用epmd (Erlang埠號對應管理系統) 指派埠號,提供節點使用。
當知道一個網路節點名稱時,可以在該節點產生新程序。
% create a remote process and call the function web:start_server(Port, MaxConnections)% on machine RemoteNodeRemoteProcess =spawn(RemoteNode, web, start_server, [Port, MaxConnections]),在遠端節點產生新程序之後,可以使用平行式程式設計的技巧,與遠端程序通訊。
Erlang / 開源電信平台提供的程式庫,於分散式程式設計可以使用net_adm、net_kernel、slave、... 等模組,做網路通訊[9]。
Erlang程式員可以使用惰性求值。不過,必須使用λ演算式,才能做到惰性求值。
以下是惰性求值的一例:假設有個剖析器程式如下,由於及早求值特徵,本程式將不會求解。expr() -> alt(then(factor(), then(literal($+), factor())), then(factor(), then(literal($-), factor()))).factor() -> alt(then(term(), then(literal($*), term())), then(term(), then(literal($/), term()))).term() -> alt(number(), xthen(literal($(), thenx(expr(), literal($))))).此處使用λ演算式及適當使用函數名稱表示,就能進行求值。示例如下。expr() -> fun () -> alt(then(fun factor/0, then(literal($+), fun factor/0)), then(fun factor/0, then(literal($-), fun factor/0))) end.factor() -> fun () -> alt(then(fun term/0, then(literal($*), fun term/0)), then(fun term/0, then(literal($/), fun term/0))) end.term() -> fun () -> alt(number(), xthen(literal($(), thenx(expr(), literal($))))) end.
Erlang is conceptually similar to theoccam programming language, though it recasts the ideas ofCSP in a functional framework and uses asynchronous message passing.
|url-status=和|dead-url=只需其一 (帮助)