Dart基礎(chǔ)(二)

級(jí)別: ★☆☆☆☆
標(biāo)簽:「Flutter 」「Dart」「Dart Functions」
作者: WYW
審校: QiShare團(tuán)隊(duì)


前言:
4篇Dart基礎(chǔ)已經(jīng)全部更新完成。目錄如下:
Dart 基礎(chǔ) (一)
Dart 基礎(chǔ) (二)
Dart 基礎(chǔ) (三)
Dart 基礎(chǔ) (四)

接著上篇文章Dart基礎(chǔ)(一),我們最后聊到了方法相關(guān)的內(nèi)容,在本篇文章中,筆者仍然以方法開(kāi)頭。

Functions(方法)

下邊筆者寫(xiě)了一段代碼:定義了返回值為bool類(lèi)型,參數(shù)為整數(shù),判斷傳入?yún)?shù)是否是奇數(shù)的方法。如果是奇數(shù)輸出true,并且返回ture。否則輸出false,返回false。并且分別傳入?yún)?shù)1,2,3,4調(diào)用這個(gè)奇數(shù)方法 。

void main() {
  
  isOdd(1);
  isOdd(2);
  isOdd(3);
  isOdd(4);
  
}

bool isOdd(int num) {
  if (num % 2 == 0) {
    print('false');
    return false;
  }
  print('true');
  return true;
}

/* 輸出結(jié)果
true
false
true
false
*/

下邊筆者重寫(xiě)寫(xiě)了一下返回值為bool類(lèi)型,參數(shù)為int 類(lèi)型,判斷參數(shù)是否為奇數(shù)的方法。如果傳入?yún)?shù)是奇數(shù),則返回ture,否則返回false。 與上邊的方法的不同之處在于,這里,方法體部分只有=>及一行代碼。
=> expr 語(yǔ)法是 { return expr; }形式的縮寫(xiě)。=> 形式 有時(shí)候也稱之為 胖箭頭 語(yǔ)法。

void main() {
  
  bool flag = isOdd(1);
  print(flag);
  
  flag = isOdd(2);
  print(flag);
  
  flag = isOdd(3);
  print(flag);
  
  flag = isOdd(4);
  print(flag);
  
}

bool isOdd(int num) => (num % 2 != 0);

可選參數(shù)

分為兩種:

  1. 可選命名參數(shù);

    • 默認(rèn)參數(shù)值,可以在定義函數(shù)的時(shí)候,指定默認(rèn)參數(shù)值。
  2. 可選位置參數(shù):可以選擇性傳入某位置的參數(shù)。

這里,筆者舉一個(gè)QiShare可選說(shuō)出姓名,年齡 的示例
可以選擇性傳入姓名和年齡參數(shù)。

1. 可選命名參數(shù):{params1,param2}

在定義方法的時(shí)候,使用{param1, param2, …} 的形式來(lái)指定命名參數(shù):
調(diào)用方法的時(shí)候,可以使用這種形式 paramName: value 來(lái)指定命名參數(shù)。

下邊筆者寫(xiě)了一個(gè)返回值為空 可選位置參數(shù)為name,age,名為qiSay的方法。并且可選地傳入了參數(shù)調(diào)用了qiSay方法。


void main() {
  
  qiSay(name: 'QiShare', age: 1);
  print('\n');
  
  qiSay(name: 'QiShare');
  print('\n');
  
  qiSay( age: 1);
}

void qiSay({String name, int age}) {
  print('name:$name');
  print('age:$age');
}



/* 輸出結(jié)果:
name:QiShare
age:1


name:QiShare
age:null


name:null
age:1

*/

可以發(fā)現(xiàn)上例,不指定名字和年齡參數(shù)的情況下,輸出的參數(shù)為null。那如果我們想要給方法參數(shù)默認(rèn)值的話,需要考慮使用如下定義方法的方式。(在參數(shù)部分指定name默認(rèn)值為'QiShare'。)

  • 指定默認(rèn)參數(shù)值{param:paramValue默認(rèn)值}

下邊筆者寫(xiě)了一個(gè)返回值為空 可選位置參數(shù)為name,age,并且制定name 默認(rèn)值為QiShare的名為qiSay的方法。并且可選地傳入了參數(shù)調(diào)用了qiSay方法。在qiSay(age:1)的輸出結(jié)果可以發(fā)現(xiàn)即使不傳入可選位置參數(shù)name輸出結(jié)果中也有name的默認(rèn)值QiShare

void main() {
  
  qiSay(name: 'QiShare', age: 1);
  print('\n');
  
  qiSay(name: 'QiShare');
  print('\n');
  
  qiSay( age: 1);
}

void qiSay({String name = 'QiShare', int age}) {
  print('name:$name');
  print('age:$age');
}

