函數
函數是一個塊,它接受一個"參數",對其進行處理,并將其作為"返回值"返回。定義如下
add x, y = x + y
# 或者
add(x, y) = x + y
在函數名之后指定的名稱稱為參數
相反,傳遞給函數的對象稱為參數
函數 add
是一個以 x
和 y
作為參數并返回它們之和的函數,x + y
可以按如下方式調用(應用/調用)定義的函數
add 1, 2
# or
add(1, 2)
冒號應用風格
函數像f x, y, ...
一樣被調用,但是如果單行參數太多,可以使用:
(冒號)來應用它們
f some_long_name_variable_1 + some_long_name_variable_2, some_long_name_variable_3 * some_long_name_variable_4
f some_long_name_variable_1 + some_long_name_variable_2:
some_long_name_variable_3 * some_long_name_variable_4
f:
some_long_name_variable_1 + some_long_name_variable_2
some_long_name_variable_3 * some_long_name_variable_4
以上三個代碼的含義相同。例如,這種風格在使用 if
函數時也很有用
result = if Bool.sample!():
do:
log "True was chosen"
1
do:
log "False was chosen"
0
在 :
之后,除了注釋之外,不得編寫任何代碼,并且必須始終在新行上
此外,您不能在函數后立即使用 :
。只有 do
和do!
可以做到這一點
# NG
f:
x
y
# Ok
f(
x,
y
)
關鍵字參數
如果使用大量參數定義函數,則存在以錯誤順序傳遞參數的危險 在這種情況下,使用關鍵字參數調用函數是安全的
f x, y, z, w, v, u: Int = ...
上面定義的函數有很多參數,并且排列順序混亂。您不應該創建這樣的函數,但是在使用別人編寫的代碼時可能會遇到這樣的代碼。因此,我們使用關鍵字參數。如果使用關鍵字參數,則值會從名稱傳遞到正確的參數,即使它們的順序錯誤
f u := 6, v := 5, w := 4, x := 1, y := 2, z := 3
定義默認參數
當某些參數大部分是固定的并且您希望能夠省略它們時,使用默認參數
默認參數由:=
(default-assign運算符)指定。如果未指定 base
,則將 math.E
分配給 base
math_log x: Ratio, base := math.E = ...
assert math_log(100, 10) == 2
assert math_log(100) == math_log(100, math.E)
請注意,不指定參數和指定None
是有區別的
p! x := 0 = print!
p!(2) # 2
p!() # 0
p!(None) # None
也可以與類型規范和模式一起使用
math_log x, base: Ratio := math.E = ...
f [x, y] := [1, 2] = ...
但是,在默認參數中,不能調用過程(稍后描述)或分配可變對象
f x := p! 1 = ... # NG
此外,剛剛定義的參數不能用作傳遞給默認參數的值
f x := 1, y := x = ... # NG
可變長度參數
輸出其參數的日志(記錄)的 log
函數可以采用任意數量的參數
log "你好", "世界", "!" # 你好 世界 !
要定義這樣的函數,請將 *
添加到參數中。這樣,函數將參數作為可變長度數組接收
f *x =
for x, i ->
log i
# x == [1, 2, 3, 4, 5]
f 1, 2, 3, 4, 5
具有多種模式的函數定義
fib n: Nat =
match n:
0 -> 0
1 -> 1
n -> fib(n - 1) + fib(n - 2)
像上面這樣的函數,其中 match
直接出現在定義下,可以重寫如下
fib 0 = 0
fib 1 = 1
fib(n: Nat): Nat = fib(n - 1) + fib(n - 2)
注意一個函數定義有多個模式不是所謂的重載(multiple definition); 一個函數只有一個定義。在上面的示例中,"n"必須與"0"或"1"屬于同一類型。此外,與 match
一樣,模式匹配是從上到下完成的
如果不同類的實例混合在一起,最后一個定義必須指定函數參數的類型為Or
f "aa" = ...
f 1 = ...
# `f x = ... ` 無效
f x: Int or Str = ...
此外,像 match
一樣,它也必須是詳盡的
fib 0 = 0
fib 1 = 1
# 模式錯誤: fib 參數的模式并不詳盡
但是,可以通過使用稍后描述的 refinement type 顯式指定類型來使其詳盡無遺
fib: 0..1 -> 0..1
fib 0 = 0
fib 1 = 1
# OK
遞歸函數
遞歸函數是在其定義中包含自身的函數
作為一個簡單的例子,讓我們定義一個執行階乘計算的函數factorial
。階乘是"將所有小于或等于的正數相乘"的計算
5 的階乘是 5*4*3*2*1 == 120
factorial 0 = 1
factorial 1 = 1
factorial(n: Nat): Nat = n * factorial(n - 1)
首先,從階乘的定義來看,0和1的階乘都是1
反過來,2的階乘是2*1 == 2
,3的階乘是3*2*1 == 6
,4的階乘是4*3*2*1 == 24
如果我們仔細觀察,我們可以看到一個數 n 的階乘是前一個數 n-1 乘以 n 的階乘
將其放入代碼中,我們得到 n * factorial(n - 1)
由于 factorial
的定義包含自身,factorial
是一個遞歸函數
提醒一下,如果您不添加類型規范,則會這樣推斷
factorial: |T <: Sub(Int, T) and Mul(Int, Int) and Eq(Int)| T -> Int
factorial 0 = 1
factorial 1 = 1
factorial n = n * factorial(n - 1)
但是,即使您可以推理,您也應該明確指定遞歸函數的類型。在上面的例子中,像"factorial(-1)"這樣的代碼可以工作,但是
factorial(-1) == -1 * factorial(-2) == -1 * -2 * factorial(-3) == ...
并且這種計算不會停止。遞歸函數必須仔細定義值的范圍,否則您可能會陷入無限循環 所以類型規范也有助于避免接受意外的值
High-order functions
高階函數是將函數作為參數或返回值的函數 例如,一個以函數為參數的高階函數可以寫成如下
arg_f = i -> log i
higher_f(x: (Int -> NoneType)) = x 10
higher_f arg_f # 10
當然,也可以將返回值作為一個函數。
add(x): (Int -> Int) = y -> x + y
add_ten = add(10) # y -> 10 + y
add_hundred = add(100) # y -> 100 + y
assert add_ten(1) == 11
assert add_hundred(1) == 101
通過這種方式將函數作為參數和返回值,可以用函數定義更靈活的表達式
編譯時函數
函數名以大寫字母開頭,表示編譯時函數。用戶定義的編譯時函數必須將所有參數作為常量,并且必須指定它們的類型 編譯時函數的功能有限。在編譯時函數中只能使用常量表達式,即只有一些運算符(例如求積、比較和類型構造操作)和編譯時函數。要傳遞的參數也必須是常量表達式 作為回報,優點是計算可以在編譯時完成
Add(X, Y: Nat): Nat = X + Y
assert Add(1, 2) == 3
Factorial 0 = 1
Factorial(X: Nat): Nat = X * Factorial(X - 1)
assert Factorial(10) == 3628800
math = import "math"
Sin X = math.sin X # 常量錯誤: 此函數在編譯時不可計算
編譯時函數也用于多態類型定義
Option T: Type = T or NoneType
Option: Type -> Type
附錄: 功能對比
Erg 沒有為函數定義 ==
。這是因為通常沒有函數的結構等價算法
f = x: Int -> (x + 1)**2
g = x: Int -> x**2 + 2x + 1
assert f == g # 類型錯誤: 無法比較函數
盡管 f
和 g
總是返回相同的結果,但要做出這樣的決定是極其困難的。我們必須向編譯器教授代數
所以 Erg 完全放棄了函數比較,并且 (x -> x) == (x -> x)
也會導致編譯錯誤。這是與 Python 不同的規范,應該注意
# Python,奇怪的例子
f = lambda x: x
assert f == f
assert (lambda x: x) ! = (lambda x: x)
Appendix2: ()-completion
f x: Object = ...
# 將完成到
f(x: Object) = ...
f a
# 將完成到
f(a)
f a, b # 類型錯誤: f() 接受 1 個位置參數,但給出了 2 個
f(a, b) # # 類型錯誤: f() 接受 1 個位置參數,但給出了 2 個
f((a, b)) # OK
函數類型T -> U
實際上是(T,) -> U
的語法糖