Dart 2 (三) 函数

函数

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

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

返回值类型使得运行高效,但是如果你省略了这些类型,这个函数仍然可以工作:

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

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

bool isNoble(int atomicNumber) => _nobleGases[atomicNumber] != null;
  • "=> expr"语法是{ return expr; }的简写,=>符号有时被称为胖箭头语法。在箭头(=>)和分号(;)之间只能出现表达式(而不是语句)。例如,您不能在那里放置if语句,但是您可以使用条件表达式。

一个函数可以有两种类型的参数:必需的和可选的。首先列出所需的参数,后跟任何可选参数。命名的可选参数也可以标记为@required。请看下面

可选参数

可选参数可以是位置参数,也可以是命名参数,但两者不能同时存在。

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

enableFlags(bold: true, hidden: false);

定义函数时,使用{param1, param2,…}指定指定参数:

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

flutter 实例创建表达式可能会变得复杂,因此小部件构造函数只使用命名参数。这使得实例创建表达式更容易阅读。
您可以用@required在任何Dart代码中修饰指定参数(不仅仅是Flutter),以表明它是必需的参数。例如:

const Scrollbar({Key key, @required Widget child})

在构建滚Scrollbar 时,当child参数缺少时,解析器在报告一个警告。

可选的位置参数

用 [] 中封装一组函数参数,Dart将其标记为可选的位置参数:

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

下面是一个没有可选参数调用这个函数的例子:

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

这里有一个用第三个参数调用这个函数的例子:

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

默认参数值

函数可以使用=为命名参数和位置参数定义默认值。默认值必须是编译时常量。如果没有提供默认值,则默认值为null。

下面是一个为命名参数设置默认值的例子:

/// 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);

弃用提醒:旧代码可能使用冒号(:)而不是=来设置命名参数的默认值。原因是,最初只支持命名参数。这种支持很可能不支持,因此我们建议您使用=来指定默认值。

下一个例子展示了如何为位置参数设置默认值:

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;
}

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

你还可以将Lists和Maps作为默认值传递,下面的示例定义了一个函数doStuff(),它为list参数指定了默认List,为gift参数指定了默认map。

void doStuff(
    {List<int> list = const [1, 2, 3],Map<String, String> gifts = const {
      'first': 'paper',
      'second': 'cotton',
      'third': 'leather'
    }}) {
  print('list:  $list');
  print('gifts: $gifts');
}

Main()函数

每个应用程序都必须有一个 顶级 main()函数,作为应用程序的入口点。main()函数返回void,并有一个可选的List<String>参数。
下面是一个web应用程序的main()函数示例:

void main() {
  querySelector('#sample_text_id')
    ..text = 'Click me!'
    ..onClick.listen(reverseText);
}

注意: .. 前面代码中的语法称为级联。使用级联,您可以对单个对象的成员执行多个操作。

下面是一个命令行应用程序的main()函数示例:

// Run the app like this: dart 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库来定义和解析命令行参数。
作为一级对象的函数
您可以将函数作为参数传递给另一个函数。例如:

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

var list = [1, 2, 3];

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

还可以将函数赋给变量,例如:

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

这个例子使用了一个匿名函数。更多关于这些在下一节。

匿名函数

大多数函数都被命名为main()或printElement()。您还可以创建一个称为匿名函数的无名函数,或者有时是lambda或闭包。您可以将匿名函数分配给变量,例如,您可以从集合中添加或删除它。
匿名函数类似于命名函数—0或多个参数,在括号之间用逗号和可选类型注释分隔。
下面的代码块包含函数的主体:
([[Type] param1[, …]]) {
codeBlock;
};
下面的示例定义了一个匿名函数,该函数带有一个未定义类型参数item。为列表中的每个项调用的函数将输出一个字符串,该字符串包含指定索引处的值。

var list = ['apples', 'bananas', 'oranges'];
list.forEach((item) {
  print('${list.indexOf(item)}: $item');
});

如果函数只包含一条语句,那么可以使用胖箭头符号缩短它。

list.forEach(
    (item) => print('${list.indexOf(item)}: $item'));

作用域

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);
    }
  }
}

闭包是一个函数对象,它可以访问词法作用域内的变量,即使函数在其原始作用域之外使用。
函数可以关闭周围范围中定义的变量。在下面的示例中,makeAdder()捕获变量addBy。无论返回的函数去哪里,它都会记住addBy。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,992评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,212评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,535评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,197评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,310评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,383评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,409评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,191评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,621评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,910评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,084评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,763评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,403评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,083评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,318评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,946评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,967评论 2 351

推荐阅读更多精彩内容

  • 目录 参考资料 语言特性 关键字 变量与常量 数据类型 运算符 operators 控制流程语句 异常 Excep...
    GuoDongW阅读 185,036评论 22 341
  • 这是16年5月份编辑的一份比较杂乱适合自己观看的学习记录文档,今天18年5月份再次想写文章,发现简书还为我保存起的...
    Jenaral阅读 2,743评论 2 9
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,644评论 18 139
  • 2017.10.12 星期四 阴 亲子日记(167) 今晚接儿子回家,儿子急匆匆的去做作业了,直到七点半,我们才去...
    于泽妈妈阅读 167评论 0 2
  • 这些天,忙的一直没有来得及准备去云南的行李,期待已久的,大理丽江古镇。因为上半年没有怎么旅游,发现自己的确需要一场...
    小璇子88阅读 152评论 0 0