Featured image of post Flutter/Dart第11天:Dart函数方法详解

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. 同步生成器:返回1个Iterable对象。
  2. 异步生成器:返回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


邮箱:obullxl@qq.com
QQ:303630027(老牛啊)
微信:imxulin(奔跑的蜗牛)
支付宝:obullxl@163.com