dart 語(yǔ)言

轉(zhuǎn)自 https://book.flutterchina.club/chapter1/dart.html(去那看 我只是想存一個(gè)備份)
Dart語(yǔ)言簡(jiǎn)介

在之前我們已經(jīng)介紹過(guò)Dart語(yǔ)言的相關(guān)特性男窟,讀者可以翻看一下柱衔,如果你熟悉Dart語(yǔ)法,可以跳過(guò)本節(jié)拓哟,如果你還不了解Dart砚婆,不用擔(dān)心狠鸳,按照筆者經(jīng)驗(yàn)傍妒,如果你有過(guò)其他編程語(yǔ)言經(jīng)驗(yàn)蛉签,尤其是Java和JavaScript的話胡陪,所以,如果你是前端或Android開發(fā)者碍舍,那么將會(huì)非常容易上手Dart柠座。當(dāng)然,如果你是iOS開發(fā)者片橡,也不用擔(dān)心妈经,dart中也有一些與swift比較相似的特性,如命名參數(shù)等捧书,筆者當(dāng)時(shí)學(xué)習(xí)Dart時(shí)吹泡,只是花了一個(gè)小時(shí),看完Dart官網(wǎng)的Language Tour经瓷,就開始動(dòng)手寫Fluttter了爆哑。

在筆者看來(lái),Dart的設(shè)計(jì)目標(biāo)應(yīng)該是既對(duì)標(biāo)Java舆吮,也對(duì)標(biāo)JavaScript揭朝,Dart在靜態(tài)語(yǔ)法方面和Java非常相似,如類型定義歪泳、函數(shù)聲明萝勤、泛型等,而在動(dòng)態(tài)特性方面又和JavaScript很像呐伞,如函數(shù)式特性敌卓、異步支持等。除了融合Java和JavaScript語(yǔ)言之所長(zhǎng)之外伶氢,Dart也具有一些其它具有表現(xiàn)力的語(yǔ)法趟径,如可選命名參數(shù)、..(級(jí)聯(lián)運(yùn)算符)和?.(條件成員訪問(wèn)運(yùn)算符)以及??(判空賦值運(yùn)算符)癣防。其實(shí)蜗巧,對(duì)編程語(yǔ)言了解比較多的讀者會(huì)發(fā)現(xiàn),在Dart中其實(shí)看到的不僅有Java和JavaScript的影子蕾盯,它還具有其它編程語(yǔ)言中的身影幕屹,如命名參數(shù)在Objective-C和Swift中早就很普遍,而??操作符在Php 7.0語(yǔ)法中就已經(jīng)存在了,因此我們可以看到Google對(duì)Dart語(yǔ)言給予厚望望拖,是想把Dart打造成一門集百家之所長(zhǎng)的編程語(yǔ)言渺尘。

接下來(lái),我們先對(duì)Dart語(yǔ)法做一個(gè)簡(jiǎn)單的介紹说敏,然后再將Dart與JavaScript和Java做一個(gè)簡(jiǎn)要的對(duì)比鸥跟,方便讀者更好的理解。

注意:由于本書并非專門介紹Dart語(yǔ)言的書籍盔沫,所以本章主要會(huì)介紹一下在Flutter開發(fā)中常用的語(yǔ)法特性医咨,如果想更多了解Dart,讀者可以去Dart官網(wǎng)學(xué)習(xí)架诞,現(xiàn)在互聯(lián)網(wǎng)上Dart相關(guān)資料已經(jīng)很多了拟淮。另外Dart 2.0已經(jīng)正式發(fā)布,所以本書所有示例均采用Dart 2.0語(yǔ)法侈贷。

變量聲明

var

類似于JavaScript中的var惩歉,它可以接收任何類型的變量等脂,但最大的不同是Dart中var變量一旦賦值俏蛮,類型便會(huì)確定,則不能再改變其類型上遥,如:

var t;
t="hi world";
// 下面代碼在dart中會(huì)報(bào)錯(cuò)搏屑,因?yàn)樽兞縯的類型已經(jīng)確定為String,
// 類型一旦確定后則不能再更改其類型粉楚。
t=1000;
上面的代碼在JavaScript是沒(méi)有問(wèn)題的辣恋,前端開發(fā)者需要注意一下,之所以有此差異是因?yàn)镈art本身是一個(gè)強(qiáng)類型語(yǔ)言模软,任何變量都是有確定類型的伟骨,在Dart中,當(dāng)用var聲明一個(gè)變量后燃异,Dart在編譯時(shí)會(huì)根據(jù)第一次賦值數(shù)據(jù)的類型來(lái)推斷其類型携狭,編譯結(jié)束后其類型就已經(jīng)被確定,而JavaScript是純粹的弱類型腳本語(yǔ)言回俐,var只是變量的聲明方式而已逛腿。

dynamic和Object

Dynamic和Object 與 var功能相似,都會(huì)在賦值時(shí)自動(dòng)進(jìn)行類型推斷仅颇,不同在于单默,賦值后可以改變其類型,如:

dynamic t;
t="hi world";
//下面代碼沒(méi)有問(wèn)題
t=1000;
Object 是dart所有對(duì)象的根基類忘瓦,也就是說(shuō)所有類型都是Object的子類搁廓,所以任何類型的數(shù)據(jù)都可以賦值給Object聲明的對(duì)象,所以表現(xiàn)效果和dynamic相似。

final和const

如果您從未打算更改一個(gè)變量境蜕,那么使用 final 或 const线欲,不是var,也不是一個(gè)類型汽摹。 一個(gè) final 變量只能被設(shè)置一次李丰,兩者區(qū)別在于:const 變量是一個(gè)編譯時(shí)常量,final變量在第一次使用時(shí)被初始化逼泣。被final或者const修飾的變量趴泌,變量類型可以省略,如:

//可以省略String這個(gè)類型聲明
final str = "hi world";
//final String str = "hi world";
const str1 = "hi world";
//const String str1 = "hi world";
函數(shù)

Dart是一種真正的面向?qū)ο蟮恼Z(yǔ)言拉庶,所以即使是函數(shù)也是對(duì)象嗜憔,并且有一個(gè)類型Function。這意味著函數(shù)可以賦值給變量或作為參數(shù)傳遞給其他函數(shù)氏仗,這是函數(shù)式編程的典型特征吉捶。

函數(shù)聲明

bool isNoble(int atomicNumber) {
return _nobleGases[atomicNumber] != null;
}
dart函數(shù)聲明如果沒(méi)有顯式聲明返回值類型時(shí)會(huì)默認(rèn)當(dāng)做dynamic處理,注意皆尔,函數(shù)返回值沒(méi)有類型推斷:

typedef bool CALLBACK();

//不指定返回類型呐舔,此時(shí)默認(rèn)為dynamic,不是bool
isNoble(int atomicNumber) {
return _nobleGases[atomicNumber] != null;
}

void test(CALLBACK cb){
print(cb());
}
//報(bào)錯(cuò)慷蠕,isNoble不是bool類型
test(isNoble);
對(duì)于只包含一個(gè)表達(dá)式的函數(shù)珊拼,可以使用簡(jiǎn)寫語(yǔ)法

bool isNoble (int atomicNumber )=> _nobleGases [ atomicNumber ] != null ;
函數(shù)作為變量

var say= (str){
print(str);
};
say("hi world");
函數(shù)作為參數(shù)傳遞

void execute(var callback){
callback();
}
execute(()=>print("xxx"))
可選的位置參數(shù)

包裝一組函數(shù)參數(shù)流炕,用[]標(biāo)記為可選的位置參數(shù):

String say(String from, String msg, [String device]) {
var result = 'from saysmsg';
if (device != null) {
result = 'result with adevice';
}
return result;
}
下面是一個(gè)不帶可選參數(shù)調(diào)用這個(gè)函數(shù)的例子:

say('Bob', 'Howdy'); //結(jié)果是: Bob says Howdy
下面是用第三個(gè)參數(shù)調(diào)用這個(gè)函數(shù)的例子:

say('Bob', 'Howdy', 'smoke signal'); //結(jié)果是:Bob says Howdy with a smoke signal
可選的命名參數(shù)

定義函數(shù)時(shí)澎现,使用{param1, param2, …},用于指定命名參數(shù)每辟。例如:

