溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

JavaScript設(shè)計模式之策略模式的實現(xiàn)方法

發(fā)布時間:2020-07-21 17:19:56 來源:億速云 閱讀:162 作者:小豬 欄目:web開發(fā)

這篇文章主要講解了JavaScript設(shè)計模式之策略模式的實現(xiàn)方法,內(nèi)容清晰明了,對此有興趣的小伙伴可以學習一下,相信大家閱讀完之后會有幫助。

策略模式的定義

定義一系列的算法,把它們一個個封裝起來,并且使它們可以相互替換。

舉個例子:表單校驗

在一個Web項目中,注冊、登錄等功能的實現(xiàn)都離不開表單提交。表單校驗也是前端常常需要做的事。假設(shè)我們正在編寫一個注冊的頁面,在點擊提交按鈕之前,有如下幾條校驗邏輯:

  • 用戶名不可為空,不允許以空白字符命名,用戶名長度不能小于2位。
  • 密碼長度不能小于6位。
  • 正確的手機號碼格式。

也許,一開始我們會這么寫:

<html>
<head>
  <title>策略模式-校驗表單</title>
  <meta content="text/html; charset=utf-8" http-equiv="Content-Type">
</head>
<body>
  <form id = "registerForm" method="post" action="http://xxxx.com/api/register">
    用戶名:<input type="text" name="userName">
    密碼:<input type="text" name="password">
    手機號碼:<input type="text" name="phoneNumber">
    <button type="submit">提交</button>
  </form>
  <script type="text/javascript">
    var registerForm = document.getElementById('registerForm');
    registerForm.onsubmit = function() {
      if (registerForm.userName.value === '') {
        alert('用戶名不可為空');
        return false;
      }
      if (registerForm.userName.value === '') {
        alert('用戶名不可為空');
        return false;
      } 
      if (registerForm.userName.value.trim() === '') {
        alert('用戶名不允許以空白字符命名');
        return false;
      } 
      if (registerForm.userName.value.trim().length < 2) {
        alert('用戶名用戶名長度不能小于2位');
        return false;
      } 
      if (registerForm.password.value.trim().length < 6) {
        alert('密碼長度不能小于6位');
        return false;
      }
      if (!/^(13[0-9]|14[5|7]|15[0|1|2|3|5|6|7|8|9]|17[7]|18[0|1|2|3|5|6|7|8|9])\d{8}$/.test(registerForm.phoneNumber.value)) {
        alert('請輸入正確的手機號碼格式');
        return errorMsg;
       } 
    }
  </script>
</body>
</html>

這是一種很常見的編碼方式,但它有很明顯的缺點:

  • registerForm.onsubmit 函數(shù)比較龐大,包含了很多if語句,這些語句要覆蓋所有的校驗規(guī)則。
  • 若校驗規(guī)則有變,不得不深入到registerForm.onsubmit 函數(shù)的內(nèi)部實現(xiàn),違反開放-封閉原則。
  • 算法的復(fù)用性差。

下面,讓我們來用策略模式重構(gòu)表單校驗

策略模式:表單校驗

<html>
<head>
  <title>策略模式-校驗表單</title>
  <meta content="text/html; charset=utf-8" http-equiv="Content-Type">
</head>
<body>
  <form id = "registerForm" method="post" action="http://xxxx.com/api/register">
    用戶名:<input type="text" name="userName">
    密碼:<input type="text" name="password">
    手機號碼:<input type="text" name="phoneNumber">
    <button type="submit">提交</button>
  </form>
  <script type="text/javascript">
    // 策略對象
    var strategies = {
      isNoEmpty: function (value, errorMsg) {
        if (value === '') {
          return errorMsg;
        }
      },
      isNoSpace: function (value, errorMsg) {
        if (value.trim() === '') {
          return errorMsg;
        }
      },
      minLength: function (value, length, errorMsg) {
        if (value.trim().length < length) {
          return errorMsg;
        }
      },
      maxLength: function (value, length, errorMsg) {
        if (value.length > length) {
          return errorMsg;
        }
      },
      isMobile: function (value, errorMsg) {
        if (!/^(13[0-9]|14[5|7]|15[0|1|2|3|5|6|7|8|9]|17[7]|18[0|1|2|3|5|6|7|8|9])\d{8}$/.test(value)) {
          return errorMsg;
        }        
      }
    }

    // 驗證類
    var Validator = function() {
      this.cache = [];
    }
    Validator.prototype.add = function(dom, rules) {
      var self = this;
      for(var i = 0, rule; rule = rules[i++];) {
        (function(rule) {
          var strategyAry = rule.strategy.split(':');
          var errorMsg = rule.errorMsg;
          self.cache.push(function() {
          var strategy = strategyAry.shift();
          strategyAry.unshift(dom.value);
          strategyAry.push(errorMsg);
          return strategies[strategy].apply(dom, strategyAry);
          })
        })(rule)
      }
    };
    Validator.prototype.start = function() {
      for(var i = 0, validatorFunc; validatorFunc = this.cache[i++];) {
        var errorMsg = validatorFunc();
        if (errorMsg) {
          return errorMsg;
        }
      }
    };

    // 調(diào)用代碼
    var registerForm = document.getElementById('registerForm');

    var validataFunc = function() {
      var validator = new Validator();
      validator.add(registerForm.userName, [{
        strategy: 'isNoEmpty',
        errorMsg: '用戶名不可為空'
      }, {
        strategy: 'isNoSpace',
        errorMsg: '不允許以空白字符命名'
      }, {
        strategy: 'minLength:2',
        errorMsg: '用戶名長度不能小于2位'
      }]);
      validator.add(registerForm.password, [ {
        strategy: 'minLength:6',
        errorMsg: '密碼長度不能小于6位'
      }]);
      validator.add(registerForm.phoneNumber, [{
        strategy: 'isMobile',
        errorMsg: '請輸入正確的手機號碼格式'
      }]);
      var errorMsg = validator.start();
      return errorMsg;
    }

    registerForm.onsubmit = function() {
      var errorMsg = validataFunc();
      if (errorMsg) {
        alert(errorMsg);
        return false;
      }
    }
  </script>
</body>
</html>

策略模式優(yōu)缺點

策略模式是一種常用且有效的設(shè)計模式。

  • 策略模式可以有效避免多重條件選擇語句。
  • 策略模式提供了對開放-封裝原則的完美支持,將方法封裝在獨立的strategy中,使得它們易于切換,易于理解,易于擴展。
  • 復(fù)用性高。

當然,策略模式也有一些缺點

  • 增加了許多策略類或者策略對象。
  • 要使用策略模式,必須了解所有的strategy,違反了最少知識原則。

看完上述內(nèi)容,是不是對JavaScript設(shè)計模式之策略模式的實現(xiàn)方法有進一步的了解,如果還想學習更多內(nèi)容,歡迎關(guān)注億速云行業(yè)資訊頻道。

向AI問一下細節(jié)

免責聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI