在 Kotlin 中创建和使用函数

1. 开始之前

在之前的 codelab 中,您看到了一个打印 Hello, world! 的简单程序。在您编写的程序中,您已经看到了两个函数

  • 一个 main() 函数,每个 Kotlin 程序都需要它。它是程序的入口点或起点。
  • 一个 println() 函数,您从 main() 中调用它来输出文本。

在本 codelab 中,您将了解有关函数的更多信息。

函数允许您将代码分解成可重用的部分,而不是将所有内容都包含在 main() 中。函数是 Android 应用的基本构建块,学习如何定义和使用它们是您成为 Android 开发者的旅程中的一个重要步骤。

先决条件

  • 了解 Kotlin 编程基础知识,包括变量以及 println()main() 函数

您将学到什么

  • 如何定义和调用您自己的函数。
  • 如何从函数返回可以存储在变量中的值。
  • 如何定义和调用具有多个参数的函数。
  • 如何使用命名参数调用函数。
  • 如何为函数参数设置默认值。

您需要什么

  • 一个可以访问 Kotlin Playground 的网络浏览器

2. 定义和调用函数

在深入探讨函数之前,让我们回顾一些基本术语。

  • 声明(或定义)函数使用 fun 关键字,并在花括号内包含代码,其中包含执行任务所需的指令。
  • 调用函数会导致该函数中包含的所有代码执行。

到目前为止,您已将所有代码都写入 main() 函数中。 main() 函数实际上并没有在您的代码中的任何地方被调用;Kotlin 编译器将其用作起点。 main() 函数仅用于包含您要执行的其他代码,例如对 println() 函数的调用。

println() 函数是 Kotlin 语言的一部分。但是,您可以定义自己的函数。如果您需要多次调用它,这将允许您的代码被重用。以下程序就是一个例子。

fun main() {
    println("Happy Birthday, Rover!")
    println("You are now 5 years old!")
}

main() 函数包含两个 println() 语句——一个祝 Rover 生日快乐,另一个说明 Rover 的年龄。

虽然 Kotlin 允许您将所有代码都放在 main() 函数中,但您可能并不总是希望这样做。例如,如果您还想让程序包含新年问候,则 main 函数也必须包含对 println() 的这些调用。或者,您可能希望多次向 Rover 问好。您可以简单地复制粘贴代码,也可以为生日问候创建一个单独的函数。您将执行后者。为特定任务创建单独的函数有很多好处。

  • 可重用代码:与其复制粘贴需要多次使用的代码,您可以简单地在需要的地方调用函数。
  • 可读性:确保函数只执行一项特定任务,这有助于其他开发人员和团队成员,以及您未来的自己准确地了解一段代码的作用。

定义函数的语法如下所示。

Syntax for defining a function

函数定义以 fun 关键字开头,后跟函数名称、一组开括号和闭括号以及一组开花括号和闭花括号。花括号中包含调用函数时将运行的代码。

您将创建一个新函数,将这两个 println() 语句从 main() 函数中移出。

  1. 在您的浏览器中,打开 Kotlin Playground 并将内容替换为以下代码。
fun main() {
    println("Happy Birthday, Rover!")
    println("You are now 5 years old!")
}
  1. main() 函数之后,定义一个名为 birthdayGreeting() 的新函数。此函数的声明语法与 main() 函数相同。
fun main() {
    println("Happy Birthday, Rover!")
    println("You are now 5 years old!")
}

fun birthdayGreeting() {
    
}
  1. main() 中的两个 println() 语句移动到 birthdayGreeting() 函数的花括号中。
fun main() {
    
}

fun birthdayGreeting() {
    println("Happy Birthday, Rover!")
    println("You are now 5 years old!")
}
  1. main() 函数中,调用 birthdayGreeting() 函数。完成后的代码应如下所示
fun main() {
    birthdayGreeting()
}

fun birthdayGreeting() {
    println("Happy Birthday, Rover!")
    println("You are now 5 years old!")
}
  1. 运行您的代码。您应该看到以下输出
Happy Birthday, Rover!
You are now 5 years old!

3. 从函数返回值

在更复杂的应用中,函数的作用不仅仅是打印文本。

Kotlin 函数还可以生成称为*返回值*的数据,该数据存储在变量中,您可以在代码的其他地方使用它。