/* 輸出結(jié)果:
name:QiShare
age:1


name:QiShare
age:null


name:QiShare
age:1
*/

2. 可選位置參數(shù):[param]

把一些方法的參數(shù)放到 [] 中就變成可選 位置參數(shù)了。可選位置參數(shù)的意思是,該位置的參數(shù)可以傳入,也可以不傳入。像如下代碼,可以傳入address 值,也可以不傳入address值。

這里,筆者舉一個(gè)QiShare說(shuō)出姓名,年齡,可選擇說(shuō)出住址 的示例


// 可選位置參數(shù) 地址參數(shù)為可選位置參數(shù)

void main() {
  
  qiSay('QiShare', 1, '北京');
  print('\n\n');
  qiSay('QiShare', 1);
}

void qiSay(String name, int age, [String address]) {
  
  if (name != null) {
    print('name: $name');
  }
  
  if (age != null) {
    print('age: $age');
  }
  
  if (address != null) {
    print('address:$address');
  } 
}

/* 輸出結(jié)果:
name: QiShare
age: 1
address:北京
*/

如果我們想指定qiSay方法可選位置參數(shù)address的默認(rèn)值為'BeiJing'可以通過(guò)如下方式。

  • 指定默認(rèn)參數(shù)值[param=paramsValue默認(rèn)值]
void main() {
  
  qiSay('QiShare', 1, '北京');
  print('\n\n');
  qiSay('QiShare', 1);
}

void qiSay(String name, int age, [String address = 'BeiJing']) {
  
  if (name != null) {
    print('name: $name');
  }
  
  if (age != null) {
    print('age: $age');
  }
  
  if (address != null) {
    print('address:$address');
  } 
}


/** 輸出結(jié)果:
name: QiShare
age: 1
address:北京



name: QiShare
age: 1
address:BeiJing
*/

還可以使用 list 或者 map 作為默認(rèn)值。 下面的示例定義了一個(gè)方法 doStuff(), 并分別為 list 和 gifts 參數(shù)指定了 默認(rèn)值。

void main() {
  doStuff();
}

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

/**
輸出結(jié)果:

list:  [1, 2, 3]
gifts: {first: paper, second: cotton, third: leather}

*/
  • 一等方法對(duì)象

可以定義一個(gè)參數(shù)為方法的方法A
然后可以把方法B 當(dāng)做參數(shù)傳遞給方法A。

如List 的遍歷方法forEach,接收的參數(shù)就是方法void f(E element){}

void forEach(void f(E element)) {
   for (E element in this) 
       f(element);
}
void main() {
  
  var list = [1, 2, 3];

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

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

/*
輸出結(jié)果:
1
2
3
*/

使用場(chǎng)景:一等方法對(duì)象適用于需要在外部方法內(nèi)部調(diào)用多次但是不能在外部方法外部調(diào)用。
以上述代碼為例,printElement 為一等方法對(duì)象,外部方法為forEach。printElement需要在forEach 內(nèi)部中調(diào)用多次,但是不能再forEach外調(diào)用。

// 友好性:
[mArticles enumerateObjectsUsingBlock:^(WTArticle * _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
    NSLog(@"%@", obj);
}];

在友好性方面考慮的話,用Dart和 如上Objective-C的遍歷對(duì)比。筆者自己感覺(jué)Objective-C的代碼更加友好,直接在block 中有相應(yīng)的當(dāng)前遍歷對(duì)象obj,及索引index,及控制是否停止的stop參數(shù)。

如果條件表達(dá)式結(jié)果不滿足需要,則可以使用 assert 語(yǔ)句倆打斷代碼的執(zhí)行。

那么或許我們可以使用斷言打斷程序運(yùn)行的方法,通過(guò)判斷當(dāng)前遍歷的對(duì)象是否符合要求,使用斷言決定是否要終止代碼執(zhí)行。

assert(obj != null);

assert 方法的參數(shù)可以為任何返回布爾值的表達(dá)式或者方法。 如果返回的值為 true, 斷言執(zhí)行通過(guò),執(zhí)行結(jié)束。 如果返回值為 false, 斷言執(zhí)行失敗,會(huì)拋出一個(gè)異常 AssertionError)。

斷言只在檢查模式下運(yùn)行有效,如果在生產(chǎn)模式 運(yùn)行,則斷言不會(huì)執(zhí)行。

  • 對(duì)于只有一個(gè)表達(dá)式的方法,可以選擇使用縮寫(xiě)語(yǔ)法定義。

這個(gè) => expr 語(yǔ)法是 { return expr; } 形式的縮寫(xiě)。=> 形式有時(shí)候也稱之為 胖箭頭 語(yǔ)法

