從表單校驗(yàn)看JavaScript策略模式的使用詳解
眾所周知的是,表單確實(shí)在前端,唔,或者說(shuō)在網(wǎng)頁(yè)中占有不小的比重。事實(shí)上,幾乎每一個(gè)中大型網(wǎng)站都會(huì)有“登錄注冊(cè)”以驗(yàn)證用戶信息、防止一些不可名狀的隱患。。。
那么表單的優(yōu)劣就成了前端開(kāi)發(fā)者急需解決的問(wèn)題。其實(shí)我更愿意稱為“代碼的可讀性”或“可復(fù)用性”以及“是否冗雜”。
表單也有“優(yōu)劣”?你在開(kāi)玩笑嘛?
我想你可以認(rèn)真看下下面的代碼,它用到了一些“新知識(shí)”:
<form action="xxx" id="registerForm"> 請(qǐng)輸入用戶名:<input type="text" name="userName" id="name" /> 請(qǐng)輸入密碼:<input type="text" name="password" id="pass" /> 請(qǐng)輸入手機(jī)號(hào):<input type="text" name="phoneNumber" id="phone" /> <button>提交</button> </form>
用戶名、密碼、手機(jī)號(hào)這應(yīng)該是表單中最常見(jiàn)的了,好,我們就以此分析!
上面這些只是簡(jiǎn)單的演示效果,你完全可以用css的valid/invalid、HTML5的required/pattern來(lái)配合完成。
<script>
var registerForm=document.getElementById('registerForm')
registerForm.onsubmit=function(){
if(registerForm.userName.value==''){
document.getElementById("name").setCustomValidity('用戶名不能為空');
return false;
}
if(registerForm.password.value.length<6){
document.getElementById("pass").setCustomValidity('密碼長(zhǎng)度不能少于6位');
return false;
}
if(!/(^1[3|5|8][0-9]{9}$)/.test(registerForm.phoneNumber.value)){
document.getElementById("phone").setCustomValidity('手機(jī)號(hào)碼格式不正確');
return false;
}
}
</script>
但即使這樣,你也不會(huì)覺(jué)得它很完美 —— 現(xiàn)在表單只有三條,如果某一天它增加到了N條,即使是「復(fù)制粘貼」也拯救不了你!
就在這時(shí),你想到了 策略模式 (看,JS總是會(huì)讓你“靈光一現(xiàn)”)
說(shuō)起策略模式,很自然地,要遵循 暴露接口和實(shí)現(xiàn)邏輯分離 的原則。
策略模式指的是定義一系列的算法,把它們一個(gè)個(gè)封裝起來(lái)。將不變的部分和變化的部分隔開(kāi)是每個(gè)設(shè)計(jì)模式的主題,策略模式也不例外,
策略模式的目的就是將算法的使用與算法的實(shí)現(xiàn)分離開(kāi)來(lái)。
一個(gè)基于策略模式的程序至少由兩部分組成。第一個(gè)部分是一組策略類,策略類封裝了具體的算法,并負(fù)責(zé)具體的計(jì)算過(guò)程。第二個(gè)部分是環(huán)境類 Context,Context 接受客戶的請(qǐng)求,隨后把請(qǐng)求委托給某一個(gè)策略類。要做到這點(diǎn),說(shuō)明 Context 中要維持對(duì)某個(gè)策略對(duì)象的引用
——《JavaScript設(shè)計(jì)模式與開(kāi)發(fā)實(shí)踐》
那么,第一步我們很顯然要把這些校驗(yàn)邏輯都封裝成【策略對(duì)象】:
var strategies={
isNoneEmpty: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;
}
}
};
接下來(lái)我們要實(shí)現(xiàn)一個(gè)“暴露出去的”、“作為調(diào)用的”方法類 —— 它將作為context(上下文),負(fù)責(zé)接收用戶的請(qǐng)求并委托給驗(yàn)證對(duì)象stratrgies:
var Validator=function(){
this.cache=[]; //用于保存接收到的校驗(yàn)規(guī)則
};
Validator.prototype.add=function(dom,rule,errorMsg){
var ary=rule.split(':');
this.cache.push(function(){
var strategy=ary.shift();
ary.unshift(dom.value);
ary.push(errorMsg);
return strategies[strategy].apply(dom,ary); //調(diào)用strategies對(duì)象的指定方法對(duì)象,并規(guī)定在函數(shù)對(duì)象內(nèi)部this指向dom元素,ary作為參數(shù)傳入
});
};
Validator.prototype.start=function(){
for(var i=0,validatorFunc;validatorFunc=this.cache[i++];){
var msg=validatorFunc();
if(msg){
return msg;
}
}
}
使用:
var validataFunc=function(){
var validator=new Validator();
//添加校驗(yàn)規(guī)則
validator.add(registerForm.userName,'isNoneEmpty','用戶名不能為空');
validator.add(registerForm.password,'minLength:6','密碼長(zhǎng)度不能少于6位');
validator.add(registerForm.phoneNumber,'isMobile','手機(jī)號(hào)碼格式不正確');
var errMsg=validator.start(); //獲得校驗(yàn)結(jié)果
return errorMsg; //返回
}
var registerForm=document.getElementById('registerForm')
registerForm.onsubmit=function(){
var errorMsg=validataFunc();
if(errorMsg){
//觸發(fā)錯(cuò)誤提示
return false; //并阻止表單提交
}
}
我們可以看到的是:當(dāng)我們往 validator 對(duì)象里添加完一系列的校驗(yàn)規(guī)則之后,會(huì)調(diào)用 validator.start() 方法來(lái)啟動(dòng)校驗(yàn)。如果 validator.start() 返回了一個(gè)確切的 errorMsg 字符串當(dāng)作返回值,說(shuō)明該次校驗(yàn)沒(méi)有通過(guò),此時(shí)需讓 registerForm.onsubmit 方法返回 false 來(lái)阻止表單的提交。
這樣確實(shí)比之前好很多:至少在我們修改驗(yàn)證規(guī)則時(shí)顯得毫不費(fèi)力:
validator.add(registerForm.userName,'minLength:2','用戶名不能少于2位')
但是問(wèn)題也就隨之而來(lái)了:我們把對(duì)于用戶名的驗(yàn)證規(guī)則“不能為空”改為了“不能少于兩位”,那么就不能驗(yàn)證“是否為空”了。
能不能像element-ui一樣可以自定義多種驗(yàn)證規(guī)則 呢?就像這樣:
validator.add(registerForm.userName,[{
strategy:'isNoneEmpty',
errorMsg:'用戶名不能為空'
},{
strategy:'minLength:2',
errorMsg:'用戶名長(zhǎng)度不能少于2位'
}])
現(xiàn)在“rule”是數(shù)組-對(duì)象的形式了,我們需要把add函數(shù)改一下:
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);
return strategies[strategy].apply(dom,strategyAry);
});
})(rule)
}
}
策略模式的優(yōu)點(diǎn):
利用組合、委托、多態(tài)等技術(shù)和思想,可以有效地避免多重條件選擇語(yǔ)句(關(guān)于這一點(diǎn),筆者在 這篇文章 中做了詳細(xì)說(shuō)明);完美實(shí)現(xiàn)了設(shè)計(jì)模式都應(yīng)該具有的“對(duì)外開(kāi)放-封閉”原則,基于策略模式實(shí)現(xiàn)的規(guī)則大多易于擴(kuò)展、易于使用避免了大量CV的工作
到此這篇關(guān)于從表單校驗(yàn)看JavaScript策略模式的使用詳解的文章就介紹到這了,更多相關(guān)JavaScript策略模式使用內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
完美實(shí)現(xiàn)js焦點(diǎn)輪播效果(一)
這篇文章主要為大家詳細(xì)介紹了完美實(shí)現(xiàn)js焦點(diǎn)輪播效果的相關(guān)代碼,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-03-03
js實(shí)現(xiàn)幻燈片效果(基于jquery插件)
說(shuō)起幻燈片效果,想必大家都不陌生吧,接下來(lái)為大家介紹下使用jquery插件jquery.smallslider.js實(shí)現(xiàn)幻燈片效果示例代碼,喜歡的朋友可以學(xué)習(xí)下2013-11-11
js實(shí)現(xiàn)獲取兩個(gè)日期之間所有日期的方法
這篇文章主要介紹了js實(shí)現(xiàn)獲取兩個(gè)日期之間所有日期的方法,涉及javascript針對(duì)日期與時(shí)間的相關(guān)操作技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2016-06-06
解決layer.open后laydate失效的問(wèn)題
今天小編就為大家分享一篇解決layer.open后laydate失效的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2019-09-09
利用進(jìn)制轉(zhuǎn)換壓縮數(shù)字函數(shù)分享
本文主要介紹了進(jìn)制轉(zhuǎn)換函數(shù),用于壓縮數(shù)字,比如Date.now()這樣的長(zhǎng)數(shù)字,用62進(jìn)制表示,就更短,大家參考使用吧2014-01-01
深入了解JavaScript中l(wèi)et/var/function的變量提升
這篇文章主要介紹了深入了解JavaScript中l(wèi)et/var/function的變量提升,文章圍繞主題展開(kāi)詳細(xì)的內(nèi)容介紹,具有一定的參考價(jià)值,需要的朋友可以參考一下2022-07-07

