和其他語(yǔ)言類(lèi)似,Dart 會(huì)忽略空白。但是,人們卻不會(huì)這樣。在代碼中加入空格調(diào)整樣式后可以讓人們看到內(nèi)容和編譯器看到的類(lèi)似。
不要使用 Tab 縮進(jìn)。
使用空白符來(lái)格式化代碼可以保證每個(gè)人在編輯器中看到的內(nèi)容是一樣的。這也會(huì)和傳送到博客上的樣式一樣,或者是一些代碼站點(diǎn),比如 Github。
現(xiàn)在的編輯器在縮進(jìn)的時(shí)候可以模擬 tab 鍵來(lái)進(jìn)行縮進(jìn),以讓你在編寫(xiě)代碼的時(shí)候更加輕松,并且可以保證代碼的一致性。
你應(yīng)該避免在編寫(xiě)代碼時(shí)一行的長(zhǎng)度超過(guò)八十個(gè)字符。
可讀性研究表明,過(guò)長(zhǎng)的文本不適合閱讀,因?yàn)楫?dāng)你看到下一行的時(shí)候眼睛移動(dòng)的距離過(guò)大。這就是為什么新聞以及雜志的文本都是多行的。
如果你在編碼時(shí)希望一行的代碼可以超過(guò)八十個(gè)字符,那么對(duì)于閱讀者而言你的代碼過(guò)于冗長(zhǎng)并且有點(diǎn)太緊湊了。你真的希望調(diào)用一個(gè)名為 AbstractWidgetFactoryManagerBuilder
的函數(shù)嗎?
應(yīng)該將二元操作符放在多行表達(dá)式的前一行。
各種樣式的參數(shù)都是有效的,但是大多數(shù)情況下代碼看起來(lái)都是下面這個(gè)樣子的,并且這樣更容易保證代碼的一致性。
// good
if (isDeepFried ||
(hasPieCrust && !vegan) ||
containsBacon) {
print('Bob likes it.');
}
注意這也可以直接使用 =>
:
// good
bobLikes() =>
isDeepFried || (hasPieCrust && !vegan) || containsBacon;
// bad
bobLikes()
=> isDeepFried || (hasPieCrust && !vegan) || containsBacon;
應(yīng)該將三元運(yùn)算符放在多行表達(dá)式的下一行。
同樣的,如果你在操作符之前換行了,那么全部操作符之前都要換行。
// good
return someCondition
? whenTrue
: whenFalse;
// bad
return someCondition ?
whenTrue :
whenFalse;
return someCondition
? whenTrue : whenFalse;
在多行表達(dá)式中應(yīng)該在下一行加入 .
操作符。
這條規(guī)則優(yōu)先級(jí)高于上一條。可其他操作符不同,如果你使用 .
來(lái)分隔一個(gè)表達(dá)式,那就應(yīng)該把它放在第二行的開(kāi)頭。
// good
someVeryLongVariable.withAVeryLongProperty
.aMethodOnThatObject();
在代碼塊中應(yīng)該縮進(jìn)兩格。
// good
if (condition) {
print('hi');
}
如果代碼過(guò)長(zhǎng)需要換行,那么下一行應(yīng)該縮進(jìn)四格。
// good
someLongObject.aReallyLongMethodName(longArg, anotherLongArg,
wrappedToNextLine);
你也可以按照你的喜好來(lái)進(jìn)行縮進(jìn):
// good
someLongObject.aReallyLongMethodName(longArg, anotherLongArg,
wrappedToNextLine);
注意使用 =>
也是可以的:
// good
bobLikes() =>
isDeepFried || (hasPieCrust && !vegan) || containsBacon;
// bad
bobLikes() =>
isDeepFried || (hasPieCrust && !vegan) || containsBacon;
如果某一行的上一行是一個(gè)函數(shù)表達(dá)式,那么該行不應(yīng)該縮進(jìn)。
上面這個(gè)規(guī)則的一個(gè)例外就是函數(shù)表達(dá)式嵌套在另一個(gè)表達(dá)式中,比如是作為參數(shù)傳遞給一個(gè)方法。這些應(yīng)該寫(xiě)成下面的格式:
// good
new Future.delayed(const Duration(seconds: 1), () {
print('I am a callback');
});
// bad
new Future.delayed(const Duration(seconds: 1), () {
print('I am a callback');
});
花括號(hào)({)應(yīng)該放在它之前那一行的后面。
// good
class Foo {
method() {
if (true) {
// ...
} else {
// ...
}
}
}
對(duì)于全部的控制語(yǔ)句,都應(yīng)該使用括號(hào)括起來(lái)。
這樣做有助于避免 else 懸掛 的問(wèn)題。
// good
if (true) {
print('sanity');
} else {
print('opposite day!');
}
// bad
if (true) print('sanity');
else
print('opposite day!');
這種情況有個(gè)特例:?jiǎn)蝹€(gè) if
語(yǔ)句,如果沒(méi)有相應(yīng)的 else
語(yǔ)句就可以不實(shí)用括號(hào)。
// good
if (arg == null) return defaultValue;
switch 語(yǔ)句中的 case 語(yǔ)句應(yīng)該縮進(jìn)兩個(gè)空格,case 語(yǔ)句的函數(shù)體應(yīng)該縮進(jìn)四個(gè)空格。
// good
switch (fruit) {
case 'apple':
print('delish');
break;
case 'durian':
print('stinky');
break;
}
在函數(shù)、操作符或者 setter 的名稱(chēng)與其參數(shù)列表間不要加入空格。
// good
bool convertToBool(arg) { ... }
bool operator ==(other) { ... }
set contents(value) { ... }
// bad
bool convertToBool (arg) { ... }
bool operator == (other) { ... }
set contents (value) { ... }
在 operator
關(guān)鍵字之后應(yīng)該空一格。
// good
bool operator ==(other) => ...;
// bad
bool operator==(other) => ...;
二元、三元運(yùn)算符的周?chē)?strong>應(yīng)該空一格,逗號(hào)之后也應(yīng)該空一格,但是在一元運(yùn)算符周?chē)粦?yīng)該留空格。
注意 <
以及 >
用在表達(dá)式中時(shí)是作為二元操作符的,但是作為泛型的時(shí)候則不是。is
以及 !is
只能作為二元運(yùn)算符。另外,.
用于訪問(wèn)成員的時(shí)候絕對(duì)不能插入空格。
// good
a = 1 + 2 / (3 * -b);
c = !condition == a > b;
d = condition ? b : object.method(a, b, c);
if (obj is! SomeType) print('not SomeType');
// bad
a=1+2/(3* - b);
c= ! condition==a>b;
d= condition?b:object.method(a,b,c);
if (obj is !SomeType) print('not SomeType');
在循環(huán)語(yǔ)句中,in
的周?chē)约懊總€(gè) ;
之后都應(yīng)該空一格。
// good
for (var i = 0; i < 100; i++) {
// ...
}
for (final item in collection) {
// ...
}
流控制的關(guān)鍵字之后應(yīng)該空一格。
這和函數(shù)調(diào)用不同,函數(shù)調(diào)用中函數(shù)名和括號(hào)之間不應(yīng)該有空格。
// good
while (foo) {
// ...
}
try {
// ...
} catch (e) {
// ...
}
在 (
、[
、{
之前,以及 )
、]
、}
之后不應(yīng)該有空格。
同樣的,如果 <
和 >
用作泛型的話也不應(yīng)該在其中間留空格。
// good
var numbers = <int>[1, 2, (3 + 4)];
在函數(shù)體內(nèi)的 {
前應(yīng)該空一格。
上述規(guī)則有一個(gè)特殊情況。如果 {
是用在某個(gè)函數(shù)的參數(shù)列表之后,那么在 )
和它之間應(yīng)該空一格。
// good
getEmptyFn(a) {
return () {};
}
// bad
getEmptyFn(a){
return (){};
}
構(gòu)造函數(shù)中的初始化應(yīng)該保證每個(gè)字段都單獨(dú)占一行。
// good
MyClass()
: firstField = "some value",
secondField = "another",
thirdField = "last" {
// ...
}
注意 :
應(yīng)該放在函數(shù)名稱(chēng)的下一行開(kāi)頭,并且應(yīng)該縮進(jìn)四格。而字段則應(yīng)該對(duì)齊(也就是說(shuō)第一個(gè)字段總共縮進(jìn)六格)。
對(duì)于已命名的的實(shí)參和形參,在 :
之后應(yīng)該空一格。
// good
class ListBox {
bool showScrollbars;
ListBox({this.showScrollbars: false});
}
main() {
new ListBox(showScrollbars: true);
}
// bad
new ListBox(showScrollbars:true);
new ListBox(showScrollbars : true);
可選位置參數(shù)的 =
周?chē)?strong>應(yīng)該空一格。
// good
class HttpServer {
static Future<HttpServer> listen([int port = 80]) {
// ...
}
}
// bad
import 'dart:async';
class HttpServer {
static Future<HttpServer> listen([int port=80]) {
// ...
}
}
方法的級(jí)聯(lián)應(yīng)該縮進(jìn)兩格。
// good
list = new List()
..addAll([1, 2, 3])
..addAll([4, 5, 6]);
// bad
list = new List()
..addAll([1, 2, 3])
..addAll([4, 5, 6]);
以前的時(shí)候這條規(guī)則是“縮進(jìn)四格”,但是根據(jù)我們對(duì)于已有代碼的調(diào)查結(jié)果來(lái)看,級(jí)聯(lián)使用兩格縮進(jìn)更加常見(jiàn)。
多行列表以及映射的字面值的代碼如果采取多行形式,那么應(yīng)該縮進(jìn)兩行,并且把閉合的 ]
或者 }
放在下一行。
多行列表以及映射的字面值最好是在逗號(hào)之后換行。
]
或者 }
應(yīng)該根據(jù)上下文代碼來(lái)進(jìn)行縮進(jìn),以匹配上下文代碼的格式。
// good
var simpleList = [1, 2, 3];
var simpleMap = {'a': 1, 'b': 2, 'c': 3};
var fooList = [
'a',
'b',
'c'
];
var barMap = {
'a': 'b',
'c': 'd'
};
var listInsideMap = {
'a': ['b', 'c', 'd'],
'e': [
'f',
'g',
'h'
]
};
var mapInsideMap = {
'a': {'b': 'c', 'd': 'e'},
'f': {
'f': 'g',
'h': 'i'
}
};
var mapInsideList = [
{
'a': 'b',
'c': 'd'
},
{
'a': 'b',
'c': 'd'
},
];
// bad
var fooList = [
1,
2,
3
];
var fooList = [
1,
2,
3];
var fooList = [1, 2,
3, 4];
更多建議: