内容

函数

Dart 是一种真正的面向对象语言,因此即使是函数也是对象,并且具有类型 Function。这意味着函数可以分配给变量或作为参数传递给其他函数。您还可以调用 Dart 类的实例,就好像它是函数一样。有关详细信息,请参阅 可调用对象

以下是如何实现函数的示例

dart
bool isNoble(int atomicNumber) {
  return _nobleGases[atomicNumber] != null;
}

尽管 Effective Dart 建议 为公共 API 添加类型注释,但即使您省略类型,函数仍然可以工作

dart
isNoble(atomicNumber) {
  return _nobleGases[atomicNumber] != null;
}

对于只包含一个表达式的函数,您可以使用简写语法

dart
bool isNoble(int atomicNumber) => _nobleGases[atomicNumber] != null;

=> expr 语法是 { return expr; } 的简写。=> 符号有时被称为箭头语法。

参数

#

函数可以具有任意数量的必需位置参数。这些参数可以后跟命名参数或可选位置参数(但不能同时使用两者)。

在向函数传递参数或定义函数参数时,可以使用 尾部逗号

命名参数

#

命名参数是可选的,除非它们被明确标记为 required

在定义函数时,使用 {param1, param2, …} 指定命名参数。如果您没有提供默认值或将命名参数标记为 required,则它们的类型必须是可空类型,因为它们的默认值为 null

dart
/// Sets the [bold] and [hidden] flags ...
void enableFlags({bool? bold, bool? hidden}) {...}

在调用函数时,您可以使用 paramName: value 指定命名参数。例如

dart
enableFlags(bold: true, hidden: false);

要为命名参数定义除了 null 之外的默认值,请使用 = 指定默认值。指定的值必须是编译时常量。例如

dart
/// Sets the [bold] and [hidden] flags ...
void enableFlags({bool bold = false, bool hidden = false}) {...}

// bold will be true; hidden will be false.
enableFlags(bold: true);

如果您希望命名参数变为强制性,要求调用者为该参数提供值,请使用 required 对其进行注释

dart
const Scrollbar({super.key, required Widget child});

如果有人尝试在不指定 child 参数的情况下创建 Scrollbar,那么分析器会报告一个问题。

您可能希望将位置参数放在前面,但 Dart 并不强制要求这样做。Dart 允许在适合您的 API 时将命名参数放在参数列表中的任何位置

dart
repeat(times: 2, () {
  ...
});

可选位置参数

#

将一组函数参数用 [] 括起来表示它们是可选位置参数。如果您没有提供默认值,则它们的类型必须是可空类型,因为它们的默认值为 null

dart
String say(String from, String msg, [String? device]) {
  var result = '$from says $msg';
  if (device != null) {
    result = '$result with a $device';
  }
  return result;
}

以下是如何在不使用可选参数的情况下调用此函数的示例

dart
assert(say('Bob', 'Howdy') == 'Bob says Howdy');

以下是如何使用第三个参数调用此函数的示例

dart
assert(say('Bob', 'Howdy', 'smoke signal') ==
    'Bob says Howdy with a smoke signal');

要为可选位置参数定义除了 null 之外的默认值,请使用 = 指定默认值。指定的值必须是编译时常量。例如

dart
String say(String from, String msg, [String device = 'carrier pigeon']) {
  var result = '$from says $msg with a $device';
  return result;
}

assert(say('Bob', 'Howdy') == 'Bob says Howdy with a carrier pigeon');

main() 函数

#

每个应用程序都必须具有一个顶层 main() 函数,它充当应用程序的入口点。main() 函数返回 void,并具有可选的 List<String> 参数用于参数。

以下是一个简单的 main() 函数

dart
void main() {
  print('Hello, World!');
}

以下是如何为接受参数的命令行应用程序编写 main() 函数的示例

args.dart
dart
// Run the app like this: dart run args.dart 1 test
void main(List<String> arguments) {
  print(arguments);

  assert(arguments.length == 2);
  assert(int.parse(arguments[0]) == 1);
  assert(arguments[1] == 'test');
}

您可以使用 args 库 来定义和解析命令行参数。

函数作为一等公民

#

您可以将函数作为参数传递给另一个函数。例如

dart
void printElement(int element) {
  print(element);
}

var list = [1, 2, 3];

// Pass printElement as a parameter.
list.forEach(printElement);

您也可以将函数分配给变量,例如

dart
var loudify = (msg) => '!!! ${msg.toUpperCase()} !!!';
assert(loudify('hello') == '!!! HELLO !!!');

此示例使用匿名函数。有关更多信息,请参阅下一节。

函数类型

#

您可以指定函数的类型,这被称为函数类型。函数类型是从函数声明头中获取的,方法是将函数名替换为关键字 Function。此外,您可以省略位置参数的名称,但不能省略命名参数的名称。例如

dart
void greet(String name, {String greeting = 'Hello'}) =>
    print('$greeting $name!');

// Store `greet` in a variable and call it.
void Function(String, {String greeting}) g = greet;
g('Dash', greeting: 'Howdy');

匿名函数

#

尽管您为大多数函数命名,例如 main()printElement(),但您也可以创建没有名称的函数。这些函数被称为匿名函数lambda 表达式闭包

匿名函数类似于命名函数,因为它具有

  • 零个或多个参数,用逗号分隔
  • 括号之间可选的类型注释。

以下代码块包含函数的主体

dart
([[Type] param1[, ...]]) {
  codeBlock;
}

