电脑知识|欧美黑人一区二区三区|软件|欧美黑人一级爽快片淫片高清|系统|欧美黑人狂野猛交老妇|数据库|服务器|编程开发|网络运营|知识问答|技术教程文章 - 好吧啦网

您的位置:首頁技術(shù)文章
文章詳情頁

如何寫出優(yōu)雅的JS 代碼

瀏覽:94日期:2024-03-27 18:42:32
變量使用有意義和可發(fā)音的變量名

// 不好的寫法const yyyymmdstr = moment().format('YYYY/MM/DD');// 好的寫法const currentDate = moment().format('YYYY/MM/DD');對(duì)同一類型的變量使用相同的詞匯

// 不好的寫法getUserInfo();getClientData();getCustomerRecord();// 好的寫法getUser();使用可搜索的名字

我們讀的會(huì)比我們寫的多得多,所以如果命名太過隨意不僅會(huì)給后續(xù)的維護(hù)帶來困難,也會(huì)傷害了讀我們代碼的開發(fā)者。讓你的變量名可被讀取,像buddy.js和ESLint這樣的工具可以幫助識(shí)別未命名的常量。

// 不好的寫法// 86400000 的用途是什么?setTimeout(blastOff, 86400000);// 好的寫法const MILLISECONDS_IN_A_DAY = 86_400_000;setTimeout(blastOff, MILLISECONDS_IN_A_DAY);使用解釋性變量

// 不好的寫法const address = 'One Infinite Loop, Cupertino 95014';const cityZipCodeRegex = /^[^,]+[,s]+(.+?)s*(d{5})?$/;saveCityZipCode( address.match(cityZipCodeRegex)[1], address.match(cityZipCodeRegex)[2]);// 好的寫法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è)

顯式用于隱式

// 不好的寫法const locations = ['Austin', 'New York', 'San Francisco'];locations.forEach(l => { doStuff(); doSomeOtherStuff(); // ... // ... // ... // 等等,“l(fā)”又是什么? dispatch(l);// 好的寫法const locations = ['Austin', 'New York', 'San Francisco'];locations.forEach(location => { doStuff(); doSomeOtherStuff(); // ... // ... // ... dispatch(location);});無需添加不必要的上下文

如果類名/對(duì)象名已經(jīng)說明了,就無需在變量名中重復(fù)。

// 不好的寫法const Car = { carMake: 'Honda', carModel: 'Accord', carColor: 'Blue'};function paintCar(car) { car.carColor = 'Red';}// 好的寫法const Car = { make: 'Honda', model: 'Accord', color: 'Blue'};function paintCar(car) { car.color = 'Red';}使用默認(rèn)參數(shù)代替邏輯或(與)運(yùn)算

// 不好的寫法function createMicrobrewery(name) { const breweryName = name || 'Hipster Brew Co.'; // ...}// 好的寫法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ì)象來代替。

// 不好的寫法function createMenu(title, body, buttonText, cancellable) { // ...}createMenu('Foo', 'Bar', 'Baz', true);// 好的寫法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ì)讀起來更清晰。

// 不好的寫法function emailClients(clients) { clients.forEach(client => { const clientRecord = database.lookup(client); if (clientRecord.isActive()) { email(client); } });}// 好的寫法function emailActiveClients(clients) { clients.filter(isActiveClient).forEach(email);}function isActiveClient(client) { const clientRecord = database.lookup(client); return clientRecord.isActive();}函數(shù)名稱應(yīng)說明其作用

// 不好的寫法function addToDate(date, month) { // ...}const date = new Date();// 從函數(shù)名稱很難知道添加什么addToDate(date, 1);// 好的寫法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è)試。

// 不好的寫法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... });}// 好的寫法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),但是它們之間的差異迫使我們編寫兩個(gè)或多個(gè)獨(dú)立的函數(shù)來完成許多相同的事情。 刪除重復(fù)的代碼意味著創(chuàng)建一個(gè)僅用一個(gè)函數(shù)/模塊/類就可以處理這組不同事物的抽象。

獲得正確的抽象是至關(guān)重要的,這就是為什么我們應(yīng)該遵循類部分中列出的SOLID原則。糟糕的抽象可能比重復(fù)的代碼更糟糕,所以要小心!說了這么多,如果你能做一個(gè)好的抽象,那就去做吧!不要重復(fù)你自己,否則你會(huì)發(fā)現(xiàn)自己在任何時(shí)候想要改變一件事的時(shí)候都要更新多個(gè)地方。

