正如在 Mojo 语言基础中提到的,Mojo 支持两种类型的函数:deffn函数。您可以在任何函数中使用这两种声明方式,包括main()函数,但它们具有不同的默认行为,如本页面所述。

我们认为deffn都有很好的用例,并且不认为其中任何一种比另一种更好。选择使用哪种方式取决于个人口味,以及哪种风格最适合给定的任务。

我们相信 Mojo 在这方面的灵活性是一种超能力,它允许您以最适合项目的方式编写代码。

在结构体内声明的函数称为 “方法”,但它们具有与此处描述的 “函数” 完全相同的特性。

def 函数

def函数提供了与 Python 的def函数相同的动态性和灵活性。例如,这个函数在 Python 和 Mojo 中的工作方式相同:

  1. def greet(name):
  2. greeting = "Hello, " + name + "!"
  3. return greeting

在 Mojo 中,您还可以选择指定参数类型和返回类型。您还可以使用var声明变量,可以带有或不带有显式类型声明。

  1. def greet(name: String) -> String:
  2. var greeting = "Hello, " + name + "!"
  3. return greeting

这样,编译器确保name是一个字符串,并且返回类型也是一个字符串。

关于def函数的所有内容如下:

  • 参数不需要声明类型。

    未声明的参数实际上以 object 的形式传递,这允许函数接收任何类型( Mojo 在运行时推断类型)。

  • 返回类型不需要声明,也默认为 object

  • 参数是可变的(通常通过按值传递使用owned参数约定)。

    如果参数是 object 类型,则以引用的方式接收,遵循对象引用语义。

    如果参数是任何其他声明的类型,则以值的方式接收(使用owned参数约定)。

  • 变量不需要使用var进行声明。

object 类型

如果您在def函数中不声明参数或返回值的类型,则它将成为一个 object ,这与标准库中的任何其他类型都不同。

object 类型允许动态类型,因为它实际上可以表示 Mojo 标准库中的任何类型,并且实际类型在运行时推断出来(实际上,在它能表示所有 Mojo 类型之前还有更多的工作要做)。这对于与 Python 的兼容性非常有用,并且可以提供动态类型所提供的所有灵活性。然而,这种类型缺乏类型强制执行,当函数接收或返回意外类型时会导致运行时错误。

为了与 Python 兼容,使用对象引用语义来传递 object 值。因此, object 类型与强制执行值语义的参数约定不兼容。因此,如果在其他强类型值旁边使用 object 值,则它们的行为可能不一致,因为 object 是标准库中唯一不符合完全值语义的类型。

fn函数

fn函数提供了严格的类型检查和额外的内存安全性。它基本上强制您在def中编写的可选内容,并确保您不会意外地改变接收到的参数。例如,这是使用fn的相同函数:

  1. fn greet(name: String) -> String:
  2. var greeting = "Hello, " + name + "!"
  3. return greeting

对于函数调用者来说,def函数和fn函数是可以互换的。也就是说,def函数可以做的任何事情fn函数也能做(反之亦然)。区别在于,与def函数相比,fn函数在内部更加严格。

关于fn函数的所有内容如下:

  • 参数必须指定类型(结构体方法中的self参数除外)。

  • 返回值必须指定类型,除非函数不返回值。

    如果不指定返回类型,默认为None(表示没有返回值)。

  • 默认情况下,参数以不可变引用的方式接收(值是只读的,使用borrowed参数约定)。

    这可以防止意外的更改,并允许使用不可复制的类型作为参数。

    如果您想要一个局部副本,可以将值赋给一个局部变量。或者,您可以通过声明inout参数约定来获取对值的可变引用。

  • 必须使用var关键字声明变量。

  • 如果函数引发异常,必须使用raises关键字显式声明异常(def函数不需要声明异常)。

通过强制执行这些类型检查,使用fn函数有助于避免各种运行时错误。与def函数中的动态类型相比,它还改善了性能,因为在运行时不需要处理来确定使用什么数据类型所需的开销 - 类型在编译时固定。

可选参数

可选参数是包括默认值的参数,例如这里的exp参数:

  1. fn pow(base: Int, exp: Int = 2) -> Int:
  2. return base ** exp
  3. fn use_defaults():
  4. # 使用`exp`的默认值
  5. var z = pow(3)
  6. print(z)

但是,您不能为声明为inout的参数定义默认值。

关键字参数

在调用函数时,您还可以使用关键字参数。关键字参数使用格式argument_name = argument_value来指定。您可以以任何顺序传递关键字参数:

  1. fn pow(base: Int, exp: Int = 2) -> Int:
  2. return base ** exp
  3. fn main():
  4. var result = pow(base = 5, exp = 3)
  5. print(result)

这样可以更加清晰地表达参数的意图,并且不需要记住参数的顺序。

以上是关于函数的全部内容。