在定义函数时,您可以指定要返回的值的数据类型。返回类型通过在括号后放置一个冒号 (:),一个空格,然后是类型的名称 (IntString 等) 来指定。然后在返回类型和开花括号之间放置一个空格。在函数体中,在所有语句之后,您使用 return 语句指定您希望函数返回的值。return 语句由 return 关键字后跟要作为输出返回的值(例如变量)组成。

声明具有返回类型的函数的语法如下所示。

Syntax for declaring a function with a return type

Unit 类型

默认情况下,如果您未指定返回类型,则默认返回类型为 UnitUnit 表示函数不返回值。 Unit 等于其他语言中的 void 返回类型(Java 和 C 中的 void;Swift 中的 Void/空元组 ();Python 中的 None 等)。任何不返回值的函数都会隐式返回 Unit。您可以通过修改代码以返回 Unit 来观察这一点。

  1. birthdayGreeting() 的函数声明中,在闭括号后添加一个冒号,并将返回类型指定为 Unit
fun main() {
    birthdayGreeting()
}

fun birthdayGreeting(): Unit {
    println("Happy Birthday, Rover!")
    println("You are now 5 years old!")
}
  1. 运行代码并观察一切仍然有效。
Happy Birthday, Rover!
You are now 5 years old!

在 Kotlin 中,指定 Unit 返回类型是可选的。对于不返回任何内容或返回 Unit 的函数,您不需要 return 语句。

birthdayGreeting() 返回 String

为了演示函数如何返回值,您将修改 birthdayGreeting() 函数以返回字符串,而不是简单地打印结果。

  1. Unit 返回类型替换为 String
fun birthdayGreeting(): String {
    println("Happy Birthday, Rover!")
    println("You are now 5 years old!")
}
  1. 运行您的代码。您将收到错误。如果为函数声明返回类型(例如,String),则该函数必须包含一个 return 语句。
A 'return' expression required in a function with a block body ('{...}')
  1. 您只能从一个函数返回一个字符串,而不是两个。将 println() 语句替换为两个变量 nameGreetingageGreeting,使用 val 关键字。因为您已从 birthdayGreeting() 中删除了对 println() 的调用,所以调用 birthdayGreeting() 不会打印任何内容。
fun birthdayGreeting(): String {
    val nameGreeting = "Happy Birthday, Rover!"
    val ageGreeting = "You are now 5 years old!"
}
  1. 使用您在之前的 codelab 中学习的字符串格式化语法,添加一个 return 语句以从函数返回一个字符串,该字符串包含这两个问候语。

为了在单独的行上格式化问候语,您还需要使用 \n 转义字符。这就像您在之前的 codelab 中学习的 \" 转义字符一样。 \n 字符将被替换为换行符,以便两个问候语分别位于单独的行上。

fun birthdayGreeting(): String {
    val nameGreeting = "Happy Birthday, Rover!"
    val ageGreeting = "You are now 5 years old!"
    return "$nameGreeting\n$ageGreeting"
}
  1. main() 中,因为 birthdayGreeting() 返回一个值,所以您可以将结果存储在字符串变量中。声明一个使用 valgreeting 变量以存储调用 birthdayGreeting() 的结果。
fun main() {
    val greeting = birthdayGreeting()
}
  1. main() 中,调用 println() 以打印 greeting 字符串。 main() 函数现在应如下所示。
fun main() {
    val greeting = birthdayGreeting()
    println(greeting)
}
  1. 运行您的代码,然后观察结果与之前相同。返回值允许您将结果存储在变量中,但是如果您在 println() 函数内调用 birthdayGreeting() 函数,您认为会发生什么?
Happy Birthday, Rover!
You are now 5 years old!
  1. 删除变量,然后将调用 birthdayGreeting() 函数的结果传递到 println() 函数中
fun main() {
    println(birthdayGreeting())
}
  1. 运行您的代码并观察输出。调用 birthdayGreeting() 的返回值直接传递到 println() 中。
Happy Birthday, Rover!
You are now 5 years old!

4. 向 birthdayGreeting() 函数添加参数

如您所见,当您调用 println() 时,您可以在括号内包含字符串或将值传递给函数。您也可以对 birthdayGreeting() 函数执行相同的操作。但是,您首先需要向 birthdayGreeting() 添加一个参数

参数指定变量的名称和数据类型,您可以将其作为数据传递给函数,以便在函数内部访问。参数在函数名后的括号内声明。

Syntax for declaring a function with parameters and a return type

每个参数都包含一个变量名和数据类型,用冒号和空格分隔。多个参数用逗号分隔。

现在,birthdayGreeting() 函数只能用于问候 Rover。您将向 birthdayGreeting() 函数添加一个参数,以便您可以问候传递给函数的任何名称。

  1. birthdayGreeting() 函数的括号内,添加一个类型为 Stringname 参数,使用语法 name: String
fun birthdayGreeting(name: String): String {
    val nameGreeting = "Happy Birthday, Rover!"
    val ageGreeting = "You are now 5 years old!"
    return "$nameGreeting\n$ageGreeting"
}

上一步中定义的参数就像使用 val 关键字声明的变量一样。它的值可以在 birthdayGreeting() 函数中的任何位置使用。在之前的 codelab 中,您了解了如何将变量的值插入字符串。

  1. nameGreeting 字符串中的 Rover 替换为 $ 符号后跟 name 参数。
fun birthdayGreeting(name: String): String {
    val nameGreeting = "Happy Birthday, $name!"
    val ageGreeting = "You are now 5 years old!"
    return "$nameGreeting\n$ageGreeting"
}
  1. 运行您的代码并观察错误。现在您已经声明了 name 参数,您需要在调用 birthdayGreeting() 时传入一个 String。当您调用一个带参数的函数时,您会向函数传递一个参数。参数是您传递的值,例如 "Rover"
No value passed for parameter 'name'
  1. "Rover" 传递到 main() 中的 birthdayGreeting() 调用。
fun main() {
    println(birthdayGreeting("Rover"))
}
  1. 运行您的代码并观察输出。名称 Rover 来自 name 参数。
Happy Birthday, Rover!
You are now 5 years old!
  1. 因为 birthdayGreeting() 带有一个参数,所以您可以使用除 Rover 之外的名称调用它。在对 println() 的调用中添加另一个对 birthdayGreeting() 的调用,传入参数 "Rex"
println(birthdayGreeting("Rover"))
println(birthdayGreeting("Rex"))
  1. 再次运行代码,然后观察输出根据传递给 birthdayGreeting() 的参数的不同而不同。
Happy Birthday, Rover!
You are now 5 years old!
Happy Birthday, Rex!
You are now 5 years old!

5. 带有多个参数的函数

之前,您添加了一个参数来根据名称更改问候语。但是,您也可以为一个函数定义多个参数,甚至不同数据类型的参数。在本节中,您将修改问候语,使其也根据狗的年龄而改变。

Function with multiple parameters

参数定义用逗号分隔。类似地,当您使用多个参数调用函数时,您也用逗号分隔传入的参数。让我们看看它是如何工作的。

  1. name 参数之后,向 birthdayGreeting() 函数添加一个类型为 Intage 参数。新的函数声明应该有两个参数,nameage,用逗号分隔
fun birthdayGreeting(name: String, age: Int): String {
    val nameGreeting = "Happy Birthday, $name!"
    val ageGreeting = "You are now 5 years old!"
    return "$nameGreeting\n$ageGreeting"
}
  1. 新的问候语字符串应该使用 age 参数。更新 birthdayGreeting() 函数以在 ageGreeting 字符串中使用 age 参数的值。
fun birthdayGreeting(name: String, age: Int): String {
    val nameGreeting = "Happy Birthday, $name!"
    val ageGreeting = "You are now $age years old!"
    return "$nameGreeting\n$ageGreeting"
}
  1. 运行函数,然后注意输出中的错误
No value passed for parameter 'age'
No value passed for parameter 'age'
  1. 修改 main() 中对 birthdayGreeting() 函数的两次调用,为每只狗传入不同的年龄。为 Rover 的年龄传入 5,为 Rex 的年龄传入 2
fun main() {
    println(birthdayGreeting("Rover", 5))
    println(birthdayGreeting("Rex", 2))
}
  1. 运行您的代码。现在您已经为两个参数都传入值,当您调用函数时,输出应该反映每只狗的姓名和年龄。
Happy Birthday, Rover!
You are now 5 years old!
Happy Birthday, Rex!
You are now 2 years old!

函数签名