//設(shè)置[bold]和[hidden]標(biāo)志
void enableFlags({bool bold, bool hidden}) {
// ...
}
調(diào)用函數(shù)時(shí)剑辫,可以使用指定命名參數(shù)。例如:paramName: value

enableFlags(bold: true, hidden: false);
可選命名參數(shù)在Flutter中使用非常多渠欺。

異步支持

Dart類庫(kù)有非常多的返回Future或者Stream對(duì)象的函數(shù)妹蔽。 這些函數(shù)被稱為異步函數(shù):它們只會(huì)在設(shè)置好一些耗時(shí)操作之后返回,比如像 IO操作峻堰。而不是等到這個(gè)操作完成讹开。

async和await關(guān)鍵詞支持了異步編程,運(yùn)行您寫出和同步代碼很像的異步代碼捐名。

Future

Future與JavaScript中的Promise非常相似旦万,表示一個(gè)異步操作的最終完成(或失敗)及其結(jié)果值的表示镶蹋。簡(jiǎn)單來(lái)說(shuō)成艘,它就是用于處理異步操作的赏半,異步處理成功了就執(zhí)行成功的操作,異步處理失敗了就捕獲錯(cuò)誤或者停止后續(xù)操作淆两。一個(gè)Future只會(huì)對(duì)應(yīng)一個(gè)結(jié)果断箫,要么成功,要么失敗秋冰。

由于本身功能較多仲义,這里我們只介紹其常用的API及特性。還有剑勾,請(qǐng)記住埃撵,F(xiàn)uture 的所有API的返回值仍然是一個(gè)Future對(duì)象,所以可以很方便的進(jìn)行鏈?zhǔn)秸{(diào)用虽另。

Future.then

為了方便示例暂刘,在本例中我們使用Future.delayed 創(chuàng)建了一個(gè)延時(shí)任務(wù)(實(shí)際場(chǎng)景會(huì)是一個(gè)真正的耗時(shí)任務(wù),比如一次網(wǎng)絡(luò)請(qǐng)求)捂刺,即2秒后返回結(jié)果字符串"hi world!"谣拣,然后我們?cè)趖hen中接收異步結(jié)果并打印結(jié)果,代碼如下:

Future.delayed(new Duration(seconds: 2),(){
return "hi world!";
}).then((data){
print(data);
});
Future.catchError

如果異步任務(wù)發(fā)生錯(cuò)誤族展,我們可以在catchError中捕獲錯(cuò)誤森缠,我們將上面示例改為:

Future.delayed(new Duration(seconds: 2),(){
//return "hi world!";
throw AssertionError("Error");
}).then((data){
//執(zhí)行成功會(huì)走到這里
print("success");
}).catchError((e){
//執(zhí)行失敗會(huì)走到這里
print(e);
});
在本示例中,我們?cè)诋惒饺蝿?wù)中拋出了一個(gè)異常苛谷,then的回調(diào)函數(shù)將不會(huì)被執(zhí)行辅鲸,取而代之的是 catchError回調(diào)函數(shù)將被調(diào)用格郁;但是腹殿,并不是只有 catchError回調(diào)才能捕獲錯(cuò)誤,then方法還有一個(gè)可選參數(shù)onError例书,我們也可以它來(lái)捕獲異常:

Future.delayed(new Duration(seconds: 2), () {
//return "hi world!";
throw AssertionError("Error");
}).then((data) {
print("success");
}, onError: (e) {
print(e);
});
Future.whenComplete

有些時(shí)候锣尉,我們會(huì)遇到無(wú)論異步任務(wù)執(zhí)行成功或失敗都需要做一些事的場(chǎng)景,比如在網(wǎng)絡(luò)請(qǐng)求前彈出加載對(duì)話框决采,在請(qǐng)求結(jié)束后關(guān)閉對(duì)話框自沧。這種場(chǎng)景,有兩種方法树瞭,第一種是分別在then或catch中關(guān)閉一下對(duì)話框拇厢,第二種就是使用Future的whenComplete回調(diào),我們將上面示例改一下:

Future.delayed(new Duration(seconds: 2),(){
//return "hi world!";
throw AssertionError("Error");
}).then((data){
//執(zhí)行成功會(huì)走到這里
print(data);
}).catchError((e){
//執(zhí)行失敗會(huì)走到這里
print(e);
}).whenComplete((){
//無(wú)論成功或失敗都會(huì)走到這里
});
Future.wait

有些時(shí)候晒喷,我們需要等待多個(gè)異步任務(wù)都執(zhí)行結(jié)束后才進(jìn)行一些操作孝偎,比如我們有一個(gè)界面,需要先分別從兩個(gè)網(wǎng)絡(luò)接口獲取數(shù)據(jù)凉敲,獲取成功后衣盾,我們需要將兩個(gè)接口數(shù)據(jù)進(jìn)行特定的處理后再顯示到UI界面上寺旺,應(yīng)該怎么做?答案是Future.wait势决,它接受一個(gè)Future數(shù)組參數(shù)阻塑,只有數(shù)組中所有Future都執(zhí)行成功后,才會(huì)觸發(fā)then的成功回調(diào)果复,只要有一個(gè)Future執(zhí)行失敗陈莽,就會(huì)觸發(fā)錯(cuò)誤回調(diào)。下面虽抄,我們通過(guò)模擬Future.delayed 來(lái)模擬兩個(gè)數(shù)據(jù)獲取的異步任務(wù)传透,等兩個(gè)異步任務(wù)都執(zhí)行成功時(shí),將兩個(gè)異步任務(wù)的結(jié)果拼接打印出來(lái)极颓,代碼如下:

Future.wait([
// 2秒后返回結(jié)果
Future.delayed(new Duration(seconds: 2), () {
return "hello";
}),
// 4秒后返回結(jié)果
Future.delayed(new Duration(seconds: 4), () {
return " world";
})
]).then((results){
print(results[0]+results[1]);
}).catchError((e){
print(e);
});
執(zhí)行上面代碼朱盐,4秒后你會(huì)在控制臺(tái)中看到“hello world”。

Async/await

Dart中的async/await 和JavaScript中的async/await功能和用法是一模一樣的菠隆,如果你已經(jīng)了解JavaScript中的async/await的用法兵琳,可以直接跳過(guò)本節(jié)。

回調(diào)地獄(Callback hell)

如果代碼中有大量異步邏輯骇径,并且出現(xiàn)大量異步任務(wù)依賴其它異步任務(wù)的結(jié)果時(shí)躯肌,必然會(huì)出現(xiàn)Future.then回調(diào)中套回調(diào)情況。舉個(gè)例子破衔,比如現(xiàn)在有個(gè)需求場(chǎng)景是用戶先登錄清女,登錄成功后會(huì)獲得用戶Id,然后通過(guò)用戶Id晰筛,再去請(qǐng)求用戶個(gè)人信息嫡丙,獲取到用戶個(gè)人信息后,為了使用方便读第,我們需要將其緩存在本地文件系統(tǒng)曙博,代碼如下:

//先分別定義各個(gè)異步任務(wù)
Future<String> login(String userName, String pwd){
...
//用戶登錄
};
Future<String> getUserInfo(String id){
...
//獲取用戶信息
};
Future saveUserInfo(String userInfo){
...
// 保存用戶信息
};
接下來(lái),執(zhí)行整個(gè)任務(wù)流:

login("alice","******").then((id){
//登錄成功后通過(guò)怜瞒,id獲取用戶信息
getUserInfo(id).then((userInfo){
//獲取用戶信息后保存
saveUserInfo(userInfo).then((){
//保存用戶信息父泳,接下來(lái)執(zhí)行其它操作
...
});
});
})
可以感受一下,如果業(yè)務(wù)邏輯中有大量異步依賴的情況吴汪,將會(huì)出現(xiàn)上面這種在回調(diào)里面套回調(diào)的情況惠窄,過(guò)多的嵌套會(huì)導(dǎo)致的代碼可讀性下降以及出錯(cuò)率提高,并且非常難維護(hù)漾橙,這個(gè)問(wèn)題被形象的稱為回調(diào)地獄(Callback hell)杆融。回調(diào)地獄問(wèn)題在之前JavaScript中非常突出近刘,也是JavaScript被吐槽最多的點(diǎn)擒贸,但隨著ECMAScript6和ECMAScript7標(biāo)準(zhǔn)發(fā)布后臀晃,這個(gè)問(wèn)題得到了非常好的解決,而解決回調(diào)地獄的兩大神器正是ECMAScript6引入了Promise介劫,以及ECMAScript7中引入的async/await徽惋。 而在Dart中幾乎是完全平移了JavaScript中的這兩者:Future相當(dāng)于Promise,而async/await連名字都沒(méi)改座韵。接下來(lái)我們看看通過(guò)Future和async/await如何消除上面示例中的嵌套問(wèn)題险绘。