以下示例定义了一个带有一个未类型化参数 item 的匿名函数。匿名函数将其传递给 map 函数。map 函数为列表中的每个项目调用,将每个字符串转换为大写。然后,传递给 forEach 的匿名函数打印每个转换后的字符串及其长度。

dart
const list = ['apples', 'bananas', 'oranges'];

var uppercaseList = list.map((item) {
  return item.toUpperCase();
}).toList();
// Convert to list after mapping

for (var item in uppercaseList) {
  print('$item: ${item.length}');
}

单击运行以执行代码。

void main() {
  const list = ['apples', 'bananas', 'oranges'];

  var uppercaseList = list.map((item) {
    return item.toUpperCase();
  }).toList();
  // Convert to list after mapping

  for (var item in uppercaseList) {
    print('$item: ${item.length}');
  }
}

如果函数只包含单个表达式或返回语句,则可以使用箭头符号将其缩短。将以下行粘贴到 DartPad 中,然后单击运行以验证它在功能上是等效的。

dart
var uppercaseList = list.map((item) => item.toUpperCase()).toList();
uppercaseList.forEach((item) => print('$item: ${item.length}'));

词法范围

#

Dart 根据其代码的布局确定变量的范围。具有此功能的编程语言被称为词法作用域语言。您可以“向外跟随大括号”以查看变量是否在范围内。

示例:一系列嵌套函数,每个函数都包含不同范围级别的变量

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() 方法可以使用所有级别的变量,一直到顶层。

词法闭包

#

一个函数对象,当函数位于该词法范围之外时可以访问其词法范围内的变量,被称为闭包

函数可以闭包周围范围中定义的变量。在以下示例中,makeAdder() 捕获变量 addBy。无论返回的函数去哪里,它都会记住 addBy

dart
/// Returns a function that adds [addBy] to the
/// function's argument.
Function makeAdder(int addBy) {
  return (int i) => addBy + i;
}

void main() {
  // Create a function that adds 2.
  var add2 = makeAdder(2);

  // Create a function that adds 4.
  var add4 = makeAdder(4);

  assert(add2(3) == 5);
  assert(add4(3) == 7);
}

断言

#

当您引用一个函数、方法或命名构造函数但不带括号时,Dart 会创建一个 *断裂*。这是一种闭包,它接收与函数相同的参数,并在您调用它时调用底层函数。如果您的代码需要一个调用命名函数的闭包,该函数具有与闭包接受的参数相同的参数,请不要将调用包装在 lambda 中。使用断裂。

dart
var charCodes = [68, 97, 114, 116];
var buffer = StringBuffer();
良好dart
// Function tear-off
charCodes.forEach(print);

// Method tear-off
charCodes.forEach(buffer.write);
不良dart
// Function lambda
charCodes.forEach((code) {
  print(code);
});

// Method lambda
charCodes.forEach((code) {
  buffer.write(code);
});

测试函数是否相等

#

以下是一个测试顶级函数、静态方法和实例方法是否相等的示例

dart
void foo() {} // A top-level function

class A {
  static void bar() {} // A static method
  void baz() {} // An instance method
}

void main() {
  Function x;

  // Comparing top-level functions.
  x = foo;
  assert(foo == x);

  // Comparing static methods.
  x = A.bar;
  assert(A.bar == x);

  // Comparing instance methods.
  var v = A(); // Instance #1 of A
  var w = A(); // Instance #2 of A
  var y = w;
  x = w.baz;

  // These closures refer to the same instance (#2),
  // so they're equal.
  assert(y.baz == x);

  // These closures refer to different instances,
  // so they're unequal.
  assert(v.baz != w.baz);
}

返回值

#

所有函数都会返回值。如果未指定返回值,则语句 return null; 会隐式附加到函数体。

dart
foo() {}

assert(foo() == null);

要在函数中返回多个值,请将这些值聚合到一个 记录 中。

dart
(String, int) foo() {
  return ('something', 42);
}

生成器

#

当您需要延迟生成一系列值时,请考虑使用 *生成器函数*。Dart 内置支持两种类型的生成器函数

  • 同步 生成器:返回一个 Iterable 对象。
  • 异步 生成器:返回一个 Stream 对象。

要实现一个 同步 生成器函数,请将函数体标记为 sync*,并使用 yield 语句来传递值

dart
Iterable<int> naturalsTo(int n) sync* {
  int k = 0;
  while (k < n) yield k++;
}

要实现一个 异步 生成器函数,请将函数体标记为 async*,并使用 yield 语句来传递值

dart
Stream<int> asynchronousNaturalsTo(int n) async* {
  int k = 0;
  while (k < n) yield k++;
}

如果您的生成器是递归的,您可以使用 yield* 来提高其性能

dart
Iterable<int> naturalsDownFrom(int n) sync* {
  if (n > 0) {
    yield n;
    yield* naturalsDownFrom(n - 1);
  }
}

外部函数

#

外部函数是一个函数,其函数体是在其声明之外实现的。在函数声明之前包含 external 关键字,如下所示

dart
external void someFunc(int i);

外部函数的实现可以来自另一个 Dart 库,或者更常见的是,来自另一种语言。在互操作上下文中,external 为外部函数或值引入类型信息,使它们可以在 Dart 中使用。实现和使用在很大程度上依赖于平台,因此请查看互操作文档,例如 CJavaScript 以了解更多信息。

外部函数可以是顶级函数、实例方法getter 或 setter非重定向构造函数。一个 实例变量 也可以是 external,这等效于一个外部 getter 和(如果变量不是 final)一个外部 setter。