設(shè)計(jì)模式的六大原則有:

Single Responsibility Principle:?jiǎn)我宦氊?zé)原則 Open Closed Principle:開閉原則 Liskov Substitution Principle:里氏替換原則 Law of Demeter:迪米特法則 Interface Segregation Principle:接口隔離原則 Dependence Inversion Principle:依賴倒置原則

把這六個(gè)原則的首字母聯(lián)合起來(兩個(gè) L 算做一個(gè))就是 SOLID (solid,穩(wěn)定的),其代表的含義就是這六個(gè)原則結(jié)合使用的好處:建立穩(wěn)定、靈活、健壯的設(shè)計(jì)。下面我們來分別看一下這六大設(shè)計(jì)原則。

不好的寫法

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

好的寫法

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ì)象

不好的寫法

const menuConfig = { title: null, body: 'Bar', buttonText: null, cancellable: true};function createMenu(config) { config.title = config.title || 'Foo'; config.body = config.body || 'Bar'; config.buttonText = config.buttonText || 'Baz'; config.cancellable = config.cancellable !== undefined ? config.cancellable : true;}createMenu(menuConfig);

好的寫法

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)拆分它們。

// 不好的寫法function createFile(name, temp) { if (temp) { fs.create(`./temp/${name}`); } else { fs.create(name); }}// 好的寫法function createFile(name) { fs.create(name);}function createTempFile(name) { createFile(`./temp/${name}`);}避免副作用(第一部分)

如果函數(shù)除了接受一個(gè)值并返回另一個(gè)值或多個(gè)值以外,不執(zhí)行任何其他操作,都會(huì)產(chǎn)生副作用。 副作用可能是寫入文件,修改某些全局變量,或者不小心將你的所有資金都匯給了陌生人。

不好的寫法

let name = 'Ryan McDermott';function splitIntoFirstAndLastName() { name = name.split(' ');}splitIntoFirstAndLastName();console.log(name); // [’Ryan’, ’McDermott’];

好的寫法

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)行了更改(例如,通過添加要購(gòu)買的商品),則使用該購(gòu)物車數(shù)組的任何其他函數(shù)都將受到此添加的影響。 那可能很棒,但是也可能不好。 來想象一個(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)求?,F(xiàn)在,如果在網(wǎng)絡(luò)請(qǐng)求開始之前,用戶不小心點(diǎn)擊了他們實(shí)際上不需要的項(xiàng)目上的“添加到購(gòu)物車”按鈕,該怎么辦?如果發(fā)生這種情況,并且網(wǎng)絡(luò)請(qǐng)求開始,那么購(gòu)買函數(shù)將發(fā)送意外添加的商品,因?yàn)樗幸粋€(gè)對(duì)購(gòu)物車數(shù)組的引用,addItemToCart函數(shù)通過添加修改了這個(gè)購(gòu)物車數(shù)組。

一個(gè)很好的解決方案是addItemToCart總是克隆cart數(shù)組,編輯它,然后返回克隆。這可以確保購(gòu)物車引用的其他函數(shù)不會(huì)受到任何更改的影響。

關(guān)于這種方法有兩點(diǎn)需要注意:

1.可能在某些情況下,我們確實(shí)需要修改輸入對(duì)象,但是當(dāng)我們采用這種編程實(shí)踐時(shí),會(huì)發(fā)現(xiàn)這種情況非常少見,大多數(shù)東西都可以被改造成沒有副作用。

2.就性能而言,克隆大對(duì)象可能會(huì)非常昂貴。 幸運(yùn)的是,在實(shí)踐中這并不是一個(gè)大問題,因?yàn)橛泻芏嗪馨舻膸焓惯@種編程方法能夠快速進(jìn)行,并且不像手動(dòng)克隆對(duì)象和數(shù)組那樣占用大量?jī)?nèi)存。

// 不好的寫法const addItemToCart = (cart, item) => { cart.push({ item, date: Date.now() });};// 好的寫法const addItemToCart = (cart, item) => { return [...cart, { item, date: Date.now() }];};不要寫全局函數(shù)

