Title here
" + "This is a paragraph
" + "" + "摘要:強制參數(shù)和返回值注釋必須包含類型信息和說明。如果重寫的形參個數(shù)類型順序和返回值類型均未發(fā)生變化,可省略,僅用標識,否則仍應(yīng)作完整注釋。
轉(zhuǎn)載:原地址
1 前言JavaScript在百度一直有著廣泛的應(yīng)用,特別是在瀏覽器端的行為管理。本文檔的目標是使JavaScript代碼風(fēng)格保持一致,容易被理解和被維護。
雖然本文檔是針對JavaScript設(shè)計的,但是在使用各種JavaScript的預(yù)編譯語言時(如TypeScript等)時,適用的部分也應(yīng)盡量遵循本文檔的約定。
2 代碼風(fēng)格 2.1 文件解釋:
UTF-8 編碼具有更廣泛的適應(yīng)性。BOM 在使用程序或工具處理文件時可能造成不必要的干擾。
示例:
javascript// good switch (variable) { case "1": // do... break; case "2": // do... break; default: // do... } // bad switch (variable) { case "1": // do... break; case "2": // do... break; default: // do... }2.2.2 空格
示例:
javascriptvar a = !arr.length; a++; a = b + c;
示例:
javascript// good if (condition) { } while (condition) { } function funcName() { } // bad if (condition){ } while (condition){ } function funcName(){ }
示例:
javascript// good if (condition) { } while (condition) { } (function () { })(); // bad if(condition) { } while(condition) { } (function() { })();
示例:
javascript// good var obj = { a: 1, b: 2, c: 3 }; // bad var obj = { a : 1, b:2, c :3 };
示例:
javascript// good function funcName() { } var funcName = function funcName() { }; funcName(); // bad function funcName () { } var funcName = function funcName () { }; funcName ();
示例:
javascript// good callFunc(a, b); // bad callFunc(a , b) ;
示例:
javascript// good callFunc(param1, param2, param3); save(this.list[this.indexes[i]]); needIncream && (variable += increament); if (num > list.length) { } while (len--) { } // bad callFunc( param1, param2, param3 ); save( this.list[ this.indexes[ i ] ] ); needIncreament && ( variable += increament ); if ( num > list.length ) { } while ( len-- ) { }
解釋:
聲明包含元素的數(shù)組與對象,只有當內(nèi)部元素的形式較為簡單時,才允許寫在一行。元素復(fù)雜的情況,還是應(yīng)該換行書寫。
示例:
javascript// good var arr1 = []; var arr2 = [1, 2, 3]; var obj1 = {}; var obj2 = {name: "obj"}; var obj3 = { name: "obj", age: 20, sex: 1 }; // bad var arr1 = [ ]; var arr2 = [ 1, 2, 3 ]; var obj1 = { }; var obj2 = { name: "obj" }; var obj3 = {name: "obj", age: 20, sex: 1};
解釋:
超長的不可分割的代碼允許例外,比如復(fù)雜的正則表達式。長字符串不在例外之列。
示例:
javascript// good if (user.isAuthenticated() && user.isInRole("admin") && user.hasAuthority("add-admin") || user.hasAuthority("delete-admin") ) { // Code } var result = number1 + number2 + number3 + number4 + number5; // bad if (user.isAuthenticated() && user.isInRole("admin") && user.hasAuthority("add-admin") || user.hasAuthority("delete-admin")) { // Code } var result = number1 + number2 + number3 + number4 + number5;
示例:
javascript// good var obj = { a: 1, b: 2, c: 3 }; foo( aVeryVeryLongArgument, anotherVeryLongArgument, callback ); // bad var obj = { a: 1 , b: 2 , c: 3 }; foo( aVeryVeryLongArgument , anotherVeryLongArgument , callback );
示例:
javascript// 僅為按邏輯換行的示例,不代表setStyle的最優(yōu)實現(xiàn) function setStyle(element, property, value) { if (element == null) { return; } element.style[property] = value; }
示例:
javascript// 較復(fù)雜的邏輯條件組合,將每個條件獨立一行,邏輯運算符放置在行首進行分隔,或?qū)⒉糠诌壿嫲催壿嫿M合進行分隔。 // 建議最終將右括號 ) 與左大括號 { 放在獨立一行,保證與 if 內(nèi)語句塊能容易視覺辨識。 if (user.isAuthenticated() && user.isInRole("admin") && user.hasAuthority("add-admin") || user.hasAuthority("delete-admin") ) { // Code } // 按一定長度截斷字符串,并使用 + 運算符進行連接。 // 分隔字符串盡量按語義進行,如不要在一個完整的名詞中間斷開。 // 特別的,對于HTML片段的拼接,通過縮進,保持和HTML相同的結(jié)構(gòu)。 var html = "" // 此處用一個空字符串,以便整個HTML片段都在新行嚴格對齊 + "" + " "; // 也可使用數(shù)組來進行拼接,相對 + 更容易調(diào)整縮進。 var html = [ "Title here
" + "This is a paragraph
" + "" + "", " " ]; html = html.join(""); // 當參數(shù)過多時,將每個參數(shù)獨立寫在一行上,并將結(jié)束的右括號 ) 獨立一行。 // 所有參數(shù)必須增加一個縮進。 foo( aVeryVeryLongArgument, anotherVeryLongArgument, callback ); // 也可以按邏輯對參數(shù)進行組合。 // 最經(jīng)典的是baidu.format函數(shù),調(diào)用時將參數(shù)分為“模板”和“數(shù)據(jù)”兩塊 baidu.format( dateFormatTemplate, year, month, date, hour, minute, second ); // 當函數(shù)調(diào)用時,如果有一個或以上參數(shù)跨越多行,應(yīng)當每一個參數(shù)獨立一行。 // 這通常出現(xiàn)在匿名函數(shù)或者對象初始化等作為參數(shù)時,如setTimeout函數(shù)等。 setTimeout( function () { alert("hello"); }, 200 ); order.data.read( "id=" + me.model.id, function (data) { me.attchToModel(data.result); callback(); }, 300 ); // 鏈式調(diào)用較長時采用縮進進行調(diào)整。 $("#items") .find(".selected") .highlight() .end(); // 三元運算符由3部分組成,因此其換行應(yīng)當根據(jù)每個部分的長度不同,形成不同的情況。 var result = thisIsAVeryVeryLongCondition ? resultA : resultB; var result = condition ? thisIsAVeryVeryLongResult : resultB; // 數(shù)組和對象初始化的混用,嚴格按照每個對象的 { 和結(jié)束 } 在獨立一行的風(fēng)格書寫。 var array = [ { // ... }, { // ... } ];Title here
", "This is a paragraph
", "", "
示例:
javascriptif (condition) { // some statements; } else { // some statements; } try { // some statements; } catch (ex) { // some statements; }2.2.4 語句
示例:
javascript// good if (condition) { callFunc(); } // bad if (condition) callFunc(); if (condition) callFunc();
示例:
javascript// good function funcName() { } // bad function funcName() { }; // 如果是函數(shù)表達式,分號是不允許省略的。 var funcName = function () { };
解釋:
IIFE = Immediately-Invoked Function Expression.
額外的 ( 能夠讓代碼在閱讀的一開始就能判斷函數(shù)是否立即被調(diào)用,進而明白接下來代碼的用途。而不是一直拖到底部才恍然大悟。
示例:
javascript// good var task = (function () { // Code return result; })(); var func = function () { }; // bad var task = function () { // Code return result; }(); var func = (function () { });2.3 命名
示例:
javascriptvar loadingModules = {};
示例:
javascriptvar HTML_ENTITY = {};
示例:
javascriptfunction stringFormat(source) { }
示例:
javascriptfunction hear(theBells) { }
示例:
javascriptfunction TextNode(options) { }
示例:
javascriptfunction TextNode(value, engine) { this.value = value; this.engine = engine; } TextNode.prototype.clone = function () { return this; };
示例:
javascriptvar TargetState = { READING: 1, READED: 2, APPLIED: 3, READY: 4 };
示例:
javascriptequipments.heavyWeapons = {};
示例:
javascriptfunction XMLParser() { } function insertHTML(element, html) { } var httpRequest = new HTTPRequest();
示例:
javascriptfunction Engine(options) { }
示例:
javascriptfunction getStyle(element) { }
示例:
javascriptvar isReady = false; var hasMoreCommands = false;
示例:
javascriptvar loadingData = ajax.get("url"); loadingData.then(callback);2.4 注釋 2.4.1 單行注釋
解釋:
文件
namespace
類
函數(shù)或方法
類屬性
事件
全局變量
常量
AMD 模塊
解釋:
常用類型如:{string}, {number}, {boolean}, {Object}, {Function}, {RegExp}, {Array}, {Date}。
類型不僅局限于內(nèi)置的類型,也可以是自定義的類型。比如定義了一個類 Developer,就可以使用它來定義一個參數(shù)和返回值的類型。
類型定義 | 語法示例 | 解釋 |
---|---|---|
String | {string} | -- |
Number | {number} | -- |
Boolean | {boolean} | -- |
Object | {Object} | -- |
Function | {Function} | -- |
RegExp | {RegExp} | -- |
Array | {Array} | -- |
Date | {Date} | -- |
單一類型集合 | {Array. |
string 類型的數(shù)組 |
多類型 | {(number|boolean)} | 可能是 number 類型, 也可能是 boolean 類型 |
允許為null | {?number} | 可能是 number, 也可能是 null |
不允許為null | {!Object} | Object 類型, 但不是 null |
Function類型 | {function(number, boolean)} | 函數(shù), 形參類型 |
Function帶返回值 | {function(number, boolean):string} | 函數(shù), 形參, 返回值類型 |
參數(shù)可選 | @param {string=} name | 可選參數(shù), =為類型后綴 |
可變參數(shù) | @param {...number} args | 變長參數(shù), ...為類型前綴 |
任意類型 | {*} | 任意類型 |
可選任意類型 | @param {*=} name | 可選參數(shù),類型不限 |
可變?nèi)我忸愋?/td> | @param {...*} args | 變長參數(shù),類型不限 |
示例:
javascript/** * @file Describe the file */
解釋:
開發(fā)者信息能夠體現(xiàn)開發(fā)人員對文件的貢獻,并且能夠讓遇到問題或希望了解相關(guān)信息的人找到維護人。通常情況文件在被創(chuàng)建時標識的是創(chuàng)建者。隨著項目的進展,越來越多的人加入,參與這個文件的開發(fā),新的作者應(yīng)該被加入 @author 標識。
@author 標識具有多人時,原則是按照 責(zé)任 進行排序。通常的說就是如果有問題,就是找第一個人應(yīng)該比找第二個人有效。比如文件的創(chuàng)建者由于各種原因,模塊移交給了其他人或其他團隊,后來因為新增需求,其他人在新增代碼時,添加 @author 標識應(yīng)該把自己的名字添加在創(chuàng)建人的前面。
@author 中的名字不允許被刪除。任何勞動成果都應(yīng)該被尊重。
業(yè)務(wù)項目中,一個文件可能被多人頻繁修改,并且每個人的維護時間都可能不會很長,不建議為文件增加 @author 標識。通過版本控制系統(tǒng)追蹤變更,按業(yè)務(wù)邏輯單元確定模塊的維護責(zé)任人,通過文檔與wiki跟蹤和查詢,是更好的責(zé)任管理方式。
對于業(yè)務(wù)邏輯無關(guān)的技術(shù)型基礎(chǔ)項目,特別是開源的公共項目,應(yīng)使用 @author 標識。
示例:
javascript/** * @file Describe the file * @author author-name([email protected]) * author-name2([email protected]) */2.4.6 命名空間注釋
示例:
javascript/** * @namespace */ var util = {};2.4.7 類注釋
解釋:
對于使用對象 constructor 屬性來定義的構(gòu)造函數(shù),可以使用 @constructor 來標記。
示例:
javascript/** * 描述 * * @class */ function Developer() { // constructor body }
示例:
javascript/** * 描述 * * @class * @extends Developer */ function Fronteer() { Developer.call(this); // constructor body } util.inherits(Fronteer, Developer);
解釋:
沒有 @lends 標記將無法為該類生成包含擴展類成員的文檔。
示例:
javascript/** * 類描述 * * @class * @extends Developer */ function Fronteer() { Developer.call(this); // constructor body } util.extend( Fronteer.prototype, /** @lends Fronteer.prototype */{ _getLevel: function () { // TODO } } );
解釋:
生成的文檔中將有可訪問性的標記,避免用戶直接使用非 public 的屬性或方法。
示例:
javascript/** * 類描述 * * @class * @extends Developer */ var Fronteer = function () { Developer.call(this); /** * 屬性描述 * * @type {string} * @private */ this._level = "T12"; // constructor body }; util.inherits(Fronteer, Developer); /** * 方法描述 * * @private * @return {string} 返回值描述 */ Fronteer.prototype._getLevel = function () { };2.4.8 函數(shù)/方法注釋
示例:
javascript/** * 函數(shù)描述 * * @param {string} p1 參數(shù)1的說明 * @param {string} p2 參數(shù)2的說明,比較長 * 那就換行了. * @param {number=} p3 參數(shù)3的說明(可選) * @return {Object} 返回值描述 */ function foo(p1, p2, p3) { var p3 = p3 || 10; return { p1: p1, p2: p2, p3: p3 }; }
示例:
javascript/** * 函數(shù)描述 * * @param {Object} option 參數(shù)描述 * @param {string} option.url option項描述 * @param {string=} option.method option項描述,可選參數(shù) */ function foo(option) { // TODO }
解釋:
簡而言之,當子類重寫的方法能直接套用父類的方法注釋時可省略對參數(shù)與返回值的注釋。
2.4.9 事件注釋示例:
javascript/** * 值變更時觸發(fā) * * @event * @param {Object} e e描述 * @param {string} e.before before描述 * @param {string} e.after after描述 */ onchange: function (e) { }
示例:
javascript/** * 點擊處理 * * @fires Select#change * @private */ Select.prototype.clickHandler = function () { /** * 值變更時觸發(fā) * * @event Select#change * @param {Object} e e描述 * @param {string} e.before before描述 * @param {string} e.after after描述 */ this.fire( "change", { before: "foo", after: "bar" } ); };2.4.10 常量注釋
示例:
javascript/** * 常量說明 * * @const * @type {string} */ var REQUEST_URL = "myurl.do";2.4.11 復(fù)雜類型注釋
示例:
javascript// `namespaceA~` 可以換成其它 namepaths 前綴,目的是為了生成文檔中能顯示 `@typedef` 定義的類型和鏈接。 /** * 服務(wù)器 * * @typedef {Object} namespaceA~Server * @property {string} host 主機 * @property {number} port 端口 */ /** * 服務(wù)器列表 * * @type {Array.2.4.12 AMD 模塊注釋} */ var servers = [ { host: "1.2.3.4", port: 8080 }, { host: "1.2.3.5", port: 8081 } ];
解釋:
@exports 與 @module 都可以用來標識模塊,區(qū)別在于 @module 可以省略模塊名稱。而只使用 @exports 時在 namepaths 中可以省略 module: 前綴。
示例:
javascriptdefine( function (require) { /** * foo description * * @exports Foo */ var foo = { // TODO }; /** * baz description * * @return {boolean} return description */ foo.baz = function () { // TODO }; return foo; } );
也可以在 exports 變量前使用 @module 標識:
javascriptdefine( function (require) { /** * module description. * * @module foo */ var exports = {}; /** * bar description * */ exports.bar = function () { // TODO }; return exports; } );
如果直接使用 factory 的 exports 參數(shù),還可以:
javascript/** * module description. * * @module */ define( function (require, exports) { /** * bar description * */ exports.bar = function () { // TODO }; return exports; } );
解釋:
namepaths 沒有 module: 前綴時,生成的文檔中將無法正確生成鏈接。
示例:
javascript/** * 點擊處理 * * @fires module:Select#change * @private */ Select.prototype.clickHandler = function () { /** * 值變更時觸發(fā) * * @event module:Select#change * @param {Object} e e描述 * @param {string} e.before before描述 * @param {string} e.after after描述 */ this.fire( "change", { before: "foo", after: "bar" } ); };
示例:
javascript/** * A module representing a jacket. * @module jacket */ define( function () { /** * @class * @alias module:jacket */ var Jacket = function () { }; return Jacket; } );
示例:
javascript// one module define("html/utils", /** * Utility functions to ease working with DOM elements. * @exports html/utils */ function () { var exports = { }; return exports; } ); // another module define("tag", /** @exports tag */ function () { var exports = { }; return exports; } );
解釋:
使用 @namespace 而不是 @module 或 @exports 時,對模塊的引用可以省略 module: 前綴。
示例:
javascript2.4.13 細節(jié)注釋
// 只使用 @class Bar 時,類方法和屬性都必須增加 @name Bar#methodName 來標識,與 @exports 配合可以免除這一麻煩,并且在引用時可以省去 module: 前綴。 // 另外需要注意類名需要使用 var 定義的方式。 /** * Bar description * * @see foo * @exports Bar * @class */ var Bar = function () { // TODO }; /** * baz description * * @return {(string|Array)} return description */ Bar.prototype.baz = function () { // TODO };
對于內(nèi)部實現(xiàn)、不容易理解的邏輯說明、摘要信息等,我們可能需要編寫細節(jié)注釋。
[建議] 細節(jié)注釋遵循單行注釋的格式。說明必須換行時,每行是一個單行注釋的起始。示例:
javascriptfunction foo(p1, p2) { // 這里對具體內(nèi)部邏輯進行說明 // 說明太長需要換行 for (...) { .... } }
解釋:
TODO: 有功能待實現(xiàn)。此時需要對將要實現(xiàn)的功能進行簡單說明。
FIXME: 該處代碼運行沒問題,但可能由于時間趕或者其他原因,需要修正。此時需要對如何修正進行簡單說明。
HACK: 為修正某些問題而寫的不太好或者使用了某些詭異手段的代碼。此時需要對思路或詭異手段進行描述。
XXX: 該處存在陷阱。此時需要對陷阱進行描述。
3 語言特性 3.1 變量解釋:
不通過 var 定義變量將導(dǎo)致變量污染全局環(huán)境。
示例:
javascript// good var name = "MyName"; // bad name = "MyName";
解釋:
一個 var 聲明多個變量,容易導(dǎo)致較長的行長度,并且在修改時容易造成逗號和分號的混淆。
示例:
javascript// good var hangModules = []; var missModules = []; var visited = {}; // bad var hangModules = [], missModules = [], visited = {};
解釋:
變量聲明與使用的距離越遠,出現(xiàn)的跨度越大,代碼的閱讀與維護成本越高。雖然JavaScript的變量是函數(shù)作用域,還是應(yīng)該根據(jù)編程中的意圖,縮小變量出現(xiàn)的距離空間。
示例:
javascript// good function kv2List(source) { var list = []; for (var key in source) { if (source.hasOwnProperty(key)) { var item = { k: key, v: source[key] }; list.push(item); } } return list; } // bad function kv2List(source) { var list = []; var key; var item; for (key in source) { if (source.hasOwnProperty(key)) { item = { k: key, v: source[key] }; list.push(item); } } return list; }3.2 條件
解釋:
使用 === 可以避免等于判斷中隱式的類型轉(zhuǎn)換。
示例:
javascript// good if (age === 30) { // ...... } // bad if (age == 30) { // ...... }
示例:
javascript// 字符串為空 // good if (!name) { // ...... } // bad if (name === "") { // ...... }
javascript// 字符串非空 // good if (name) { // ...... } // bad if (name !== "") { // ...... }
javascript// 數(shù)組非空 // good if (collection.length) { // ...... } // bad if (collection.length > 0) { // ...... }
javascript// 布爾不成立 // good if (!notTrue) { // ...... } // bad if (notTrue === false) { // ...... }
javascript// null 或 undefined // good if (noValue == null) { // ...... } // bad if (noValue === null || typeof noValue === "undefined") { // ...... }
解釋:
按執(zhí)行頻率排列分支的順序好處是:
閱讀的人容易找到最常見的情況,增加可讀性。
提高執(zhí)行效率。
示例:
javascript// good switch (typeof variable) { case "object": // ...... break; case "number": case "boolean": case "string": // ...... break; } // bad var type = typeof variable; if (type === "object") { // ...... } else if (type === "number" || type === "boolean" || type === "string") { // ...... }
示例:
javascript// good function getName() { if (name) { return name; } return "unnamed"; } // bad function getName() { if (name) { return name; } else { return "unnamed"; } }3.3 循環(huán)
解釋:
循環(huán)體中的函數(shù)表達式,運行過程中會生成循環(huán)次數(shù)個函數(shù)對象。
示例:
javascript// good function clicker() { // ...... } for (var i = 0, len = elements.length; i < len; i++) { var element = elements[i]; addListener(element, "click", clicker); } // bad for (var i = 0, len = elements.length; i < len; i++) { var element = elements[i]; addListener(element, "click", function () {}); }
示例:
javascript// good var width = wrap.offsetWidth + "px"; for (var i = 0, len = elements.length; i < len; i++) { var element = elements[i]; element.style.width = width; // ...... } // bad for (var i = 0, len = elements.length; i < len; i++) { var element = elements[i]; element.style.width = wrap.offsetWidth + "px"; // ...... }
解釋:
雖然現(xiàn)代瀏覽器都對數(shù)組長度進行了緩存,但對于一些宿主對象和老舊瀏覽器的數(shù)組對象,在每次 length 訪問時會動態(tài)計算元素個數(shù),此時緩存 length 能有效提高程序性能。
示例:
javascriptfor (var i = 0, len = elements.length; i < len; i++) { var element = elements[i]; // ...... }
解釋:
逆序遍歷可以節(jié)省變量,代碼比較優(yōu)化。
示例:
javascriptvar len = elements.length; while (len--) { var element = elements[len]; // ...... }3.4 類型 3.4.1 類型檢測
示例:
javascript// string typeof variable === "string" // number typeof variable === "number" // boolean typeof variable === "boolean" // Function typeof variable === "function" // Object typeof variable === "object" // RegExp variable instanceof RegExp // Array variable instanceof Array // null variable === null // null or undefined variable == null // undefined typeof variable === "undefined"3.4.2 類型轉(zhuǎn)換
示例:
javascript// good num + ""; // bad new String(num); num.toString(); String(num);
示例:
javascript// good +str; // bad Number(str);
示例:
javascriptvar width = "200px"; parseInt(width, 10);
示例:
javascript// good parseInt(str, 10); // bad parseInt(str);
示例:
javascriptvar num = 3.14; !!num;
示例:
javascript// good var num = 3.14; Math.ceil(num); // bad var num = 3.14; parseInt(num, 10);3.5 字符串
解釋:
輸入單引號不需要按住 shift,方便輸入。
實際使用中,字符串經(jīng)常用來拼接 HTML。為方便 HTML 中包含雙引號而不需要轉(zhuǎn)義寫法。
示例:
javascriptvar str = "我是一個字符串"; var html = "拼接HTML可以省去雙引號轉(zhuǎn)義";
解釋:
使用 + 拼接字符串,如果拼接的全部是 StringLiteral,壓縮工具可以對其進行自動合并的優(yōu)化。所以,靜態(tài)字符串建議使用 + 拼接。
在現(xiàn)代瀏覽器下,使用 + 拼接字符串,性能較數(shù)組的方式要高。
如需要兼顧老舊瀏覽器,應(yīng)盡量使用數(shù)組拼接字符串。
示例:
javascript// 使用數(shù)組拼接字符串 var str = [ // 推薦換行開始并縮進開始第一個字符串, 對齊代碼, 方便閱讀. "", "
" ].join(""); // 使用 + 拼接字符串 var str2 = "" // 建議第一個為空字符串, 第二個換行開始并縮進開始, 對齊代碼, 方便閱讀 + "- 第一項
", "- 第二項
", "", + "
";- 第一項
", + "- 第二項
", + "
解釋:
使用模板引擎有如下好處:
在開發(fā)過程中專注于數(shù)據(jù),將視圖生成的過程由另外一個層級維護,使程序邏輯結(jié)構(gòu)更清晰。
優(yōu)秀的模板引擎,通過模板編譯技術(shù)和高質(zhì)量的編譯產(chǎn)物,能獲得比手工拼接字符串更高的性能。
artTemplate: 體積較小,在所有環(huán)境下性能高,語法靈活。
dot.js: 體積小,在現(xiàn)代瀏覽器下性能高,語法靈活。
etpl: 體積較小,在所有環(huán)境下性能高,模板復(fù)用性高,語法靈活。
handlebars: 體積大,在所有環(huán)境下性能高,擴展性高。
hogon: 體積小,在現(xiàn)代瀏覽器下性能高。
nunjucks: 體積較大,性能一般,模板復(fù)用性高。
3.6 對象示例:
javascript// good var obj = {}; // bad var obj = new Object();
示例:
javascriptvar info = { name: "someone", age: 28 };
解釋:
如果屬性不符合 Identifier 和 NumberLiteral 的形式,就需要以 StringLiteral 的形式提供。
示例:
javascript// good var info = { "name": "someone", "age": 28, "more-info": "..." }; // bad var info = { name: "someone", age: 28, "more-info": "..." };
示例:
javascript// 以下行為絕對禁止 String.prototype.trim = function () { };
解釋:
屬性名符合 Identifier 的要求,就可以通過 . 來訪問,否則就只能通過 [expr] 方式訪問。
通常在 JavaScript 中聲明的對象,屬性命名是使用 Camel 命名法,用 . 來訪問更清晰簡潔。部分特殊的屬性(比如來自后端的JSON),可能采用不尋常的命名方式,可以通過 [expr] 方式訪問。
示例:
javascriptinfo.age; info["more-info"];
示例:
javascriptvar newInfo = {}; for (var key in info) { if (info.hasOwnProperty(key)) { newInfo[key] = info[key]; } }3.7 數(shù)組
示例:
javascript// good var arr = []; // bad var arr = new Array();
解釋:
數(shù)組對象可能存在數(shù)字以外的屬性, 這種情況下 for in 不會得到正確結(jié)果.
示例:
javascriptvar arr = ["a", "b", "c"]; arr.other = "other things"; // 這里僅作演示, 實際中應(yīng)使用Object類型 // 正確的遍歷方式 for (var i = 0, len = arr.length; i < len; i++) { console.log(i); } // 錯誤的遍歷方式 for (i in arr) { console.log(i); }
解釋:
自己實現(xiàn)的常規(guī)排序算法,在性能上并不優(yōu)于數(shù)組默認的 sort 方法。以下兩種場景可以自己實現(xiàn)排序:
需要穩(wěn)定的排序算法,達到嚴格一致的排序結(jié)果。
數(shù)據(jù)特點鮮明,適合使用桶排。
解釋:
將過多的邏輯單元混在一個大函數(shù)中,易導(dǎo)致難以維護。一個清晰易懂的函數(shù)應(yīng)該完成單一的邏輯單元。復(fù)雜的操作應(yīng)進一步抽取,通過函數(shù)的調(diào)用來體現(xiàn)流程。
特定算法等不可分割的邏輯允許例外。
示例:
javascriptfunction syncViewStateOnUserAction() { if (x.checked) { y.checked = true; z.value = ""; } else { y.checked = false; } if (!a.value) { warning.innerText = "Please enter it"; submitButton.disabled = true; } else { warning.innerText = ""; submitButton.disabled = false; } } // 直接閱讀該函數(shù)會難以明確其主線邏輯,因此下方是一種更合理的表達方式: function syncViewStateOnUserAction() { syncXStateToView(); checkAAvailability(); } function syncXStateToView() { if (x.checked) { y.checked = true; z.value = ""; } else { y.checked = false; } } function checkAAvailability() { if (!a.value) { displayWarningForAMissing(); } else { clearWarnignForA(); } }3.8.2 參數(shù)設(shè)計
解釋:
除去不定長參數(shù)以外,函數(shù)具備不同邏輯意義的參數(shù)建議控制在 6 個以內(nèi),過多參數(shù)會導(dǎo)致維護難度增大。
某些情況下,如使用 AMD Loader 的 require 加載多個模塊時,其 callback 可能會存在較多參數(shù),因此對函數(shù)參數(shù)的個數(shù)不做強制限制。
解釋:
有些函數(shù)的參數(shù)并不是作為算法的輸入,而是對算法的某些分支條件判斷之用,此類參數(shù)建議通過一個 options 參數(shù)傳遞。
如下函數(shù):
javascript/** * 移除某個元素 * * @param {Node} element 需要移除的元素 * @param {boolean} removeEventListeners 是否同時將所有注冊在元素上的事件移除 */ function removeElement(element, removeEventListeners) { element.parent.removeChild(element); if (removeEventListeners) { element.clearEventListeners(); } }
可以轉(zhuǎn)換為下面的簽名:
javascript/** * 移除某個元素 * * @param {Node} element 需要移除的元素 * @param {Object} options 相關(guān)的邏輯配置 * @param {boolean} options.removeEventListeners 是否同時將所有注冊在元素上的事件移除 */ function removeElement(element, options) { element.parent.removeChild(element); if (options.removeEventListeners) { element.clearEventListeners(); } }
這種模式有幾個顯著的優(yōu)勢:
boolean 型的配置項具備名稱,從調(diào)用的代碼上更易理解其表達的邏輯意義。
當配置項有增長時,無需無休止地增加參數(shù)個數(shù),不會出現(xiàn) removeElement(element, true, false, false, 3) 這樣難以理解的調(diào)用代碼。
當部分配置參數(shù)可選時,多個參數(shù)的形式非常難處理重載邏輯,而使用一個 options 對象只需判斷屬性是否存在,實現(xiàn)得以簡化。
3.8.3 閉包解釋:
在 JavaScript 中,無需特別的關(guān)鍵詞就可以使用閉包,一個函數(shù)可以任意訪問在其定義的作用域外的變量。需要注意的是,函數(shù)的作用域是靜態(tài)的,即在定義時決定,與調(diào)用的時機和方式?jīng)]有任何關(guān)系。
閉包會阻止一些變量的垃圾回收,對于較老舊的JavaScript引擎,可能導(dǎo)致外部所有變量均無法回收。
首先一個較為明確的結(jié)論是,以下內(nèi)容會影響到閉包內(nèi)變量的回收:
嵌套的函數(shù)中是否有使用該變量。
嵌套的函數(shù)中是否有 直接調(diào)用eval。
是否使用了 with 表達式。
Chakra、V8 和 SpiderMonkey 將受以上因素的影響,表現(xiàn)出不盡相同又較為相似的回收策略,而JScript.dll和Carakan則完全沒有這方面的優(yōu)化,會完整保留整個 LexicalEnvironment 中的所有變量綁定,造成一定的內(nèi)存消耗。
由于對閉包內(nèi)變量有回收優(yōu)化策略的 Chakra、V8 和 SpiderMonkey 引擎的行為較為相似,因此可以總結(jié)如下,當返回一個函數(shù) fn 時:
如果 fn 的 [[Scope]] 是ObjectEnvironment(with 表達式生成 ObjectEnvironment,函數(shù)和 catch 表達式生成 DeclarativeEnvironment),則:
如果是 V8 引擎,則退出全過程。
如果是 SpiderMonkey,則處理該 ObjectEnvironment 的外層 LexicalEnvironment。
獲取當前 LexicalEnvironment 下的所有類型為 Function 的對象,對于每一個 Function 對象,分析其 FunctionBody:
如果 FunctionBody 中含有 直接調(diào)用eval,則退出全過程。
否則得到所有的 Identifier。
對于每一個 Identifier,設(shè)其為 name,根據(jù)查找變量引用的規(guī)則,從 LexicalEnvironment 中找出名稱為 name 的綁定 binding。
對 binding 添加 notSwap 屬性,其值為 true。
檢查當前 LexicalEnvironment 中的每一個變量綁定,如果該綁定有 notSwap 屬性且值為 true,則:
如果是V8引擎,刪除該綁定。
如果是SpiderMonkey,將該綁定的值設(shè)為 undefined,將刪除 notSwap 屬性。
對于Chakra引擎,暫無法得知是按 V8 的模式還是按 SpiderMonkey 的模式進行。
如果有 非常龐大 的對象,且預(yù)計會在 老舊的引擎 中執(zhí)行,則使用閉包時,注意將閉包不需要的對象置為空引用。
解釋:
在引用函數(shù)外部變量時,函數(shù)執(zhí)行時外部變量的值由運行時決定而非定義時,最典型的場景如下:
javascriptvar tasks = []; for (var i = 0; i < 5; i++) { tasks[tasks.length] = function () { console.log("Current cursor is at " + i); }; } var len = tasks.length; while (len--) { tasks[len](); }
以上代碼對 tasks 中的函數(shù)的執(zhí)行均會輸出 Current cursor is at 5,往往不符合預(yù)期。
此現(xiàn)象稱為 Lift 效應(yīng) 。解決的方式是通過額外加上一層閉包函數(shù),將需要的外部變量作為參數(shù)傳遞來解除變量的綁定關(guān)系:
javascriptvar tasks = []; for (var i = 0; i < 5; i++) { // 注意有一層額外的閉包 tasks[tasks.length] = (function (i) { return function () { console.log("Current cursor is at " + i); }; })(i); } var len = tasks.length; while (len--) { tasks[len](); }3.8.4 空函數(shù)
示例:
javascriptvar emptyFunction = function () {};
示例:
javascriptvar EMPTY_FUNCTION = function () {}; function MyClass() { } MyClass.prototype.abstractMethod = EMPTY_FUNCTION; MyClass.prototype.hooks.before = EMPTY_FUNCTION; MyClass.prototype.hooks.after = EMPTY_FUNCTION;3.9 面向?qū)ο?/b>
解釋:
通常使用其他 library 的類繼承方案都會進行 constructor 修正。如果是自己實現(xiàn)的類繼承方案,需要進行 constructor 修正。
示例:
javascript/** * 構(gòu)建類之間的繼承關(guān)系 * * @param {Function} subClass 子類函數(shù) * @param {Function} superClass 父類函數(shù) */ function inherits(subClass, superClass) { var F = new Function(); F.prototype = superClass.prototype; subClass.prototype = new F(); subClass.prototype.constructor = subClass; }
示例:
javascriptfunction Animal(name) { this.name = name; } // 直接prototype等于對象時,需要修正constructor Animal.prototype = { constructor: Animal, jump: function () { alert("animal " + this.name + " jump"); } }; // 這種方式擴展prototype則無需理會constructor Animal.prototype.jump = function () { alert("animal " + this.name + " jump"); };
解釋:
原型對象的成員被所有實例共享,能節(jié)約內(nèi)存占用。所以編碼時我們應(yīng)該遵守這樣的原則:原型對象包含程序不會修改的成員,如方法函數(shù)或配置項。
javascriptfunction TextNode(value, engine) { this.value = value; this.engine = engine; } TextNode.prototype.clone = function () { return this; };
解釋:
在 JavaScript 廣泛應(yīng)用的瀏覽器環(huán)境,絕大多數(shù) DOM 事件名稱都是全小寫的。為了遵循大多數(shù) JavaScript 開發(fā)者的習(xí)慣,在設(shè)計自定義事件時,事件名也應(yīng)該全小寫。
解釋:
一個事件對象的好處有:
順序無關(guān),避免事件監(jiān)聽者需要記憶參數(shù)順序。
每個事件信息都可以根據(jù)需要提供或者不提供,更自由。
擴展方便,未來添加事件信息時,無需考慮會破壞監(jiān)聽器參數(shù)形式而無法向后兼容。
解釋:
常見禁止默認行為的方式有兩種:
事件監(jiān)聽函數(shù)中 return false。
事件對象中包含禁止默認行為的方法,如 preventDefault。
3.10 動態(tài)特性 3.10.1 eval解釋:
直接 eval,指的是以函數(shù)方式調(diào)用 eval 的調(diào)用方法。直接 eval 調(diào)用執(zhí)行代碼的作用域為本地作用域,應(yīng)當避免。
如果有特殊情況需要使用直接 eval,需在代碼中用詳細的注釋說明為何必須使用直接 eval,不能使用其它動態(tài)執(zhí)行代碼的方式,同時需要其他資深工程師進行 Code Review。
解釋:
通過 new Function 生成的函數(shù)作用域是全局使用域,不會影響當當前的本地作用域。如果有動態(tài)代碼執(zhí)行的需求,建議使用 new Function。
示例:
javascriptvar handler = new Function("x", "y", "return x + y;"); var result = handler($("#x").val(), $("#y").val());3.10.3 with
解釋:
使用 with 可能會增加代碼的復(fù)雜度,不利于閱讀和管理;也會對性能有影響。大多數(shù)使用 with 的場景都能使用其他方式較好的替代。所以,盡量不要使用 with。
3.10.4 delete解釋:
如果沒有特別的需求,減少或避免使用delete。delete的使用會破壞部分 JavaScript 引擎的性能優(yōu)化。
解釋:
對于有被遍歷需求,且值 null 被認為具有業(yè)務(wù)邏輯意義的值的對象,移除某個屬性必須使用 delete 操作。
在嚴格模式或IE下使用 delete 時,不能被刪除的屬性會拋出異常,因此在不確定屬性是否可以刪除的情況下,建議添加 try-catch 塊。
示例:
javascripttry { delete o.x; } catch (deleteError) { o.x = null; }3.10.5 對象屬性
解釋:
JavaScript 因其腳本語言的動態(tài)特性,當一個對象未被 seal 或 freeze 時,可以任意添加、刪除、修改屬性值。
但是隨意地對 非自身控制的對象 進行修改,很容易造成代碼在不可預(yù)知的情況下出現(xiàn)問題。因此,設(shè)計良好的組件、函數(shù)應(yīng)該避免對外部傳入的對象的修改。
下面代碼的 selectNode 方法修改了由外部傳入的 datasource 對象。如果 datasource 用在其它場合(如另一個 Tree 實例)下,會造成狀態(tài)的混亂。
javascriptfunction Tree(datasource) { this.datasource = datasource; } Tree.prototype.selectNode = function (id) { // 從datasource中找出節(jié)點對象 var node = this.findNode(id); if (node) { node.selected = true; this.flushView(); } };
對于此類場景,需要使用額外的對象來維護,使用由自身控制,不與外部產(chǎn)生任何交互的 selectedNodeIndex 對象來維護節(jié)點的選中狀態(tài),不對 datasource 作任何修改。
javascriptfunction Tree(datasource) { this.datasource = datasource; this.selectedNodeIndex = {}; } Tree.prototype.selectNode = function (id) { // 從datasource中找出節(jié)點對象 var node = this.findNode(id); if (node) { this.selectedNodeIndex[id] = true; this.flushView(); } };
除此之外,也可以通過 deepClone 等手段將自身維護的對象與外部傳入的分離,保證不會相互影響。
解釋:
如果一個屬性被設(shè)計為 boolean 類型,則不要使用 1 / 0 作為其值。對于標識性的屬性,如對代碼體積有嚴格要求,可以從一開始就設(shè)計為 number 類型且將 0 作為否定值。
從 DOM 中取出的值通常為 string 類型,如果有對象或函數(shù)的接收類型為 number 類型,提前作好轉(zhuǎn)換,而不是期望對象、函數(shù)可以處理多類型的值。
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/87682.html
摘要:這樣的變量增加了代碼量,并且混淆讀者。錯誤代碼示例變量雖然聲明了,但沒被使用持續(xù)更新 JavaScript 編碼規(guī)范 一、命名規(guī)范 1. 變量 命名方法:小駝峰式命名法(由小寫字母開始,后續(xù)每個單詞首字母都大寫) 命名建議:語義化的名詞 特殊:布爾值變量建議添加符合其含義的前綴動詞 is:是否 can:能不能 has:有沒有 示例: // 頁面標題 let pageT...
摘要:編碼規(guī)范是獨角獸公司內(nèi)部的編碼規(guī)范,該項目是上很受歡迎的一個開源項目,在前端開發(fā)中使用廣泛,本文的配置規(guī)則就是以編碼規(guī)范和編碼規(guī)范作為基礎(chǔ)的。 更新時間:2019-01-22React.js create-react-app 項目 + VSCode 編輯器 + ESLint 代碼檢查工具 + Airbnb 編碼規(guī)范 前言 為什么要使用 ESLint 在項目開發(fā)過程中,編寫符合團隊編碼規(guī)...
摘要:當然我們還可以引入框架,這些框架一般都自帶模板處理引擎,比如等語義化命名和語義化標簽我們盡量多采用語義化來命名,并且采用語義化標簽來書寫代碼,多用中新增的標簽來書寫。 1.黃金法則(Golden rule) 不管有多少人參與同一個項目,一定要確保每一行代碼都像是同一個人編寫的。 Every line of code should appear to be written by a si...
摘要:寫在前面對于不同的編程語言來說,具體的編碼規(guī)范各不相同,但是其宗旨都是一致的,就是保證代碼在高質(zhì)量完成需求的同時具備良好的可讀性可維護性。減少標簽的數(shù)量編寫代碼時,盡量避免多余的父元素。 寫在前面 對于不同的編程語言來說,具體的編碼規(guī)范各不相同,但是其宗旨都是一致的,就是保證代碼在高質(zhì)量完成需求的同時具備良好的可讀性、可維護性。 本文大部分內(nèi)容來自網(wǎng)上,僅供個人參考學(xué)習(xí)! 網(wǎng)絡(luò)上的知...
摘要:用兩個空格代替制表符這是唯一能保證在所有環(huán)境下獲得一致展現(xiàn)的方法。編輯器配置將你的編輯器按照下面的配置進行設(shè)置,以免常見的代碼不一致和差異用兩個空格代替制表符保存文件時刪除尾部的空白符設(shè)置文件編碼為在文件結(jié)尾添加一個空白行。 黃金定律 永遠遵循同一套編碼規(guī)范 - 可以是這里列出的,也可以是你自己總結(jié)的。如果發(fā)現(xiàn)規(guī)范中有任何錯誤,敬請指正。 HTML 語法 用兩個空格代替制表符 (ta...
摘要:六字符編碼通過明確聲明字符編碼,能夠確保瀏覽器快速并容易的判斷頁面內(nèi)容的渲染方式。十一減少標簽的數(shù)量編寫代碼時,盡量避免多余的父元素。未完待續(xù)編寫靈活穩(wěn)定高質(zhì)量的代碼的規(guī)范閱讀更多 一、唯一定律 無論有多少人共同參與同一項目,一定要確保每一行代碼都像是唯一個人編寫的。 二、HTML 2.1 語法 (1)用兩個空格來代替制表符(tab) -- 這是唯一能保證在所有環(huán)境下獲得一致展現(xiàn)的方法...
閱讀 3631·2021-11-22 09:34
閱讀 3198·2021-11-15 11:38
閱讀 3078·2021-10-27 14:16
閱讀 1263·2021-10-18 13:35
閱讀 2437·2021-09-30 09:48
閱讀 3439·2021-09-29 09:34
閱讀 1662·2019-08-30 15:54
閱讀 1830·2019-08-26 11:57