到目前为止,您已经了解了如何定义函数名、输入(参数)和输出。函数名及其输入(参数)统称为函数签名。函数签名包含返回类型之前的所有内容,并在以下代码片段中显示。

fun birthdayGreeting(name: String, age: Int)

用逗号分隔的参数有时称为参数列表。

您经常会在其他开发人员编写的代码的文档中看到这些术语。函数签名告诉您函数的名称以及可以传入哪些数据类型。

您已经学习了很多关于定义函数的新语法。查看以下图表以回顾函数语法。

Diagram for a recap of function syntax

6. 命名参数

在前面的示例中,您无需在调用函数时指定参数名称 nameage。但是,如果选择,您可以这样做。例如,您可能使用许多参数调用函数,或者您可能希望以不同的顺序传递参数,例如将 age 参数放在 name 参数之前。当您在调用函数时包含参数名称时,称为命名参数。尝试使用 birthdayGreeting() 函数使用命名参数。

  1. 修改 Rex 的调用以使用命名参数,如以下代码片段所示。您可以通过包含参数名称后跟等号,然后是值(例如 name = "Rex")来执行此操作。
println(birthdayGreeting(name = "Rex", age = 2))
  1. 运行代码,然后观察输出未发生变化
Happy Birthday, Rover!
You are now 5 years old!
Happy Birthday, Rex!
You are now 2 years old!
  1. 重新排序命名参数。例如,将 age 命名参数放在 name 命名参数之前。
println(birthdayGreeting(age = 2, name = "Rex"))
  1. 运行代码并观察输出保持不变。即使您更改了参数的顺序,也会为相同的参数传入相同的值。
Happy Birthday, Rover!
You are now 5 years old!
Happy Birthday, Rex!
You are now 2 years old!

7. 默认参数

函数参数也可以指定默认参数。也许 Rover 是您最喜欢的狗,或者您期望函数在大多数情况下使用特定参数被调用。当您调用函数时,您可以选择省略具有默认值的参数,在这种情况下,将使用默认值。

要添加默认参数,您需要在参数的数据类型之后添加赋值运算符 (=) 并将其设置为一个值。修改您的代码以使用默认参数。

  1. birthdayGreeting() 函数中,将 name 参数设置为默认值 "Rover"
fun birthdayGreeting(name: String = "Rover", age: Int): String {
    return "Happy Birthday, $name! You are now $age years old!"
}
  1. main() 中对 Rover 的 birthdayGreeting() 函数的第一次调用中,将 age 命名参数设置为 5。因为 age 参数是在 name 之后定义的,所以您需要使用命名参数 age。如果没有命名参数,Kotlin 会假设参数的顺序与参数定义中的顺序相同。命名参数用于确保 Kotlin 期望 age 参数为 Int
println(birthdayGreeting(age = 5))
println(birthdayGreeting("Rex", 2))
  1. 运行您的代码。对 birthdayGreeting() 函数的第一次调用打印 "Rover" 作为名称,因为您从未指定名称。对 birthdayGreeting() 函数的第二次调用仍然使用 Rex 值,您为 name 传入了该值。
Happy Birthday, Rover! You are now 5 years old!
Happy Birthday, Rex! You are now 2 years old!
  1. 从对 birthdayGreeting() 函数的第二次调用中删除名称。同样,因为省略了 name,所以您需要为年龄使用命名参数。
println(birthdayGreeting(age = 5))
println(birthdayGreeting(age = 2))
  1. 运行您的代码,然后观察现在对 birthdayGreeting() 函数的两次调用都打印 "Rover" 作为名称,因为没有传入名称参数。
Happy Birthday, Rover! You are now 5 years old!
Happy Birthday, Rover! You are now 2 years old!

8. 结论

恭喜!您学习了如何在 Kotlin 中定义和调用函数。

摘要

  • 函数使用 fun 关键字定义,并包含可重用的代码段。
  • 函数有助于使大型程序更易于维护并防止代码的无谓重复。
  • 函数可以返回值,您可以将其存储在变量中以供以后使用。
  • 函数可以接受参数,参数是在函数体中可用的变量。
  • 参数是您在调用函数时传入的值。
  • 您可以在调用函数时命名参数。当您使用命名参数时,您可以重新排序参数而不影响输出。
  • 您可以指定一个默认参数,让您在调用函数时省略该参数。