欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

怎樣用Javascript實現(xiàn)策略模式

 更新時間:2021年04月26日 09:00:02   作者:有夢想的咸魚前端  
這篇文章主要介紹了JavaScript設計模式之策略模式,想學習設計模式的同學,一定要看一下

概述

策略模式是JavaScript設計模式中行為型的設計模式;

定義:

定義一系列算法,并將這些算法各自封裝成策略類(方法),然后將不變的部分和變化的部分分離開來,并且這些算法可以相互替換

白話解釋:

實際上所謂的策略模式就是指根據(jù)不同的策略來執(zhí)行不同的方法,是不是很類似與if-else分支判斷;但是策略模式是用來解決多重條件判斷語句的;

代碼實現(xiàn)

需求:

年終將至,某公司決定提前發(fā)年終獎,但是年終獎的計算是有一定的規(guī)則的,年終獎的多少跟績效考核密切相關;所以某公司的年終獎方案是這樣的:

績效考核為S的員工,年終獎是個人月工資的4倍;

績效考核為A的員工,年終獎是個人月工資的3倍;

績效考核為B的員工,年終獎是個人月工資的2倍;

看到這里讓你開始編寫程序,一般大部分的代碼是這樣的:

function calculateBonus(level,salary){
    if(level === 'S'){
        return salary*4;
    }
    
    if(level === 'A'){
        return salary*3
    }

    if(level === 'B'){
        return salary*2
    }
}

console.log(calculateBonus("S",14000));  //56000
console.log(calculateBonus("A",10000)); //30000
console.log(calculateBonus("B",5000));  //10000

上面的代碼用來解決當前需求固然沒有問題,但是在程序設計的角度來說,上面的代碼是還有可以優(yōu)化的點的;因為該方法相對來說比較龐大,有很多的分支判斷,缺乏彈性;如果年終獎方案改了,需要增加一個C方案呢?那是不是又得去方法里面加分支判斷呢?這就違反了開放封閉原則;

優(yōu)化:

var strategies  = {
    "S":function(salary){
        return salary*4
    },
    "A":function(salary){
        return salary*3;
    },
    "B":function(salary){
        return salary*2
    }
}

var calculateBonus =function(level,salary){
    return strategies[level](salary);
} 
console.log(calculateBonus("S",14000));  //56000
console.log(calculateBonus("A",10000));  //30000
console.log(calculateBonus("B",5000));   //10000

通過優(yōu)化上述代碼之后,上面就是用策略模式來進行改造代碼的,我們可以看到我們定義了一個策略對象,然后calculateBonus根據(jù)用戶傳入的等級和工資即可算出年終獎的金額,經(jīng)過改造之后,代碼的結構變得更加簡潔;

在web開發(fā)中,登錄頁的注冊、登錄等功能都是需要進行表單提交的;然而在提交的過程中肯定要進行校驗和篩選,不符合校驗規(guī)則的將不能直接提交;在沒有學習設計模式之前我們的校驗可能也是跟上面一樣都是多重if分支判斷,然后我們現(xiàn)在用策略模式來實現(xiàn)一個表單校驗:

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <meta http-equiv="X-UA-Compatible" content="ie=edge">
    <title>Document</title>
</head>
<body>
        <form action="http:// xxx.com/register" id="registerForm" method="post">
            請輸入用戶名:<input type="text" name="userName"/ >
            請輸入密碼:<input type="text" name="password"/ >
            請輸入手機號碼:<input type="text" name="phoneNumber"/ >
            <button>提交</button>
        </form>
</body>
<script>
        // 定義策略類算法校驗規(guī)則
        var strategies = {
        isNonEmpty: function( value, errorMsg ){
            if ( value === '' ){
                return errorMsg;
            }
        },
        minLength: function( value, length, errorMsg ){
            if ( value.length < length ){
                return errorMsg;
            }
        },
        isMobile: function( value, errorMsg ){
            if ( !/(^1[3|5|8][0-9]{9}$)/.test( value ) ){
                return errorMsg;
            }
        }
    };
    //Validator 類  
    var Validator = function(){
        // 保存校驗規(guī)則
        this.cache = [];
    };
    //添加校驗規(guī)則的方法
    Validator.prototype.add = function( dom, rules ){
        var self = this;
        for ( var i = 0, rule; rule = rules[ i++ ]; ){
            (function( rule ){
                //將校驗規(guī)則對象中的strategy屬性的值進行分割
                var strategyAry = rule.strategy.split( ':' );
                var errorMsg = rule.errorMsg;
                self.cache.push(function(){
                    //將校驗規(guī)則對象中的strategy屬性的第一個值返回回來裝進strategy中
                    var strategy = strategyAry.shift();
                    //組成參數(shù)
                    strategyAry.unshift( dom.value );
                    //組裝參數(shù)
                    strategyAry.push( errorMsg );
                    //找到策略對象執(zhí)行方法裝進cache變量中
                    return strategies[ strategy ].apply( dom, strategyAry );
                });
                console.log(strategyAry);
            })( rule )
        }
    };
    //開始校驗方法
    Validator.prototype.start = function(){
        for ( var i = 0, validatorFunc; validatorFunc = this.cache[ i++ ]; ){
             //循環(huán)cache執(zhí)行方法校驗
            var errorMsg = validatorFunc();
            //如果執(zhí)行策略對象方法中返回了errorMsg,就說明方法已經(jīng)報錯(沒有通過校驗規(guī)則)
            if ( errorMsg ){
                return errorMsg;
            }
        }
    };

    //調用校驗
    var registerForm = document.getElementById( 'registerForm' );
    //定義方法可以自定義添加校驗規(guī)則
    var validataFunc = function(){
        //實例化對象
        var validator = new Validator();
        //自定義添加校驗規(guī)則
        validator.add( registerForm.userName, [{
            strategy: 'isNonEmpty',
            errorMsg: '用戶名不能為空'
        }, {
            strategy: 'minLength:6',
            errorMsg: '用戶名長度不能小于10 位'
        }]);
        validator.add( registerForm.password, [{
            strategy: 'minLength:6',
            errorMsg: '密碼長度不能小于6 位'
        }]);
        //調用方法循環(huán)執(zhí)行校驗
        var errorMsg = validator.start();
        return errorMsg;
    }
    //點擊提交按鈕(提交事件)
    registerForm.onsubmit = function(){
        //執(zhí)行上面自定義的校驗方法
        var errorMsg = validataFunc();
        //如果errorMsg存在,即代表校驗沒有通過
        if ( errorMsg ){
            alert ( errorMsg );
            return false;
        }

    };
