2025-06-03
语言更新
1. 调用副作用函数时可省略 !
,IDE 自动高亮错误与异步
- 调用带副作用的函数(抛出错误或异步)时的 ! 现在可以省略,IDE 的语义着色会自动给带错误的函数加上下划线,将异步函数标记为斜体
2. f?(..)
替换为更灵活的 try?
表达式语法
f?(..)
语法被替代为一个新语法 try? f(..)
,try? expr
等价于 try Ok(expr) catch { err => Err(err) }
,可以把错误转换为Result
类型。相比原来的 f?(..)
,try?
具有更多的灵活性,它可以同时处理多处函数调用的错误。
例如
try? {
f(..)
g(..)
}
在迁移时,需要特别注意一种情况:f?(g!(..))
不能简单翻译为 try? f(g(..))
,因为这会导致 g
中的错误被重定向到 Result
中。这种情况需要先用 let
将 g
的结果提取出来,再使用 try?
3. 新增了 “错误多态” 功能,允许高阶函数同时处理带错误和不带错误的参数:
fn[X] Array::each(arr : Array[X], f : (X) -> Unit?Error) -> Unit?Error {
for x in arr {
f(x)
}
}
fn main {
let arr = [ "a", "b", "c" ]
arr.each(println) // 无错误
println(try? arr.each(fail)) // 输出 Err(Failure)
}
错误多态的语法是 (..) -> T?Error
,在调用时,?Error
可以被替换为任意 错误类型,也可以被抹去,变成没有错误的情况。例如,在上面的 each
函数中,如果参数 f
没有错误,那么 ?Error
会被替换为 “没有错误”,整个 each
函数的调用也不会抛出错误。如果 f
会抛出错误,那么 ?Error
会被替换为 f
实际的错误类型,整个 each
函数的调用也会抛出同样的错误
4. 废弃 fn meth(self: T)
风格,统一使用 fn T::meth(..)
定义方法
之前,使用 fn meth(self : T)
形式定义的方法既是方法也是函数,可以用 meth(..)
或 @pkg.meth(..)
的形式直接调用。这一 “将方法当作普通函数使用” 的行为将被废弃,目前编译器会对这一行为的使用提出警告。
这一调整后,我们鼓励的 API 设计方式是:
- 永远使用
fn T::meth(..)
的形式定义方法,新代码中不再使用fn meth(self : T)
(在以后,fn meth(self : T)
语法本身也可能被废弃) - 和某个类型绑定的 API 若无特殊理由,都鼓励设计成方法
5. .
语法支持使用 _
的匿名函数
写法是 _.meth(..)
,同时,管道运算符|>
的右手侧也支持了这种形式的匿名函数。注意使用这一语法时,必须能从上下文中得知_
的类型,否则无法解析方法。例子:
fn main {
let array_of_array = [ [], [ 1 ], [ 1, 2 ] ]
let lens = array_of_array.map(_.length())
lens |> _.each(println)
}
和直接写 x.meth(..)
相比,x |> _.meth(..)
的好处是,它可以把方法嵌入到一个现有的 |>
管线里,例如 x |> regular_func(..) |> _.meth(..)
6. 方法定义中支持用 Self
替代类型名,简化签名
在 fn TypeName::meth(..)
形式的方法声明中,可以使用 Self
来指代 TypeName
,缩短类型签名的长度:
type MyLongTypeName Int
fn MyLongTypeName::to_string(x : Self) -> String {
x._.to_string()
}
如果 TypeName
有参数,使用 Self
时也需要提供参数