污染全局變量在 JS 中是一種不好的做法,因?yàn)榭赡軙?huì)與另一個(gè)庫發(fā)生沖突,并且在他們的生產(chǎn)中遇到異常之前,API 的用戶將毫無用處。 讓我們考慮一個(gè)示例:如果想擴(kuò)展 JS 的原生Array方法以具有可以顯示兩個(gè)數(shù)組之間差異的diff方法,該怎么辦? 可以將新函數(shù)寫入Array.prototype,但它可能與另一個(gè)嘗試執(zhí)行相同操作的庫發(fā)生沖突。 如果其他庫僅使用diff來查找數(shù)組的第一個(gè)元素和最后一個(gè)元素之間的區(qū)別怎么辦? 這就是為什么只使用 ES6 類并簡(jiǎn)單地?cái)U(kuò)展Array全局會(huì)更好的原因。

// 不好的寫法Array.prototype.diff = function diff(comparisonArray) { const hash = new Set(comparisonArray); return this.filter(elem => !hash.has(elem));};// 好的寫法class SuperArray extends Array { diff(comparisonArray) { const hash = new Set(comparisonArray); return this.filter(elem => !hash.has(elem)); }}盡量使用函數(shù)式編程而非命令式

JavaScript不像Haskell那樣是一種函數(shù)式語言,但它具有函數(shù)式的風(fēng)格。函數(shù)式語言可以更簡(jiǎn)潔、更容易測(cè)試。如果可以的話,盡量喜歡這種編程風(fēng)格。

不好的寫法

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

好的寫法

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);封裝條件

// 不好的寫法if (fsm.state === 'fetching' && isEmpty(listNode)) { // ...}// 好的寫法function shouldShowSpinner(fsm, listNode) { return fsm.state === 'fetching' && isEmpty(listNode);}if (shouldShowSpinner(fsmInstance, listNodeInstance)) { // ...}避免使用非條件

// 不好的寫法function isDOMNodeNotPresent(node) { // ...}if (!isDOMNodeNotPresent(node)) { // ...}// 好的寫法function isDOMNodePresent(node) { // ...}if (isDOMNodePresent(node)) { // ...}避免使用過多條件

這似乎是一個(gè)不可能完成的任務(wù)。一聽到這個(gè),大多數(shù)人會(huì)說,“沒有if語句,我怎么能做任何事情呢?”答案是,你可以在許多情況下使用多態(tài)性來實(shí)現(xiàn)相同的任務(wù)。

第二個(gè)問題通常是,“那很好,但是我為什么要那樣做呢?”答案是上面講過一個(gè)概念:一個(gè)函數(shù)應(yīng)該只做一件事。當(dāng)具有if語句的類和函數(shù)時(shí),這是在告訴你的使用者該函數(shù)執(zhí)行不止一件事情。

不好的寫法

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

好的寫法

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 是無類型的,這意味著函數(shù)可以接受任何類型的參數(shù)。 有時(shí)q我們會(huì)被這種自由所困擾,并且很想在函數(shù)中進(jìn)行類型檢查。 有很多方法可以避免這樣做。 首先要考慮的是一致的API。

// 不好的寫法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')); }}// 好的寫法function travelToTexas(vehicle) { vehicle.move(this.currentLocation, new Location('texas'));}不要過度優(yōu)化

現(xiàn)代瀏覽器在運(yùn)行時(shí)做了大量的優(yōu)化工作。很多時(shí)候,如果你在優(yōu)化,那么你只是在浪費(fèi)時(shí)間。有很好的資源可以查看哪里缺乏優(yōu)化,我們只需要針對(duì)需要優(yōu)化的地方就行了。

// 不好的寫法// 在舊的瀏覽器上,每一次使用無緩存“l(fā)ist.length”的迭代都是很昂貴的// 會(huì)為“l(fā)ist.length”重新計(jì)算。在現(xiàn)代瀏覽器中,這是經(jīng)過優(yōu)化的for (let i = 0, len = list.length; i < len; i++) { // ...}// 好的寫法for (let i = 0; i < list.length; i++) { // ...}

以上就是如何寫出優(yōu)雅的JS 代碼的詳細(xì)內(nèi)容,更多關(guān)于優(yōu)雅的JS 代碼的資料請(qǐng)關(guān)注好吧啦網(wǎng)其它相關(guān)文章!

