JavaScript設(shè)計(jì)模式之策略模式實(shí)現(xiàn)原理詳解
俗話說(shuō),條條大路通羅馬。在現(xiàn)實(shí)生活中,我們可以采用很多方法實(shí)現(xiàn)同一個(gè)目標(biāo)。比如我們先定個(gè)小目標(biāo),先掙它一個(gè)億。我們可以根據(jù)具體的實(shí)際情況來(lái)完成這個(gè)目標(biāo)。
策略模式的定義
定義一系列的算法,把它們一個(gè)個(gè)封裝起來(lái),并且使它們可以相互替換。
舉個(gè)例子:表單校驗(yàn)
在一個(gè)Web項(xiàng)目中,注冊(cè)、登錄等功能的實(shí)現(xiàn)都離不開(kāi)表單提交。表單校驗(yàn)也是前端常常需要做的事。假設(shè)我們正在編寫一個(gè)注冊(cè)的頁(yè)面,在點(diǎn)擊提交按鈕之前,有如下幾條校驗(yàn)邏輯:
- 用戶名不可為空,不允許以空白字符命名,用戶名長(zhǎng)度不能小于2位。
- 密碼長(zhǎng)度不能小于6位。
- 正確的手機(jī)號(hào)碼格式。
也許,一開(kāi)始我們會(huì)這么寫:
<html>
<head>
<title>策略模式-校驗(yàn)表單</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">
手機(jī)號(hào)碼:<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('用戶名用戶名長(zhǎng)度不能小于2位');
return false;
}
if (registerForm.password.value.trim().length < 6) {
alert('密碼長(zhǎng)度不能小于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('請(qǐng)輸入正確的手機(jī)號(hào)碼格式');
return errorMsg;
}
}
</script>
</body>
</html>
這是一種很常見(jiàn)的編碼方式,但它有很明顯的缺點(diǎn):
- registerForm.onsubmit 函數(shù)比較龐大,包含了很多if語(yǔ)句,這些語(yǔ)句要覆蓋所有的校驗(yàn)規(guī)則。
- 若校驗(yàn)規(guī)則有變,不得不深入到registerForm.onsubmit 函數(shù)的內(nèi)部實(shí)現(xiàn),違反開(kāi)放-封閉原則。
- 算法的復(fù)用性差。
下面,讓我們來(lái)用策略模式重構(gòu)表單校驗(yàn)
策略模式:表單校驗(yàn)
<html>
<head>
<title>策略模式-校驗(yàn)表單</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">
手機(jī)號(hào)碼:<input type="text" name="phoneNumber">
<button type="submit">提交</button>
</form>
<script type="text/javascript">
// 策略對(duì)象
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;
}
}
}
// 驗(yàn)證類
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: '用戶名長(zhǎng)度不能小于2位'
}]);
validator.add(registerForm.password, [ {
strategy: 'minLength:6',
errorMsg: '密碼長(zhǎng)度不能小于6位'
}]);
validator.add(registerForm.phoneNumber, [{
strategy: 'isMobile',
errorMsg: '請(qǐng)輸入正確的手機(jī)號(hào)碼格式'
}]);
var errorMsg = validator.start();
return errorMsg;
}
registerForm.onsubmit = function() {
var errorMsg = validataFunc();
if (errorMsg) {
alert(errorMsg);
return false;
}
}
</script>
</body>
</html>
策略模式優(yōu)缺點(diǎn)
策略模式是一種常用且有效的設(shè)計(jì)模式。
- 策略模式可以有效避免多重條件選擇語(yǔ)句。
- 策略模式提供了對(duì)開(kāi)放-封裝原則的完美支持,將方法封裝在獨(dú)立的strategy中,使得它們易于切換,易于理解,易于擴(kuò)展。
- 復(fù)用性高。
當(dāng)然,策略模式也有一些缺點(diǎn)
- 增加了許多策略類或者策略對(duì)象。
- 要使用策略模式,必須了解所有的strategy,違反了最少知識(shí)原則。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
JS實(shí)現(xiàn)密碼框的顯示密碼和隱藏密碼功能示例
這篇文章主要介紹了JS實(shí)現(xiàn)密碼框的顯示密碼和隱藏密碼功能,涉及javascript針對(duì)頁(yè)面form表單元素動(dòng)態(tài)操作的相關(guān)技巧,需要的朋友可以參考下2016-12-12
Javascript實(shí)現(xiàn)基本運(yùn)算器
這篇文章主要為大家詳細(xì)介紹了Javascript實(shí)現(xiàn)基本運(yùn)算器的相關(guān)代碼,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-07-07
js代碼延遲一定時(shí)間后執(zhí)行一個(gè)函數(shù)的實(shí)例
下面小編就為大家?guī)?lái)一篇js代碼延遲一定時(shí)間后執(zhí)行一個(gè)函數(shù)的實(shí)例。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-02-02
JavaScript 計(jì)算圖片加載數(shù)量的代碼
先定義一個(gè)圖片的數(shù)組,然后通過(guò)image的onload事件來(lái)計(jì)算,注意,onload在ie和火狐有所不同。2011-01-01
JavaScript實(shí)現(xiàn)離開(kāi)頁(yè)面前提示功能【附j(luò)Query實(shí)現(xiàn)方法】
這篇文章主要介紹了JavaScript實(shí)現(xiàn)離開(kāi)頁(yè)面前提示功能,結(jié)合具體實(shí)例形式分析了javascript實(shí)現(xiàn)針對(duì)關(guān)閉頁(yè)面的事件響應(yīng)原理與操作技巧,并附帶jQuery的相應(yīng)實(shí)現(xiàn)方法,需要的朋友可以參考下2017-09-09
詳解TypeScript中的箭頭函數(shù)如何實(shí)現(xiàn)重載
這篇文章主要為大家詳細(xì)介紹了TypeScript中的箭頭函數(shù)是如何實(shí)現(xiàn)重載的,文中的示例代碼講解詳細(xì),具有一定的參考價(jià)值,需要的可以參考一下2023-05-05