使用Future消除callback hell

login("alice","******").then((id){
return getUserInfo(id);
}).then((userInfo){
return saveUserInfo(userInfo);
}).then((e){
//執(zhí)行接下來(lái)的操作
}).catchError((e){
//錯(cuò)誤處理
print(e);
});
正如上文所述, “Future 的所有API的返回值仍然是一個(gè)Future對(duì)象誉碴,所以可以很方便的進(jìn)行鏈?zhǔn)秸{(diào)用” 宦棺,如果在then中返回的是一個(gè)Future的話,該future會(huì)執(zhí)行黔帕,執(zhí)行結(jié)束后會(huì)觸發(fā)后面的then回調(diào)代咸,這樣依次向下,就避免了層層嵌套成黄。

使用async/await消除callback hell

通過(guò)Future回調(diào)中再返回Future的方式雖然能避免層層嵌套呐芥,但是還是有一層回調(diào),有沒(méi)有一種方式能夠讓我們可以像寫同步代碼那樣來(lái)執(zhí)行異步任務(wù)而不使用回調(diào)的方式奋岁?答案是肯定的思瘟,這就要使用async/await了,下面我們先直接看代碼闻伶,然后再解釋滨攻,代碼如下:

task() async {
try{
String id = await login("alice","******");
String userInfo = await getUserInfo(id);
await saveUserInfo(userInfo);
//執(zhí)行接下來(lái)的操作
} catch(e){
//錯(cuò)誤處理
print(e);
}
}
async用來(lái)表示函數(shù)是異步的,定義的函數(shù)會(huì)返回一個(gè)Future對(duì)象蓝翰,可以使用then方法添加回調(diào)函數(shù)光绕。
await 后面是一個(gè)Future,表示等待該異步任務(wù)完成霎箍,異步完成后才會(huì)往下走奇钞;await必須出現(xiàn)在 async 函數(shù)內(nèi)部。
可以看到漂坏,我們通過(guò)async/await將一個(gè)異步流用同步的代碼表示出來(lái)了。

其實(shí)媒至,無(wú)論是在JavaScript還是Dart中顶别,async/await都只是一個(gè)語(yǔ)法糖,編譯器或解釋器最終都會(huì)將其轉(zhuǎn)化為一個(gè)Promise(Future)的調(diào)用鏈拒啰。

Stream

Stream 也是用于接收異步事件數(shù)據(jù)驯绎,和Future 不同的是,它可以接收多個(gè)異步操作的結(jié)果(成功或失斈钡)剩失。 也就是說(shuō)屈尼,在執(zhí)行異步任務(wù)時(shí),可以通過(guò)多次觸發(fā)成功或失敗事件來(lái)傳遞結(jié)果數(shù)據(jù)或錯(cuò)誤異常拴孤。 Stream 常用于會(huì)多次讀取數(shù)據(jù)的異步任務(wù)場(chǎng)景脾歧,如網(wǎng)絡(luò)內(nèi)容下載、文件讀寫等演熟。舉個(gè)例子:

Stream.fromFutures([
// 1秒后返回結(jié)果
Future.delayed(new Duration(seconds: 1), () {
return "hello 1";
}),
// 拋出一個(gè)異常
Future.delayed(new Duration(seconds: 2),(){
throw AssertionError("Error");
}),
// 3秒后返回結(jié)果
Future.delayed(new Duration(seconds: 3), () {
return "hello 3";
})
]).listen((data){
print(data);
}, onError: (e){
print(e.message);
},onDone: (){

});
上面的代碼依次會(huì)輸出:

I/flutter (17666): hello 1
I/flutter (17666): Error
I/flutter (17666): hello 3
代碼很簡(jiǎn)單鞭执,就不贅述了。

思考題:既然Stream可以接收多次事件芒粹,那能不能用Stream來(lái)實(shí)現(xiàn)一個(gè)訂閱者模式的事件總線兄纺?

總結(jié)

通過(guò)上面介紹,相信你對(duì)Dart應(yīng)該有了一個(gè)初步的印象化漆,由于筆者平時(shí)也使用Java和JavaScript估脆,下面筆者根據(jù)自己的經(jīng)驗(yàn),結(jié)合Java和JavaScript座云,談一下自己的看法旁蔼。

之所以將Dart與Java和JavaScript對(duì)比,是因?yàn)楦斫蹋@兩者分別是強(qiáng)類型語(yǔ)言和弱類型語(yǔ)言的典型代表棺聊,并且Dart 語(yǔ)法中很多地方也都借鑒了Java和JavaScript佳励。

