Flutter/Dart第11天:Dart函数方法详解
摘要:Dart语言是纯面向对象的编程语言,就算是函数(对象的成员函数一般称为方法)也是对象,它也有类型,那么函数也可以作为其他函数的参数,或者赋值给其他变量。除此之外,Dart中的函数还有什么特别之处、它有什么规则和约束……
Dart官方文档:https://dart.dev/language/functions
重要说明:本博客基于Dart官网文档,但并不是简单的对官网进行翻译,在覆盖核心功能情况下,我会根据个人研发经验,加入自己的一些扩展问题和场景验证。
Dart语言是纯面向对象的编程语言,就是是函数也是对象,它的类型就是Function
类(https://api.dart.dev/stable/3.1.3/dart-core/Function-class.html)。
如下代码样例,函数的不同实现。如果函数实现仅仅只有1个表达式,那么函数可以使用箭头语法:=> return expression;
1
2
3
4
5
6
7
| // 函数实现
bool isNoble(int atomicNumber) {
return _nobleGases[atomicNumber] != null;
}
// 箭头语法,上诉函数的简单语法
bool isNoble(int atomicNumber) => _nobleGases[atomicNumber] != null;
|
命名参数(必选,默认值)
参数格式:functionName({param1 = value1, param2, ...})
函数调用:functionName(param1: valueX, param2: value2, ...)
默认情况下,命名参数是可选的,除非显示增加required
标记。在Flutter中,尤其是Widget是构造函数,仅仅使用命名参数,尽管参数是必填参数。
特别注意:就算了增加required
的必选参数,它也可以是可空的。
位置参数(可选,默认值)
可选位置参数通过[]
包裹的参数列表,默认值为null
,可以设置默认值:
参数格式:functionName(param1, param2, [param3 = value3, param4]);
函数调用:functionName(value1, value2);或functionName(value1, value2, value3);或functionName(value1, value2, value3, value4);
main()函数
任何Dart应用,都必须包含一个顶级main()
函数,它是应用的唯一入口。它的返回值是void
,入参是List<String>
类型。
代码样例:如下代码,我们在执行Dart文件时,指定了参数dart args.dart 1 test
1
2
3
4
5
6
7
| void main(List<String> arguments) {
print(arguments);
assert(arguments.length == 2);
assert(int.parse(arguments[0]) == 1);
assert(arguments[1] == 'test');
}
|
函数作为参数
函数可以作为其他函数的入参,也可以赋值给变量。
1
2
3
4
5
6
7
8
9
10
11
| // 1. 函数作为函数入参
void printElement(int element) {
print(element);
}
var list = [1, 2, 3];
list.forEach(printElement);
// 2. 函数赋值给变量(匿名函数)
var loudify = (msg) => '!!! ${msg.toUpperCase()} !!!';
assert(loudify('hello') == '!!! HELLO !!!');
|
匿名函数
如上代码,main()
和printElement()
函数都是命名函数。我们也可以创建匿名函数(如上代码loudify
变量的值),特别是在Lambda表达式
或者在闭包函数
中,匿名函数使用场景很多。
代码样例:如下代码,map()
和forEach()
的入参就是匿名函数。
1
2
3
4
5
6
| void main() {
const list = ['apples', 'bananas', 'oranges'];
list
.map((item) => item.toUpperCase())
.forEach((item) => print('$item: ${item.length}'));
}
|
函数相等校验
可以进行相等校验的函数包括:顶级函数,静态函数和实例函数。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
| // 1. 顶级函数
void foo() {}
class A {
// 2. 静态方法
static void bar() {}
// 3. 实例方法
void baz() {}
}
void main() {
Function x;
// 顶级函数相等校验
x = foo;
assert(foo == x);
// 静态方法相等校验
x = A.bar;
assert(A.bar == x);
// 实例方法相等校验
var v = A(); // A实例#1
var w = A(); // A实例#2
var y = w;
x = w.baz;
// 同是A实例#2方法
assert(y.baz == x);
// 不同实例方法
assert(v.baz != w.baz);
}
|
函数返回值
如果没有显示返回值,函数默认返回null
,Record记录可以聚合返回多个值。
1
2
3
| (String, int) foo() {
return ('something', 42);
}
|
生成器函数
生成器函数可以延迟产出一系列值,Dart中内置2类生成器函数:
- 同步生成器:返回1个Iterable对象。
- 异步生成器:返回1个Stream对象。
同步生成器函数:使用sync*
标记函数体,并且使用yield
表达式产生值。
1
2
3
4
| Iterable<int> naturalsTo(int n) sync* {
int k = 0;
while (k < n) yield k++;
}
|
异步生成器函数:使用async*
标记函数体,并且使用yield
表达式产生值。
1
2
3
4
| Stream<int> asynchronousNaturalsTo(int n) async* {
int k = 0;
while (k < n) yield k++;
}
|
如果生成器是递归的,可以通过yield*
来提升性能。
1
2
3
4
5
6
| Iterable<int> naturalsDownFrom(int n) sync* {
if (n > 0) {
yield n;
yield* naturalsDownFrom(n - 1);
}
}
|
我的本博客原地址:https://ntopic.cn/p/2023100601