這篇文章主要講解了“如何正確使用JS的變量和函數(shù)”,文中的講解內(nèi)容簡(jiǎn)單清晰,易于學(xué)習(xí)與理解,下面請(qǐng)大家跟著小編的思路慢慢深入,一起來(lái)研究和學(xué)習(xí)“如何正確使用JS的變量和函數(shù)”吧!
10年積累的成都網(wǎng)站建設(shè)、網(wǎng)站建設(shè)經(jīng)驗(yàn),可以快速應(yīng)對(duì)客戶對(duì)網(wǎng)站的新想法和需求。提供各種問(wèn)題對(duì)應(yīng)的解決方案。讓選擇我們的客戶得到更好、更有力的網(wǎng)絡(luò)服務(wù)。我雖然不認(rèn)識(shí)你,你也不認(rèn)識(shí)我。但先網(wǎng)站制作后付款的網(wǎng)站建設(shè)流程,更有崇左免費(fèi)網(wǎng)站建設(shè)讓你可以放心的選擇與我們合作。
一、變量
使用有意義和可發(fā)音的變量名
// 不好的寫(xiě)法 const yyyymmdstr = moment().format("YYYY/MM/DD"); // 好的寫(xiě)法 const currentDate = moment().format("YYYY/MM/DD");
對(duì)同一類型的變量使用相同的詞匯
// 不好的寫(xiě)法 getUserInfo(); getClientData(); getCustomerRecord(); // 好的寫(xiě)法 getUser();
使用可搜索的名字
我們讀的會(huì)比我們寫(xiě)的多得多,所以如果命名太過(guò)隨意不僅會(huì)給后續(xù)的維護(hù)帶來(lái)困難,也會(huì)傷害了讀我們代碼的開(kāi)發(fā)者。讓你的變量名可被讀取,像 buddy.js 和 ESLint 這樣的工具可以幫助識(shí)別未命名的常量。
// 不好的寫(xiě)法 // 86400000 的用途是什么? setTimeout(blastOff, 86400000); // 好的寫(xiě)法 const MILLISECONDS_IN_A_DAY = 86_400_000; setTimeout(blastOff, MILLISECONDS_IN_A_DAY);
使用解釋性變量
// 不好的寫(xiě)法 const address = "One Infinite Loop, Cupertino 95014"; const cityZipCodeRegex = /^[^,\\]+[,\\\s]+(.+?)\s*(\d{5})?$/; saveCityZipCode( address.match(cityZipCodeRegex)[1], address.match(cityZipCodeRegex)[2] ); // 好的寫(xiě)法 const address = "One Infinite Loop, Cupertino 95014"; const cityZipCodeRegex = /^[^,\\]+[,\\\s]+(.+?)\s*(\d{5})?$/; const [_, city, zipCode] = address.match(cityZipCodeRegex) || []; saveCityZipCode(city, zipCode);
避免費(fèi)腦的猜測(cè)
顯式用于隱式
// 不好的寫(xiě)法 const locations = ["Austin", "New York", "San Francisco"]; locations.forEach(l => { doStuff(); doSomeOtherStuff(); // ... // ... // ... // 等等,“l(fā)”又是什么? dispatch(l); // 好的寫(xiě)法 const locations = ["Austin", "New York", "San Francisco"]; locations.forEach(location => { doStuff(); doSomeOtherStuff(); // ... // ... // ... dispatch(location); });
無(wú)需添加不必要的上下文
如果類名/對(duì)象名已經(jīng)說(shuō)明了,就無(wú)需在變量名中重復(fù)。
// 不好的寫(xiě)法 const Car = { carMake: "Honda", carModel: "Accord", carColor: "Blue" }; function paintCar(car) { car.carColor = "Red"; } // 好的寫(xiě)法 const Car = { make: "Honda", model: "Accord", color: "Blue" }; function paintCar(car) { car.color = "Red"; }
使用默認(rèn)參數(shù)代替邏輯或(與)運(yùn)算
// 不好的寫(xiě)法 function createMicrobrewery(name) { const breweryName = name || "Hipster Brew Co."; // ... } // 好的寫(xiě)法 function createMicrobrewery(name = "Hipster Brew Co.") { // ... }
二、函數(shù)
函數(shù)參數(shù)(理想情況下為2個(gè)或更少)
限制函數(shù)參數(shù)的數(shù)量是非常重要的,因?yàn)樗箿y(cè)試函數(shù)變得更容易。如果有三個(gè)以上的參數(shù),就會(huì)導(dǎo)致組合爆炸,必須用每個(gè)單獨(dú)的參數(shù)測(cè)試大量不同的情況。
一個(gè)或兩個(gè)參數(shù)是理想的情況,如果可能,應(yīng)避免三個(gè)參數(shù)。除此之外,還應(yīng)該合并。大多數(shù)情況下,大于三個(gè)參數(shù)可以用對(duì)象來(lái)代替。
// 不好的寫(xiě)法 function createMenu(title, body, buttonText, cancellable) { // ... } createMenu("Foo", "Bar", "Baz", true); // 好的寫(xiě)法 function createMenu({ title, body, buttonText, cancellable }) { // ... } createMenu({ title: "Foo", body: "Bar", buttonText: "Baz", cancellable: true });
函數(shù)應(yīng)該只做一件事
這是目前為止軟件工程中最重要的規(guī)則。當(dāng)函數(shù)做不止一件事時(shí),它們就更難組合、測(cè)試和推理。可以將一個(gè)函數(shù)隔離為一個(gè)操作時(shí),就可以很容易地重構(gòu)它,代碼也會(huì)讀起來(lái)更清晰。
// 不好的寫(xiě)法 function emailClients(clients) { clients.forEach(client => { const clientRecord = database.lookup(client); if (clientRecord.isActive()) { email(client); } }); } // 好的寫(xiě)法 function emailActiveClients(clients) { clients.filter(isActiveClient).forEach(email); } function isActiveClient(client) { const clientRecord = database.lookup(client); return clientRecord.isActive(); }
函數(shù)名稱應(yīng)說(shuō)明其作用
// 不好的寫(xiě)法 function addToDate(date, month) { // ... } const date = new Date(); // 從函數(shù)名稱很難知道添加什么 addToDate(date, 1); // 好的寫(xiě)法 function addMonthToDate(month, date) { // ... } const date = new Date(); addMonthToDate(1, date);
函數(shù)應(yīng)該只有一個(gè)抽象層次
當(dāng)有一個(gè)以上的抽象層次函數(shù),意味該函數(shù)做得太多了,需要將函數(shù)拆分可以實(shí)現(xiàn)可重用性和更簡(jiǎn)單的測(cè)試。
// 不好的寫(xiě)法 function parseBetterJSAlternative(code) { const REGEXES = [ // ... ]; const statements = code.split(" "); const tokens = []; REGEXES.forEach(REGEX => { statements.forEach(statement => { // ... }); }); const ast = []; tokens.forEach(token => { // lex... }); ast.forEach(node => { // parse... }); } // 好的寫(xiě)法 function parseBetterJSAlternative(code) { const tokens = tokenize(code); const syntaxTree = parse(tokens); syntaxTree.forEach(node => { // parse... }); } function tokenize(code) { const REGEXES = [ // ... ]; const statements = code.split(" "); const tokens = []; REGEXES.forEach(REGEX => { statements.forEach(statement => { tokens.push(/* ... */); }); }); return tokens; } function parse(tokens) { const syntaxTree = []; tokens.forEach(token => { syntaxTree.push(/* ... */); }); return syntaxTree; }
刪除重復(fù)的代碼
盡量避免重復(fù)的代碼,重復(fù)的代碼是不好的,它意味著如果我們需要更改某些邏輯,要改很多地方。
通常,有重復(fù)的代碼,是因?yàn)橛袃蓚€(gè)或多個(gè)稍有不同的事物,它們有很多共同點(diǎn),但是它們之間的差異迫使我們編寫(xiě)兩個(gè)或多個(gè)獨(dú)立的函數(shù)來(lái)完成許多相同的事情。 刪除重復(fù)的代碼意味著創(chuàng)建一個(gè)僅用一個(gè)函數(shù)/模塊/類就可以處理這組不同事物的抽象。
獲得正確的抽象是至關(guān)重要的,這就是為什么我們應(yīng)該遵循類部分中列出的 「SOLID原則」。糟糕的抽象可能比重復(fù)的代碼更糟糕,所以要小心!說(shuō)了這么多,如果你能做一個(gè)好的抽象,那就去做吧!不要重復(fù)你自己,否則你會(huì)發(fā)現(xiàn)自己在任何時(shí)候想要改變一件事的時(shí)候都要更新多個(gè)地方。
「設(shè)計(jì)模式的六大原則有」
Single Responsibility Principle:?jiǎn)我宦氊?zé)原則
Open Closed Principle:開(kāi)閉原則
Liskov Substitution Principle:里氏替換原則
Law of Demeter:迪米特法則
Interface Segregation Principle:接口隔離原則
Dependence Inversion Principle:依賴倒置原則
把這六個(gè)原則的首字母聯(lián)合起來(lái)(兩個(gè) L 算做一個(gè))就是 SOLID (solid,穩(wěn)定的),其代表的含義就是這六個(gè)原則結(jié)合使用的好處:建立穩(wěn)定、靈活、健壯的設(shè)計(jì)。下面我們來(lái)分別看一下這六大設(shè)計(jì)原則。
「不好的寫(xiě)法」
function showDeveloperList(developers) { developers.forEach(developer => { const expectedSalary = developer.calculateExpectedSalary(); const experience = developer.getExperience(); const githubLink = developer.getGithubLink(); const data = { expectedSalary, experience, githubLink }; render(data); }); } function showManagerList(managers) { managers.forEach(manager => { const expectedSalary = manager.calculateExpectedSalary(); const experience = manager.getExperience(); const portfolio = manager.getMBAProjects(); const data = { expectedSalary, experience, portfolio }; render(data); }); }
「好的寫(xiě)法」
function showEmployeeList(employees) { employees.forEach(employee => { const expectedSalary = employee.calculateExpectedSalary(); const experience = employee.getExperience(); const data = { expectedSalary, experience }; switch (employee.type) { case "manager": data.portfolio = employee.getMBAProjects(); break; case "developer": data.githubLink = employee.getGithubLink(); break; } render(data); }); }
使用Object.assign設(shè)置默認(rèn)對(duì)象
「不好的寫(xiě)法」
const menuConfig = { title: null, body: "Bar", buttonText: null, cancellable: true }; function createMenu(config) { configconfig.title = config.title || "Foo"; configconfig.body = config.body || "Bar"; configconfig.buttonText = config.buttonText || "Baz"; configconfig.cancellable = config.cancellable !== undefined ? config.cancellable : true; } createMenu(menuConfig);
「好的寫(xiě)法」
const menuConfig = { title: "Order", // User did not include 'body' key buttonText: "Send", cancellable: true }; function createMenu(config) { config = Object.assign( { title: "Foo", body: "Bar", buttonText: "Baz", cancellable: true }, config ); // config now equals: {title: "Order", body: "Bar", buttonText: "Send", cancellable: true} // ... } createMenu(menuConfig);
不要使用標(biāo)志作為函數(shù)參數(shù)
標(biāo)志告訴使用者,此函數(shù)可以完成多項(xiàng)任務(wù),函數(shù)應(yīng)該做一件事。如果函數(shù)遵循基于布爾的不同代碼路徑,請(qǐng)拆分它們。
// 不好的寫(xiě)法 function createFile(name, temp) { if (temp) { fs.create(`./temp/${name}`); } else { fs.create(name); } } // 好的寫(xiě)法 function createFile(name) { fs.create(name); } function createTempFile(name) { createFile(`./temp/${name}`); }
避免副作用(第一部分)
如果函數(shù)除了接受一個(gè)值并返回另一個(gè)值或多個(gè)值以外,不執(zhí)行任何其他操作,都會(huì)產(chǎn)生副作用。副作用可能是寫(xiě)入文件,修改某些全局變量,或者不小心將你的所有資金都匯給了陌生人。
「不好的寫(xiě)法」
let name = "Ryan McDermott"; function splitIntoFirstAndLastName() { namename = name.split(" "); } splitIntoFirstAndLastName(); console.log(name); // ['Ryan', 'McDermott'];
「好的寫(xiě)法」
function splitIntoFirstAndLastName(name) { return name.split(" "); } const name = "Ryan McDermott"; const newName = splitIntoFirstAndLastName(name); console.log(name); // 'Ryan McDermott'; console.log(newName); // ['Ryan', 'McDermott'];
避免副作用(第二部分)
在JavaScript中,原始類型值是按值傳遞,而對(duì)象/數(shù)組按引用傳遞。對(duì)于對(duì)象和數(shù)組,如果有函數(shù)在購(gòu)物車數(shù)組中進(jìn)行了更改(例如,通過(guò)添加要購(gòu)買的商品),則使用該購(gòu)物車數(shù)組的任何其他函數(shù)都將受到此添加的影響。那可能很棒,但是也可能不好。來(lái)想象一個(gè)糟糕的情況:
用戶單擊“購(gòu)買”按鈕,該按鈕調(diào)用一個(gè)purchase 函數(shù),接著,該函數(shù)發(fā)出一個(gè)網(wǎng)絡(luò)請(qǐng)求并將cart數(shù)組發(fā)送到服務(wù)器。由于網(wǎng)絡(luò)連接不好,purchase函數(shù)必須不斷重試請(qǐng)求。現(xiàn)在,如果在網(wǎng)絡(luò)請(qǐng)求開(kāi)始之前,用戶不小心點(diǎn)擊了他們實(shí)際上不需要的項(xiàng)目上的“添加到購(gòu)物車”按鈕,該怎么辦?如果發(fā)生這種情況,并且網(wǎng)絡(luò)請(qǐng)求開(kāi)始,那么購(gòu)買函數(shù)將發(fā)送意外添加的商品,因?yàn)樗幸粋€(gè)對(duì)購(gòu)物車數(shù)組的引用,addItemToCart函數(shù)通過(guò)添加修改了這個(gè)購(gòu)物車數(shù)組。
一個(gè)很好的解決方案是addItemToCart總是克隆cart數(shù)組,編輯它,然后返回克隆。這可以確保購(gòu)物車引用的其他函數(shù)不會(huì)受到任何更改的影響。
關(guān)于這種方法有兩點(diǎn)需要注意:
可能在某些情況下,我們確實(shí)需要修改輸入對(duì)象,但是當(dāng)我們采用這種編程實(shí)踐時(shí),會(huì)發(fā)現(xiàn)這種情況非常少見(jiàn),大多數(shù)東西都可以被改造成沒(méi)有副作用。
就性能而言,克隆大對(duì)象可能會(huì)非常昂貴。幸運(yùn)的是,在實(shí)踐中這并不是一個(gè)大問(wèn)題,因?yàn)橛泻芏嗪馨舻膸?kù)使這種編程方法能夠快速進(jìn)行,并且不像手動(dòng)克隆對(duì)象和數(shù)組那樣占用大量?jī)?nèi)存。
// 不好的寫(xiě)法 const addItemToCart = (cart, item) => { cart.push({ item, date: Date.now() }); }; // 好的寫(xiě)法 const addItemToCart = (cart, item) => { return [...cart, { item, date: Date.now() }]; };
不要寫(xiě)全局函數(shù)
污染全局變量在 JS 中是一種不好的做法,因?yàn)榭赡軙?huì)與另一個(gè)庫(kù)發(fā)生沖突,并且在他們的生產(chǎn)中遇到異常之前,API 的用戶將毫無(wú)用處。讓我們考慮一個(gè)示例:如果想擴(kuò)展 JS 的原生Array方法以具有可以顯示兩個(gè)數(shù)組之間差異的diff方法,該怎么辦?可以將新函數(shù)寫(xiě)入Array.prototype,但它可能與另一個(gè)嘗試執(zhí)行相同操作的庫(kù)發(fā)生沖突。如果其他庫(kù)僅使用diff來(lái)查找數(shù)組的第一個(gè)元素和最后一個(gè)元素之間的區(qū)別怎么辦?這就是為什么只使用 ES6 類并簡(jiǎn)單地?cái)U(kuò)展Array全局會(huì)更好的原因。
// 不好的寫(xiě)法 Array.prototype.diff = function diff(comparisonArray) { const hash = new Set(comparisonArray); return this.filter(elem => !hash.has(elem)); }; // 好的寫(xiě)法 class SuperArray extends Array { diff(comparisonArray) { const hash = new Set(comparisonArray); return this.filter(elem => !hash.has(elem)); } }
盡量使用函數(shù)式編程而非命令式
JavaScript不像Haskell那樣是一種函數(shù)式語(yǔ)言,但它具有函數(shù)式的風(fēng)格。函數(shù)式語(yǔ)言可以更簡(jiǎn)潔、更容易測(cè)試。如果可以的話,盡量喜歡這種編程風(fēng)格。
「不好的寫(xiě)法」
const programmerOutput = [ { name: "Uncle Bobby", linesOfCode: 500 }, { name: "Suzie Q", linesOfCode: 1500 }, { name: "Jimmy Gosling", linesOfCode: 150 }, { name: "Gracie Hopper", linesOfCode: 1000 } ]; let totalOutput = 0; for (let i = 0; i < programmerOutput.length; i++) { totalOutput += programmerOutput[i].linesOfCode; }
「好的寫(xiě)法」
const programmerOutput = [ { name: "Uncle Bobby", linesOfCode: 500 }, { name: "Suzie Q", linesOfCode: 1500 }, { name: "Jimmy Gosling", linesOfCode: 150 }, { name: "Gracie Hopper", linesOfCode: 1000 } ]; const totalOutput = programmerOutput.reduce( (totalLines, output) => totalLines + output.linesOfCode, 0 );
封裝條件
// 不好的寫(xiě)法 if (fsm.state === "fetching" && isEmpty(listNode)) { // ... } // 好的寫(xiě)法 function shouldShowSpinner(fsm, listNode) { return fsm.state === "fetching" && isEmpty(listNode); } if (shouldShowSpinner(fsmInstance, listNodeInstance)) { // ... }
避免使用非條件
// 不好的寫(xiě)法 function isDOMNodeNotPresent(node) { // ... } if (!isDOMNodeNotPresent(node)) { // ... } // 好的寫(xiě)法 function isDOMNodePresent(node) { // ... } if (isDOMNodePresent(node)) { // ... }
避免使用過(guò)多條件
這似乎是一個(gè)不可能完成的任務(wù)。一聽(tīng)到這個(gè),大多數(shù)人會(huì)說(shuō),“沒(méi)有if語(yǔ)句,我怎么能做任何事情呢?”答案是,你可以在許多情況下使用多態(tài)性來(lái)實(shí)現(xiàn)相同的任務(wù)。
第二個(gè)問(wèn)題通常是,“那很好,但是我為什么要那樣做呢?”答案是上面講過(guò)一個(gè)概念:一個(gè)函數(shù)應(yīng)該只做一件事。當(dāng)具有if語(yǔ)句的類和函數(shù)時(shí),這是在告訴你的使用者該函數(shù)執(zhí)行不止一件事情。
「不好的寫(xiě)法」
class Airplane { // ... getCruisingAltitude() { switch (this.type) { case "777": return this.getMaxAltitude() - this.getPassengerCount(); case "Air Force One": return this.getMaxAltitude(); case "Cessna": return this.getMaxAltitude() - this.getFuelExpenditure(); } } }
「好的寫(xiě)法」
class Airplane { // ... } class Boeing777 extends Airplane { // ... getCruisingAltitude() { return this.getMaxAltitude() - this.getPassengerCount(); } } class AirForceOne extends Airplane { // ... getCruisingAltitude() { return this.getMaxAltitude(); } } class Cessna extends Airplane { // ... getCruisingAltitude() { return this.getMaxAltitude() - this.getFuelExpenditure(); } }
避免類型檢查
JavaScript 是無(wú)類型的,這意味著函數(shù)可以接受任何類型的參數(shù)。有時(shí)q我們會(huì)被這種自由所困擾,并且很想在函數(shù)中進(jìn)行類型檢查。有很多方法可以避免這樣做。首先要考慮的是一致的API。
// 不好的寫(xiě)法 function travelToTexas(vehicle) { if (vehicle instanceof Bicycle) { vehicle.pedal(this.currentLocation, new Location("texas")); } else if (vehicle instanceof Car) { vehicle.drive(this.currentLocation, new Location("texas")); } } // 好的寫(xiě)法 function travelToTexas(vehicle) { vehicle.move(this.currentLocation, new Location("texas")); }
不要過(guò)度優(yōu)化
現(xiàn)代瀏覽器在運(yùn)行時(shí)做了大量的優(yōu)化工作。很多時(shí)候,如果你在優(yōu)化,那么你只是在浪費(fèi)時(shí)間。有很好的資源可以查看哪里缺乏優(yōu)化,我們只需要針對(duì)需要優(yōu)化的地方就行了。
// 不好的寫(xiě)法 // 在舊的瀏覽器上,每一次使用無(wú)緩存“l(fā)ist.length”的迭代都是很昂貴的 // 會(huì)為“l(fā)ist.length”重新計(jì)算。在現(xiàn)代瀏覽器中,這是經(jīng)過(guò)優(yōu)化的 for (let i = 0, len = list.length; i < len; i++) { // ... } // 好的寫(xiě)法 for (let i = 0; i < list.length; i++) { // ... }
感謝各位的閱讀,以上就是“如何正確使用JS的變量和函數(shù)”的內(nèi)容了,經(jīng)過(guò)本文的學(xué)習(xí)后,相信大家對(duì)如何正確使用JS的變量和函數(shù)這一問(wèn)題有了更深刻的體會(huì),具體使用情況還需要大家實(shí)踐驗(yàn)證。這里是創(chuàng)新互聯(lián),小編將為大家推送更多相關(guān)知識(shí)點(diǎn)的文章,歡迎關(guān)注!
新聞名稱:如何正確使用JS的變量和函數(shù)
本文地址:http://m.newbst.com/article10/gdcpgo.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供營(yíng)銷型網(wǎng)站建設(shè)、用戶體驗(yàn)、網(wǎng)站維護(hù)、ChatGPT、自適應(yīng)網(wǎng)站、定制網(wǎng)站
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如需處理請(qǐng)聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來(lái)源: 創(chuàng)新互聯(lián)