mirror of
https://github.com/erg-lang/erg.git
synced 2025-09-30 21:01:10 +00:00
3.9 KiB
3.9 KiB
元组
元组类似于数组,但可以保存不同类型的对象。 这样的集合称为不等集合。 相比之下,同构集合包括数组、集合等。
t = (1, True, "a")
(i, b, s) = t
assert(i == 1 and b == True and s == "a")
元组t
可以以t.n
的形式检索第n个元素; 请注意,与 Python 不同,它不是 t[n]
。
这是因为访问元组元素更像是一个属性(在编译时检查元素的存在,并且类型可以根据 n
改变)而不是方法(数组的 []
是一种方法)。
assert t.0 == 1
assert t.1 == True
assert t.2 == "a"
括号 ()
在不嵌套时是可选的。
t = 1, True, "a"
i, b, s = t
元组可以保存不同类型的对象,因此它们不能像数组一样被迭代。
t: ({1}, {2}, {3}) = (1, 2, 3)
(1, 2, 3).iter().map(x -> x + 1) # 类型错误:类型 ({1}, {2}, {3}) 没有方法 `.iter()`
# 如果所有类型都相同,则可以像数组一样用`(T; n)`表示,但这仍然不允许迭代
t: (Int; 3) = (1, 2, 3)
assert (Int; 3) == (Int, Int, Int)
但是,非同质集合(如元组)可以通过向上转换、相交等方式转换为同质集合(如数组)。 这称为均衡。
(Int, Bool, Str) can be [T; 3] where T :> Int, T :> Bool, T :> Str
t: (Int, Bool, Str) = (1, True, "a") # 非同质
a: [Int or Bool or Str; 3] = [1, True, "a"] # 同质的
_a: [Show; 3] = [1, True, "a"] # 同质的
_a.iter().map(x -> log x) # OK
t.try_into([Show; 3])? .iter().map(x -> log x) # OK
单元
零元素的元组称为 unit。 一个单元是一个值,但也指它自己的类型。
unit = ()
(): ()
Unit 是所有元素 0 元组的父类。
() > (Int; 0)
() > (Str; 0)
该对象的用途是用于没有参数和没有返回值的过程等。Erg 子例程必须有参数和返回值。 但是,在某些情况下,例如过程,可能没有有意义的参数或返回值,只有副作用。 在这种情况下,我们将单位用作"无意义的正式值"
# ↓ Actually, this parenthesis is a unit
p!() =.
# `print!` does not return a meaningful value
print! "Hello, world!"
p!: () => ()
但是,在这种情况下,Python 倾向于使用"无"而不是单位。
在 Erg 中,当您从一开始就确定操作不会返回有意义的值(例如在过程中)时,您应该使用 ()
,并且当操作可能失败并且您可能会返回 None
将一无所获,例如在检索元素时。
参数和元组
实际上,Erg 的所有 Callable
对象都是一个参数和一个返回值; 一个接受 N 个参数的子例程只是接收"一个具有 N 个元素的元组"作为参数。
# f x = ... 被隐式假设为 f(x) = ... 被认为是
f x = x
assert f(1) == 1
f(1, 2, 3) # 参数错误:f 接受 1 个位置参数,但给出了 3 个
g x: Int, . . y: Int = y
assert (2, 3) == g 1, 2, 3
这也解释了函数类型。
assert f in T: {(T,) -> T | T}
assert g in {(Int, ... (Int; N)) -> (Int; N) | N: Nat}
准确地说,函数的输入不是元组,而是"具有默认属性的命名元组"。 这是一个特殊的元组,只能在函数参数中使用,可以像记录一样命名,并且可以有默认值。
f(x: Int, y=0) = x + y
f: (Int, y=Int) -> Int
f(x=0, y=1)
f(y=1, x=0)
f(x=0)
f(0)