void main() {
  
  bool flag = isOdd(1);
  print(flag);
  
  flag = isOdd(2);
  print(flag);
  
  flag = isOdd(3);
  print(flag);
  
  flag = isOdd(4);
  print(flag);
  
}

bool isOdd(int num) => (num % 2 != 0);

/**
輸出結(jié)果:

true
false
true
false
*/

void main() {
  var loudify = (msg) => 'Hello ${msg.toUpperCase()} !!!';
  print(loudify('QiShare'));
}

// 輸出結(jié)果:
// Hello QISHARE !!!

匿名方法

大部分方法都帶有名字,例如 main() 或者 printElement()。 我們還可以創(chuàng)建沒(méi)有名字的方法,稱之為 匿名方法,有時(shí)候也被稱為 lambda 或者 closure 閉包。 我們可以把匿名方法賦值給一個(gè)變量, 然后可以通過(guò)使用變量調(diào)用方法,比如遍歷List 中的數(shù)據(jù)。

匿名函數(shù)和命名函數(shù)看起來(lái)類(lèi)似— 在括號(hào)之間可以定義一些參數(shù),參數(shù)使用逗號(hào) 分割,也可以是可選參數(shù)。 后面大括號(hào)中的代碼為函數(shù)體:

([[Type] param1[, …]]) { 
  codeBlock; 
}; 

下邊我們看一段 foreach 遍歷的list,并且輸出對(duì)應(yīng)的obj的索引的有意思的代碼。直接使用indexOf (obj)的方式可以發(fā)現(xiàn),當(dāng)輸出第二個(gè)apples的索引的時(shí)候,發(fā)現(xiàn)輸出的index結(jié)果仍為0;
筆者看了indexOf的方法聲明后才發(fā)現(xiàn),原來(lái)indexOf有一個(gè)可選位置參數(shù)start,并且默認(rèn)值為0;
所以如果我們想要在遍歷list 的時(shí)候獲取到準(zhǔn)確地索引,可以記錄遍歷過(guò)的次數(shù),并且給start參數(shù) 傳入相應(yīng)的值。

int indexOf(E element, [int start = 0]);
  var list = ['apples', 'oranges', 'apples', 'grapes', 'bananas', 'plums'];
  list.forEach((obj){
    print('當(dāng)前遍歷項(xiàng): $obj');
    print('當(dāng)前遍歷項(xiàng)索引: ${list.indexOf(obj).toString()}');
    
  });
  
  print('');
  print('');
  
  int count = 0;
  list.forEach((obj){
    print('當(dāng)前遍歷項(xiàng): $obj');
    print('當(dāng)前遍歷項(xiàng)索引: ${list.indexOf(obj,count).toString()}');
    print('當(dāng)前遍歷項(xiàng)索引Count: $count');
    ++count;
    
    
  });
  
  print('第二個(gè)apples 索引:');
  print(list.indexOf('apples', 1));
  
  /**
  輸出結(jié)果:
  
  當(dāng)前遍歷項(xiàng): apples
當(dāng)前遍歷項(xiàng)索引: 0
當(dāng)前遍歷項(xiàng): oranges
當(dāng)前遍歷項(xiàng)索引: 1
當(dāng)前遍歷項(xiàng): apples
當(dāng)前遍歷項(xiàng)索引: 0
當(dāng)前遍歷項(xiàng): grapes
當(dāng)前遍歷項(xiàng)索引: 3
當(dāng)前遍歷項(xiàng): bananas
當(dāng)前遍歷項(xiàng)索引: 4
當(dāng)前遍歷項(xiàng): plums
當(dāng)前遍歷項(xiàng)索引: 5


當(dāng)前遍歷項(xiàng): apples
當(dāng)前遍歷項(xiàng)索引: 0
當(dāng)前遍歷項(xiàng)索引Count: 0
當(dāng)前遍歷項(xiàng): oranges
當(dāng)前遍歷項(xiàng)索引: 1
當(dāng)前遍歷項(xiàng)索引Count: 1
當(dāng)前遍歷項(xiàng): apples
當(dāng)前遍歷項(xiàng)索引: 2
當(dāng)前遍歷項(xiàng)索引Count: 2
當(dāng)前遍歷項(xiàng): grapes
當(dāng)前遍歷項(xiàng)索引: 3
當(dāng)前遍歷項(xiàng)索引Count: 3
當(dāng)前遍歷項(xiàng): bananas
當(dāng)前遍歷項(xiàng)索引: 4
當(dāng)前遍歷項(xiàng)索引Count: 4
當(dāng)前遍歷項(xiàng): plums
當(dāng)前遍歷項(xiàng)索引: 5
當(dāng)前遍歷項(xiàng)索引Count: 5
第二個(gè)apples 索引:
2

  
  */
  
void forEach(void f(E element)) {
   for (E element in this) f(element);
}

