優雅的類寫法

来源:https://www.cnblogs.com/walls/archive/2018/07/20/9339300.html
-Advertisement-
Play Games

前言 雖然現在已經是ES6的時代,但是,還是有必要瞭解下ES5是怎麼寫一個類的。 本文詳述JavaScript面向對象編程中的類寫法,並分步驟講述如何寫出優雅的類。 一、例子 例子為一個輕提示組件 。 需要實現的功能: 方法,顯示提示 方法,隱藏提示 方法,初始化提示語 二、類的構成 JavaScr ...


前言

雖然現在已經是ES6的時代,但是,還是有必要瞭解下ES5是怎麼寫一個類的。
本文詳述JavaScript面向對象編程中的類寫法,並分步驟講述如何寫出優雅的類。

一、例子

例子為一個輕提示組件Toast
需要實現的功能:

  • on方法,顯示提示
  • off方法,隱藏提示
  • init方法,初始化提示語
function Toast(option){
  this.prompt = '';
  this.elem = null;
  this.init(option);
}

Toast.prototype = {
  // 構造器
  constructor: Toast,
  // 初始化方法
  init: function(option){
    this.prompt = option.prompt || '';
    this.render();
    this.bindEvent();
  },
  // 顯示
  show: function(){
    this.changeStyle(this.elem, 'display', 'block');
  },
  // 隱藏
  hide: function(){
    this.changeStyle(this.elem, 'display', 'none');
  },
  // 畫出dom
  render: function(){
    var html = '';
    this.elem = document.createElement('div');
    this.changeStyle(this.elem, 'display', 'none');

    html += '<a class="J-close" href="javascript:;">x</a>'
    html += '<p>'+ this.prompt +'</p>';
    
    this.elem.innerHTML = html;

    return document.body.appendChild(this.elem);
  },
  // 綁定事件
  bindEvent: function(){
    var self = this;
    
    this.addEvent(this.elem, 'click', function(e){
      if(e.target.className.indexOf('J-close') != -1){
        console.log('close Toast!');
        self.hide();
      }
    });
  },
  // 添加事件方法
  addEvent: function(node, name, fn){
    var self = this;
    
    node.addEventListener(name, function(){
      fn.apply(self, Array.prototype.slice.call(arguments));
    }, false);
  },
  // 改變樣式
  changeStyle: function(node, key, value){
      node.style[key] = value;
  }
};

var T = new Toast({prompt:'I\'m Toast!'});
T.show();

二、類的構成

JavaScript的類,是用函數對象來實現。
類的實例化形式如下:

var T = new Toast();

其中的重點,就是Function的編寫。

類分為兩部分:constructor+prototype。也即構造器+原型

2.1 構造器

構造器從直觀上來理解,就是寫在函數內部的代碼
從Toast例子上看,構造器就是以下部分:

function Toast(option){
  this.prompt = '';
  this.elem = null;
  this.init(option);
}

這裡的this,指向的是實例化的類。
每次通過new Toast()的方式進行實例化,構造器都會執行一遍

2.2 原型

原型上的方法和變數的聲明,都是通過Toast.prototype.*的方式。
那麼在原型上普通的寫法如下:

Toast.prototype.hide = function(){/*code*/}
Toast.prototype.myValue = 1;

但是,該寫法不好的地方:就是每次都要寫前半部分Toast.prorotype,略顯累贅。
在代碼壓縮優化方面也不友好,無法做到最佳的壓縮。
改進的方式如下:

Toast.prorotype = {
  constructor: Toast,
  hide: function(){/*code*/},
  myValue: 1 
}

這裡的優化,是把原型指向一個新的空對象{}
帶來的好處,就是可以用{key:value}的方式寫原型上的方法和變數。
但是,這種方式會改變原型上構造器prototype.constructor的指向。
如果不重新顯式聲明constructor的指向,Toast.constructor.prototype.constructor的會隱式被指向Object。而正確的指向,應該是Toast
雖然通過new實例化沒有出現異常,但是在類繼承方面,constructor的指向異常,會產生不正確的繼承判斷結果。這是我們不希望看到的。
所以,需要修正constructor

2.3 構造器和原型的不同

原型上的方法和變數,是該類所有實例化對象共用的。也就是說,只有一份。
而構造器內的代碼塊,則是每個實例化對象單獨占有。不管是否用this.**方式,還是私有變數的方式,都是獨占的。
所以,在寫一個類的時候,需要考慮該新增屬性是共用的,還是獨占的。以此,決定在構造器還是原型上進行聲明。

三、代碼規範

  • 類的命名規範,業界有不成文的規定,就是首字母大寫。
  • 原型上的私有方法,預設以下劃線開始。這種只是團隊合作方面有review代碼的好處,實際上還是暴露出來的方法。

四、使實例化與new無關

類的實例化,一個強制要求的行為,就是需要使用new操作符。如果不使用new操作符,那麼構造器內的this指向,將不是當前的實例化對象。
優化的方式,就是使用instanceof做一層防護。

function Toast(option){
  if(!(this instanceof Toast)){
    return new Toast(option);
  }
  
  this.prompt = '';
  this.elem = null;
  this.init(option);
}

從上述代碼可以看出,使用這個技巧,可以防止團隊一些大頭蝦出現使用錯誤實例化方式,導致代碼污染的問題。
這種忍者技巧很酷,但從另一方面考慮,還是希望使用者可以用正確的方式去實例化類。
所以,改成以下這種防護方式

function Toast(option){
  if(!(this instanceof Toast)){
    throw new Error('Toast instantiation error');
  }
  
  this.prompt = '';
  this.elem = null;
  this.init(option);
}

這樣,把鍋甩回去,豈不是更妙

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 前言 ​ 本文是基於《第一行代碼》整理成的筆記,mark下自己的學習路程。 ​ 本章將通過實驗著重介紹ACtivity。 項目結構 ​ 首先我們創建一個安卓項目,來認識一下項目結構; 四大組件之首——Activity Activity 就是我們所看到的界面,新建一個類,繼承Activity,重寫on ...
  • 1.將字體加入到項目中 2.在info.plist文件中加入相應信息,這一步實際上實在項目的Info頁裡面增加Fonts provided by application項,並設置相應的ttf文件進去,這樣就告訴了應用,我要加入新的字體了,對應的字體文件是哪些。 3.字型檔文件應該是準備好了,下麵我們需 ...
  • iOS 判斷數字 - (BOOL) deptNumInputShouldNumber:(NSString *)str { if (str.length == 0) { return NO; } NSString *regex = @"[0-9]*"; NSPredicate *pred = [NSP ...
  • 1.首先看效果圖 2.自定義PieChartView,繼承自View,下邊為PieChartView代碼 之後,在activity中,只需找到組件,傳入數據,調用 invalidate() 進行重繪即可。 ...
  • 使用 ConstraintLayout 佈局出現警告: 此視圖不受垂直約束。在運行時,除非添加垂直約束,否則它將跳轉到左側 解決辦法: 從Android Studio v3及更高版本開始,從下拉列表中刪除了Infer Constraint。 使用設計預覽上方工具欄菜單中的魔棒圖標 ; 有“ 推斷約束 ...
  • 編譯 Luajit 庫,的確是一個挑戰。因為官網的教程,在當前版本的 Xcode 和 NDK 環境中,已經不適用了。以前只是編譯了適用於真機的 Luajit 庫。最近在嘗試編譯模擬器 Luajit 庫,就順便梳理了下 Luajit 庫的編譯經驗,供以後查閱。網上的討論也是有一些,但是相當一部分都已經... ...
  • 在我們日常的項目中,合理的使用巨集定義,會大大減少我們的代碼量,以及代碼的可讀性,為方便讀者使用,總結如下: pragma mark - Application相關 /// /// @name Application /// #define APPLICATION [UIApplication sha ...
  • 動態生成HTML元素的方法有三種: 第一種:document.createElement()創建元素,再用appendChild( )方法將元素添加到指定節點 第二種:使用innerHTML直接將元素添加到指定節點 第三種:jQuery創建節點 jQuery中創建DOM對象,使用jQuery的工廠函 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...