函数
Dart 是一门真正面向对象的语言,甚至其中的函数也是对象,并且有它自己的类型 Function
。这也意味着函数可以被赋值给变量或者作为参数传递给其他函数。也可以把 Dart
类的实例当作方法来调用。
1、函数声明
下面是函数的示例:
bool isPositiveInteger(int num) {
return num > 0;
}
如果函数中只有一句表达式,可以使用简写语法:
bool isPositiveInteger(int num) => num > 0;
=> expr
语法是{ return expr; }
的简写。=>
符号 有时也被称为 箭头 语法。
2、函数作为变量
var say = (str){
print(str);
};
say("hello world");
3、函数作为参数传递
void fn(var callback) {
callback();
}
fn(() => print("xxx"))
4、可选参数
为什么要使用可选参数?因为在方法参数过多,调用显得麻烦,在方法调用时不必传递所有参数,可选参数,又称为“默认参数”。
可选参数可以是命名参数或者位置参数,但是一个参数只能选择其中一种方式修饰。
命名可选参数
为什么要使用命名参数?因为使用命名参数可忽略参数的顺序,在调用时候非常方便,尤其是参数多的情况,调用时用参数名称和参数值同时出现的方法,同时提高代码的可读性。
定义函数是使用 *{ param1, param2, ... }
*来指定命名参数:
void enableFlags({bool bold, bool hidden}) {...}
调用函数时,可以使用指定命名参数 paramName: value
。 例如:
enableFlags(bold: true, hidden: false);
使用 @required
注释表示参数是 required
性质的命名参数:
const Scrollbar({Key key, Widget child});
此时 Scrollbar
是一个构造函数, 当 child
参数缺少时,分析器会提示错误。
位置可选参数
包装一组函数参数,用[]标记为可选的位置参数,并放在参数列表的最后面:
String say(String from, String msg, [String? device]) {
var result = '$from says $msg';
if (device != null) {
result = '$result with a $device';
}
print(result);
return result;
}
say('Bob', 'Howdy'); // 结果是: Bob says Howdy
say('Bob', 'Howdy', 'smoke signal'); // 结果是:Bob says Howdy with a smoke signal
默认参数值
在定义方法的时候,可以使用 =
来定义可选参数的默认值。 默认值只能是编译时常量。 如果没有提供默认值,则默认值为 null
。
下面是设置可选参数默认值示例:
/// 设置 [bold] 和 [hidden] 标志 ...
void enableFlags({bool bold = false, bool hidden = false}) {...}
// bold 值为 true; hidden 值为 false.
enableFlags(bold: true);
不推荐: 旧版本代码中可能使用的是冒号 (
:
) 而不是=
来设置参数默认值。 原因是起初命名参数只支持:
。 这种支持可能会被弃用。 建议 使用=
指定默认值。
下面示例演示了如何为位置参数设置默认值:
String say(String from, String msg, [
String device = 'carrier pigeon',
String mood
]) {
var result = '$from says $msg';
if (device != null) {
result = '$result with a $device';
}
if (mood != null) {
result = '$result (in a $mood mood)';
}
return result;
}
5、main() 函数
任何应用都必须有一个顶级 main()
函数,作为应用服务的入口。 main()
函数返回值为空,参数为一个可选的 List<String>
。
下面是 web 应用的 main()
函数:
void main() {
querySelector('#sample_text_id')
..text = 'Click me!'
..onClick.listen(reverseText);
}
提示:
以上代码中的
..
语法为级联调用
(cascade)。 使用级联调用, 可以简化在一个对象上执行的多个操作。
6、匿名函数
多数函数是有名字的,比如 main()
,也可以创建没有名字的函数,这种函数被称为 匿名函数,有时候也被称为 lambda
或者 closure
。匿名函数可以赋值到一个变量中,举个栗子,在一个集合中可以添加或者删除一个匿名函数。
匿名函数和命名函数看起来类似—在括号之间可以定义一些参数或可选参数,参数使用逗号分隔。后面大括号中的代码为函数体:
([[Type] param1[, …]]) {
codeBlock;
};
下面例子中定义了一个包含一个无类型参数 item
的匿名函数。 list 中的每个元素都会调用这个函数,打印元素位置和值的字符串。
var list = ['apples', 'bananas', 'oranges'];
list.forEach((item) {
print('${list.indexOf(item)}: $item');
});
如果函数只有一条语句, 可以使用箭头简写。
7、词法作用域
Dart 是一门词法作用域的编程语言,就意味着变量的作用域是固定的, 简单说变量的作用域在编写代码的时候就已经确定了。 花括号内的是变量可见的作用域。
下面示例关于多个嵌套函数的变量作用域:
bool topLevel = true;
void main() {
var insideMain = true;
void myFunction() {
var insideFunction = true;
void nestedFunction() {
var insideNestedFunction = true;
assert(topLevel);
assert(insideMain);
assert(insideFunction);
assert(insideNestedFunction);
}
}
}
注意 nestedFunction()
可以访问所有的变量,一直到顶级作用域变量。
8、词法闭包
闭包 即一个函数对象,即使函数对象的调用在它原始作用域之外, 依然能够访问在它词法作用域内的变量。
函数可以封闭定义到它作用域内的变量。 接下来的示例中, makeAdder()
捕获了变量 addBy
。 无论在什么时候执行返回函数,函数都会使用捕获的 addBy
变量。
// 返回一个函数,返回的函数参数与 [addBy] 相加。
Function makeAdder(num addBy) {
return (num i) => addBy + i;
}
void main() {
// 创建一个加 2 的函数。
var add2 = makeAdder(2);
// 创建一个加 4 的函数。
var add4 = makeAdder(4);
assert(add2(3) == 5);
assert(add4(3) == 7);
}