list.forEach((obj){ print('當(dāng)前遍歷項(xiàng): $obj'); print('當(dāng)前遍歷項(xiàng)索引: ${list.indexOf(obj).toString()}'); });
上述代碼紅色部分即為匿名函數(shù),就是一個(gè)沒(méi)有名字的函數(shù)。

  • Lexical scope(靜態(tài)作用域)

Dart 是靜態(tài)作用域語(yǔ)言,變量的作用域在寫(xiě)代碼的時(shí)候就確定過(guò)了。 大括號(hào)里面定義的變量就 只能在大括號(hào)里面訪問(wèn),和 Java 作用域 類(lèi)似。

void main() {
  debugPaintSizeEnabled = false;
  runApp(WebTech());

  var insideMain = true;

  myFunction() {
    var insideFunction = true;
    print('insideFunction: $insideFunction');
    
    nestedFunction() {
      var insideNestedFunction = true;

      assert(topLevel);
      assert(insideMain);
      assert(insideFunction);
      assert(insideNestedFunction);
      print('topLevel: $topLevel');
      print('insideMain: $insideMain');
      print('insideFunction: $insideFunction');
      print('insideNestedFunction: $insideNestedFunction');
    }
    nestedFunction();

    print('insideFunction:$insideFunction');

  }
  myFunction();

}

注意 nestedFunction() 可以訪問(wèn)所有的變量, 包含頂級(jí)變量。

/**
輸出結(jié)果:
flutter: insideFunction: true                                           
flutter: topLevel: true                                                 
flutter: insideMain: true                                               
flutter: insideFunction: true                                           
flutter: insideNestedFunction: true                                     
flutter: insideFunction:true  
*/


假如說(shuō)在QiShare1.dart 文件中定義了變量topLevel,那么在QiShare1.dart類(lèi)文件中任何地方都可以訪問(wèn)。

對(duì)于QiShare2.dart,如果import了QiShare1.dart。那么QiShare2.dart中topLevel也是可見(jiàn)的。

  • Lexical closures(詞法閉包)

一個(gè) 閉包 是一個(gè)方法對(duì)象,不管該對(duì)象在何處被調(diào)用, 該對(duì)象都可以訪問(wèn)其作用域內(nèi) 的變量。

方法可以封閉定義到其作用域內(nèi)的變量。 下面的示例中,makeAdder() 捕獲到了變量 addBy。 不管你在哪里執(zhí)行 makeAdder() 所返回的函數(shù),都可以使用 addBy 參數(shù)。

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

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);
}
Function makeAdder(num addBy) {
  return (num i) => addBy + i;
}

相當(dāng)于
Function makeAdder(num addBy) {
  return num f(num i) { 
    return addBy + i;
}
}

add2 = makeAdder(2);
add2(3) = f(3) + 2;
add2(3)= 5;

同理:
add3(3) = 7;

Testing functions for equality(測(cè)試函數(shù)是否相等)

下面是測(cè)試頂級(jí)方法、靜態(tài)函數(shù)和實(shí)例函數(shù) 相等的示例:

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

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

main() {
  var 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 = new A(); // Instance #1 of A
  var w = new 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);
}

筆者還沒(méi)有遇到測(cè)試函數(shù)的應(yīng)用場(chǎng)景。

  • Return values(返回值)

所有的函數(shù)都返回一個(gè)值。如果沒(méi)有指定返回值,則 默認(rèn)把語(yǔ)句 return null; 作為函數(shù)的最后一個(gè)語(yǔ)句執(zhí)行。

參考學(xué)習(xí)資料


推薦文章:
Dart基礎(chǔ)(一)
iOS 短信驗(yàn)證碼倒計(jì)時(shí)按鈕
iOS 環(huán)境變量配置
iOS 中處理定時(shí)任務(wù)的常用方法
算法小專欄:貪心算法
iOS 快速實(shí)現(xiàn)分頁(yè)界面的搭建
iOS 中的界面旋轉(zhuǎn)

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 227,702評(píng)論 6 531
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,143評(píng)論 3 415
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人,你說(shuō)我怎么就攤上這事。” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 175,553評(píng)論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我,道長(zhǎng),這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 62,620評(píng)論 1 307
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 71,416評(píng)論 6 405
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 54,940評(píng)論 1 321
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,024評(píng)論 3 440
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 42,170評(píng)論 0 287
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 48,709評(píng)論 1 333
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 40,597評(píng)論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 42,784評(píng)論 1 369
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,291評(píng)論 5 357
  • 正文 年R本政府宣布,位于F島的核電站,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,029評(píng)論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 34,407評(píng)論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 35,663評(píng)論 1 280
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 51,403評(píng)論 3 390
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 47,746評(píng)論 2 370

推薦閱讀更多精彩內(nèi)容