Dart函数
函数
Dart 是一种面向对象的语言,所以即便函数也是对象并且类型为Function
。函数可以被赋值给变量或者作为其它函数的参数。也可以像调用函数一样调用 Dart 类的实例。
1 | bool isNoble(int atomicNumber) { |
函数可以没有返回值
1 | isNoble(atomicNumber) { |
如果函数体内只包含一个表达式,可以使用简写语法:
1 | bool isNoble(int atomicNumber) => _nobleGases[atomicNumber] != null; |
语法
=> *表达式*
是{ return *表达式*; }
的简写,=>
有时也称之为 箭头 函数。在 => 与 ; 之间的只能是 表达式 而非 语句。比如不能将一个 if语句 放在其中,但是可以放置 条件表达式。
参数
函数可以有两种形式的参数:必要参数 和 可选参数。必要参数定义在参数列表前面,可选参数则定义在必要参数后面。可选参数可以是 命名的 或 位置的。
命名参数
命名参数默认为可选参数,除非他们被特别标记为 required
。
定义函数时,使用 {*参数1*, *参数2*, …}
来指定命名参数:
1 | /// Sets the [bold] and [hidden] flags ... |
当调用函数时,你可以使用 *参数名*: *参数值*
指定一个命名参数的值。例如:
1 | enableFlags(bold: true, hidden: false); |
尽管先使用位置参数会比较合理,但你也可以在任意位置使用命名参数,让整个调用的方式看起来更适合你的 API:
1 | repeat(times: 2, () { |
如果一个参数是可选的,但是不能为
null
,你需要为它提供一个 默认值。
虽然命名参数是可选参数的一种类型,但是你仍然可以使用 required
来标识一个命名参数是必须的参数,此时调用者必须为该参数提供一个值。例如:
1 | const Scrollbar({super.key, required Widget child}); |
可选的位置参数
使用 []
将一系列参数包裹起来作为位置参数:
1 | String say(String from, String msg, [String? device]) { |
默认参数值
可以用 =
为函数的命名参数和位置参数定义默认值,默认值必须为编译时常量,没有指定默认值的情况下默认值为 null
。
下面是设置可选参数默认值示例:
1 | /// Sets the [bold] and [hidden] flags ... |
main() 函数
每个 Dart 程序都必须有一个 main()
顶级函数作为程序的入口, main()
函数返回值为 void
并且有一个 List<String>
类型的可选参数。
下面是一个简单 main()
函数:
1 | void main() { |
下面是使用命令行访问带参数的 main()
函数示例:
1 | // Run the app like this: dart args.dart 1 test |
函数是一级对象
可以将函数作为参数传递给另一个函数。例如:
1 | void printElement(int element) { |
你也可以将函数赋值给一个变量,比如:
1 | var loudify = (msg) => '!!! ${msg.toUpperCase()} !!!'; |
匿名函数
大多数方法都是有名字的,比如 main()
或 printElement()
。你可以创建一个没有名字的方法,称之为 匿名函数、 Lambda 表达式 或 Closure 闭包。你可以将匿名方法赋值给一个变量然后使用它,比如将该变量添加到集合或从中删除。
匿名方法看起来与命名方法类似,在括号之间可以定义参数,参数之间用逗号分割。
后面大括号中的内容则为函数体:
([[类型] 参数[, …]]) {
函数体;
};
下面代码定义了只有一个参数 item
且没有参数类型的匿名方法。 List 中的每个元素都会调用这个函数,打印元素位置和值的字符串:
1 | const list = ['apples', 'bananas', 'oranges']; |
如果函数体内只有一行返回语句,你可以使用胖箭头缩写法。
词法作用域
Dart 是词法有作用域语言,变量的作用域在写代码的时候就确定了,大括号内定义的变量只能在大括号内访问,与 Java 类似。
下面是一个嵌套函数中变量在多个作用域中的示例:
1 | bool topLevel = true; |
词法闭包
闭包 即一个函数对象,即使函数对象的调用在它原始作用域之外,依然能够访问在它词法作用域内的变量。
函数可以封闭定义到它作用域内的变量。接下来的示例中,函数 makeAdder()
捕获了变量 addBy
。无论函数在什么时候返回,它都可以使用捕获的 addBy
变量。
1 | /// Returns a function that adds [addBy] to the |
测试函数是否相等
下面是顶级函数,静态方法和示例方法相等性的测试示例:
1 | void foo() {} // A top-level function |
返回值
所有的函数都有返回值。没有显示返回语句的函数最后一行默认为执行 return null;
。
1 | foo() {} |