</script>
</html>

我們可以通過策略模式來解決表單校驗大規(guī)模重復if-else判斷等問題,上面的代碼注釋我已經(jīng)給的很詳細了,學習設計模式一定要去細品代碼,學習思路;反正策略模式的一個主要思路就是通過定義一系列的算法,然后傳入?yún)?shù),根據(jù)不同的參數(shù)來執(zhí)行不同的算法規(guī)則;

總結

優(yōu)點:

1、利用組合、委托和多態(tài)技術和思想,可以避免多重條件選擇語句;

2、將算法封裝在獨立的策略類里,使得易于切換,易于理解,易于擴展;

3、策略模式可以復用在系統(tǒng)的其他地方,從而避免重復的復制粘貼工作;   

缺點:

1、程序中會增加許多策略類或者策略對象;

2、使用策略類必須要對所有的策略類算法了解清楚,否則不知道怎么選擇。

以上就是JavaScript策略模式的詳細內容,更多關于JavaScript策略模式的資料請關注腳本之家其它相關文章!

相關文章

  • 前端常見的兩種pc適配方案介紹

    前端常見的兩種pc適配方案介紹

    這篇文章主要介紹了前端常見的兩種pc適配方案的相關資料,在PC端自適應設計通過優(yōu)化用戶體驗和提高網(wǎng)站適配性,確保了電商網(wǎng)站、企業(yè)官網(wǎng)和在線教育平臺等多種場景下內容的清晰展示和流暢操作,需要的朋友可以參考下
    2024-10-10
  • js實現(xiàn)延遲加載的幾種方法詳解

    js實現(xiàn)延遲加載的幾種方法詳解

    今天小編就為大家分享一篇關于js實現(xiàn)延遲加載的幾種方法詳解,小編覺得內容挺不錯的,現(xiàn)在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧
    2019-01-01
  • 微信小程序icon組件使用詳解

    微信小程序icon組件使用詳解

    這篇文章主要為大家詳細介紹了微信小程序icon組件的相關資料,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2018-01-01
  • 使用PDFJS遇到的坑及解決辦法記錄

    使用PDFJS遇到的坑及解決辦法記錄

    PDF.js是由Mozilla支持的基于HTML5的PDF查看器,可以在web上解析和呈現(xiàn)PDF文件,它支持內容檢索、頁面跳轉、文件打印等功能,文中通過代碼介紹的非常詳細,需要的朋友可以參考下
    2024-09-09
  • javascript中的異步調用機制

    javascript中的異步調用機制

    這篇文章主要介紹了javascript中的異步調用機制,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2024-02-02
  • SSM+layUI 根據(jù)登錄信息顯示不同的頁面方法

    SSM+layUI 根據(jù)登錄信息顯示不同的頁面方法

    今天小編就為大家分享一篇SSM+layUI 根據(jù)登錄信息顯示不同的頁面方法,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2019-09-09
  • 不使用 JS 匿名函數(shù)理由

    不使用 JS 匿名函數(shù)理由

    本文給大家分析了不使用js匿名函數(shù)的三大理由,匿名函數(shù)的作用是避免全局變量的污染以及函數(shù)名的沖突,關于js匿名函數(shù)的三大理由大家參考下本文
    2017-11-11
  • document.write的幾點使用心得

    document.write的幾點使用心得

    一直用document.write()方法向瀏覽器中顯示數(shù)據(jù)用,把它當做Alert()使用, 看來這樣用有些大材小用了,下面說說它的主要用處。
    2014-05-05
  • JavaScript運算符小結

    JavaScript運算符小結

    本文主要給大家詳細總結了下javascript中的運算符,包括常見的算數(shù)運算符、比較運算符和邏輯運算符。十分的清晰,有需要的小伙伴可以參考下。
    2015-06-06
  • ES6中的迭代器、Generator函數(shù)及Generator函數(shù)的異步操作方法

    ES6中的迭代器、Generator函數(shù)及Generator函數(shù)的異步操作方法

    這篇文章主要介紹了ES6中的迭代器、Generator函數(shù)以及Generator函數(shù)的異步操作方法,非常不錯,具有一定的參考借鑒價值,需要的朋友可以參考下
    2019-05-05

最新評論