標(biāo)簽: JavaScript
相關(guān)文章:
主站蜘蛛池模板: 代做标书-代写标书-专业标书文件编辑-「深圳卓越创兴公司」 | 托盘租赁_塑料托盘租赁_托盘出租_栈板出租_青岛托盘租赁-优胜必达 | 菏泽商标注册_菏泽版权登记_商标申请代理_菏泽商标注册去哪里 | 环球电气之家-中国专业电气电子产品行业服务网站! | 温州中研白癜风专科_温州治疗白癜风_温州治疗白癜风医院哪家好_温州哪里治疗白癜风 | 青州搬家公司电话_青州搬家公司哪家好「鸿喜」青州搬家 | 齿轮减速马达一体式_蜗轮蜗杆减速机配电机-德国BOSERL齿轮减速电动机生产厂家 | 酵素生产厂家_酵素OEM_酵素加盟_酵素ODM_酵素原料厂家_厦门益力康 | 办公室家具公司_办公家具品牌厂家_森拉堡办公家具【官网】 | 上海防爆真空干燥箱-上海防爆冷库-上海防爆冷柜?-上海浦下防爆设备厂家? | 「钾冰晶石」氟铝酸钾_冰晶石_氟铝酸钠「价格用途」-亚铝氟化物厂家 | 高低温试验房-深圳高低温湿热箱-小型高低温冲击试验箱-爱佩试验设备 | 加中寰球移民官网-美国移民公司,移民机构,移民中介,移民咨询,投资移民 | 政府园区专业委托招商平台_助力企业选址项目快速落地_东方龙商务集团 | 西安展台设计搭建_西安活动策划公司_西安会议会场布置_西安展厅设计西安旭阳展览展示 | 隧道风机_DWEX边墙风机_SDS射流风机-绍兴市上虞科瑞风机有限公司 | 熔体泵|换网器|熔体齿轮泵|熔体计量泵厂家-郑州巴特熔体泵有限公司 | 混合气体腐蚀试验箱_盐雾/硫化氢/气体腐蚀试验箱厂家-北京中科博达 | 大学食堂装修设计_公司餐厅效果图_工厂食堂改造_迈普装饰 | 超声波焊接机,振动摩擦焊接机,激光塑料焊接机,超声波焊接模具工装-德召尼克(常州)焊接科技有限公司 | 土壤检测仪器_行星式球磨仪_土壤团粒分析仪厂家_山东莱恩德智能科技有限公司 | 东莞注册公司-代办营业执照-东莞公司注册代理记账-极刻财税 | 建筑消防设施检测系统检测箱-电梯**检测仪器箱-北京宇成伟业科技有限责任公司 | 水厂自动化-水厂控制系统-泵站自动化|控制系统-闸门自动化控制-济南华通中控科技有限公司 | 工业PH计|工业ph酸度计|在线PH计价格-合肥卓尔仪器仪表有限公司 济南画室培训-美术高考培训-山东艺霖艺术培训画室 | 车辆定位管理系统_汽车GPS系统_车载北斗系统 - 朗致物联 | 工业风机_环保空调_冷风机_工厂车间厂房通风降温设备旺成服务平台 | 北京包装设计_标志设计公司_包装设计公司-北京思逸品牌设计 | 理化生实验室设备,吊装实验室设备,顶装实验室设备,实验室成套设备厂家,校园功能室设备,智慧书法教室方案 - 东莞市惠森教学设备有限公司 | 首页|专注深圳注册公司,代理记账报税,注册商标代理,工商变更,企业400电话等企业一站式服务-慧用心 | 右手官网|右手工业设计|外观设计公司|工业设计公司|产品创新设计|医疗产品结构设计|EMC产品结构设计 | 佛山商标注册_商标注册代理|专利注册申请_商标注册公司_鸿邦知识产权 | 探鸣起名网-品牌起名-英文商标起名-公司命名-企业取名包满意 | 真丝围巾|真丝丝巾|羊绒围巾|围巾品牌|浙江越缇围巾厂家定制 | 预制舱-电力集装箱预制舱-模块化预制舱生产厂家-腾达电器设备 | 混合反应量热仪-高温高压量热仪-微机差热分析仪DTA|凯璞百科 | 阴离子_阳离子聚丙烯酰胺厂家_聚合氯化铝价格_水处理絮凝剂_巩义市江源净水材料有限公司 | 定做大型恒温循环水浴槽-工业用不锈钢恒温水箱-大容量低温恒温水槽-常州精达仪器 | 动物麻醉机-数显脑立体定位仪-北京易则佳科技有限公司 | 打造全球沸石生态圈 - 国投盛世 锂电混合机-新能源混合机-正极材料混料机-高镍,三元材料混料机-负极,包覆混合机-贝尔专业混合混料搅拌机械系统设备厂家 | 校车_校车价格_19座幼儿园校车_幼儿园校车_大鼻子校车 |