Dart vs Java

客觀的來(lái)講剩檀,Dart在語(yǔ)法層面確實(shí)比Java更有表現(xiàn)力;在VM層面塑悼,Dart VM在內(nèi)存回收和吞吐量都進(jìn)行了反復(fù)的優(yōu)化裸弦,但具體的性能對(duì)比祟同,筆者沒(méi)有找到相關(guān)測(cè)試數(shù)據(jù),但在筆者看來(lái)理疙,只要Dart語(yǔ)言能流行晕城,VM的性能就不用擔(dān)心,畢竟Google在go(沒(méi)用vm但有GC)窖贤、javascript(v8)砖顷、dalvik(android上的java vm)上已經(jīng)有了很多技術(shù)積淀。值得注意的是Dart在Flutter中已經(jīng)可以將GC做到10ms以內(nèi)赃梧,所以Dart和Java相比滤蝠,決勝因素并不會(huì)是在性能方面。而在語(yǔ)法層面授嘀,Dart要比java更有表現(xiàn)力物咳,最重要的是Dart對(duì)函數(shù)式編程支持要遠(yuǎn)強(qiáng)于Java(目前只停留在lamda表達(dá)式),而Dart目前真正的不足是生態(tài)蹄皱,但筆者相信览闰,隨著Flutter的逐漸火熱芯肤,會(huì)回過(guò)頭來(lái)反推Dart生態(tài)加速發(fā)展,對(duì)于Dart來(lái)說(shuō)压鉴,現(xiàn)在需要的是時(shí)間崖咨。

Dart vs JavaScript

JavaScript的弱類型一直被抓短,所以TypeScript晴弃、Coffeescript甚至是Facebook的flow(雖然并不能算JavaScript的一個(gè)超集掩幢,但也通過(guò)標(biāo)注和打包工具提供了靜態(tài)類型檢查)才有市場(chǎng)。就筆者使用過(guò)的腳本語(yǔ)言中(筆者曾使用過(guò)Python上鞠、PHP)际邻,JavaScript無(wú)疑是動(dòng)態(tài)化支持最好的腳本語(yǔ)言,比如在JavaScript中芍阎,可以給任何對(duì)象在任何時(shí)候動(dòng)態(tài)擴(kuò)展屬性世曾,對(duì)于精通JavaScript的高手來(lái)說(shuō),這無(wú)疑是一把利劍谴咸。但是轮听,任何事物都有兩面性,JavaScript的強(qiáng)大的動(dòng)態(tài)化特性也是把雙刃劍岭佳,你可經(jīng)常聽到另一個(gè)聲音血巍,認(rèn)為JavaScript的這種動(dòng)態(tài)性糟糕透了,太過(guò)靈活反而導(dǎo)致代碼很難預(yù)期珊随,無(wú)法限制不被期望的修改述寡。畢竟有些人總是對(duì)自己或別人寫的代碼不放心,他們希望能夠讓代碼變得可控叶洞,并期望有一套靜態(tài)類型檢查系統(tǒng)來(lái)幫助自己減少錯(cuò)誤鲫凶。正因如此,在Flutter中衩辟,Dart幾乎放棄了腳本語(yǔ)言動(dòng)態(tài)化的特性螟炫,如不支持反射、也不支持動(dòng)態(tài)創(chuàng)建函數(shù)等艺晴。并且Dart在2.0強(qiáng)制開啟了類型檢查(Strong Mode)昼钻,原先的檢查模式(checked mode)和可選類型(optional type)將淡出,所以在類型安全這個(gè)層面來(lái)說(shuō)财饥,Dart和TypeScript换吧、Coffeescript是差不多的,所以單從這一點(diǎn)來(lái)看钥星,Dart并不具備什么明顯優(yōu)勢(shì),但綜合起來(lái)看满着,dart既能進(jìn)行服務(wù)端腳本谦炒、APP開發(fā)贯莺、web開發(fā),這就有優(yōu)勢(shì)了宁改!

綜上所述缕探,筆者還是很看好Dart語(yǔ)言的將來(lái),之所以表這個(gè)態(tài)还蹲,是因?yàn)樵谛录夹g(shù)發(fā)展初期爹耗,很多人可能還有所搖擺,有所猶豫谜喊,所以有必要給大家打一劑強(qiáng)心針潭兽,當(dāng)然,這是一個(gè)見仁見智的問(wèn)題斗遏,大家可以各抒己見山卦。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市诵次,隨后出現(xiàn)的幾起案子账蓉,更是在濱河造成了極大的恐慌,老刑警劉巖逾一,帶你破解...
    沈念sama閱讀 219,427評(píng)論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件铸本,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡遵堵,警方通過(guò)查閱死者的電腦和手機(jī)箱玷,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,551評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)鄙早,“玉大人汪茧,你說(shuō)我怎么就攤上這事∠薹” “怎么了舱污?”我有些...
    開封第一講書人閱讀 165,747評(píng)論 0 356
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)弥虐。 經(jīng)常有香客問(wèn)我扩灯,道長(zhǎng),這世上最難降的妖魔是什么霜瘪? 我笑而不...
    開封第一講書人閱讀 58,939評(píng)論 1 295
  • 正文 為了忘掉前任珠插,我火速辦了婚禮,結(jié)果婚禮上颖对,老公的妹妹穿的比我還像新娘捻撑。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,955評(píng)論 6 392
  • 文/花漫 我一把揭開白布顾患。 她就那樣靜靜地躺著番捂,像睡著了一般。 火紅的嫁衣襯著肌膚如雪江解。 梳的紋絲不亂的頭發(fā)上设预,一...
    開封第一講書人閱讀 51,737評(píng)論 1 305
  • 那天,我揣著相機(jī)與錄音犁河,去河邊找鬼鳖枕。 笑死,一個(gè)胖子當(dāng)著我的面吹牛桨螺,可吹牛的內(nèi)容都是我干的宾符。 我是一名探鬼主播,決...
    沈念sama閱讀 40,448評(píng)論 3 420
  • 文/蒼蘭香墨 我猛地睜開眼彭谁,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼吸奴!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起缠局,我...
    開封第一講書人閱讀 39,352評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤则奥,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后狭园,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體读处,經(jīng)...
    沈念sama閱讀 45,834評(píng)論 1 317
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,992評(píng)論 3 338
  • 正文 我和宋清朗相戀三年唱矛,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了罚舱。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 40,133評(píng)論 1 351
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡绎谦,死狀恐怖管闷,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情窃肠,我是刑警寧澤包个,帶...
    沈念sama閱讀 35,815評(píng)論 5 346
  • 正文 年R本政府宣布,位于F島的核電站冤留,受9級(jí)特大地震影響碧囊,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜纤怒,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,477評(píng)論 3 331
  • 文/蒙蒙 一糯而、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧泊窘,春花似錦熄驼、人聲如沸像寒。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,022評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)萝映。三九已至吴叶,卻和暖如春阐虚,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背蚌卤。 一陣腳步聲響...
    開封第一講書人閱讀 33,147評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工实束, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人逊彭。 一個(gè)月前我還...
    沈念sama閱讀 48,398評(píng)論 3 373
  • 正文 我出身青樓咸灿,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親侮叮。 傳聞我的和親對(duì)象是個(gè)殘疾皇子避矢,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,077評(píng)論 2 355