詳解Java實(shí)現(xiàn)設(shè)計(jì)模式之責(zé)任鏈模式
假設(shè)我們現(xiàn)在需要在我們的系統(tǒng)中導(dǎo)入一批關(guān)于學(xué)生信息的Excel的數(shù)據(jù),其主要的信息有:學(xué)號(hào)、姓名、年齡、性別等等,在導(dǎo)入系統(tǒng)的時(shí)候,我們肯定不能直接的保存到數(shù)據(jù)庫(kù),我們肯定是先要對(duì)這個(gè)Excel的數(shù)據(jù)進(jìn)行校驗(yàn),看是否符合系統(tǒng)的要求,只有都符合了系統(tǒng)的要求了,我們把這些數(shù)據(jù)保存到數(shù)據(jù)庫(kù)中去。假如我們的學(xué)生對(duì)應(yīng)的實(shí)體類(lèi)如下:
@Datapublic class Student {/** * 學(xué)生編號(hào) */private String stNo;/** * 學(xué)生姓名 */private String stName;/** * 學(xué)生年齡 */private Integer age;/** * 性別 */private String gender;}
那么假設(shè)我們現(xiàn)在的需求是:在我們的StudentServiceImpl業(yè)務(wù)實(shí)現(xiàn)類(lèi)里面已經(jīng)接收到了一個(gè)List studentList集合的數(shù)據(jù),這個(gè)集合的數(shù)據(jù)就是剛剛從Excel里導(dǎo)進(jìn)來(lái)的學(xué)生的數(shù)據(jù)信息,但是集合里面的每個(gè)Student對(duì)象的屬性都沒(méi)有進(jìn)行過(guò)校驗(yàn),現(xiàn)要求你對(duì)這些屬性進(jìn)行校驗(yàn)完全通過(guò)后再把這些學(xué)生的信息保存到數(shù)據(jù)庫(kù)中去。
二、小步小跑的迭代開(kāi)發(fā)好,一開(kāi)始,業(yè)務(wù)那邊的小姑娘小美說(shuō)這些學(xué)生的數(shù)據(jù)沒(méi)有什么重要的,只要校驗(yàn)這個(gè)學(xué)生的姓名不能為空就行且不超過(guò)20個(gè)字就行了,這個(gè)對(duì)于你來(lái)說(shuō)就是個(gè)小意思,于是你可能在業(yè)務(wù)代碼中先對(duì)集合進(jìn)行遍歷然后寫(xiě)下這樣的判斷:
//判斷學(xué)生的姓名是否符合條件if(Objects.nonNull(stu.getStName()) && stu.getStName().length() < 20) {//TODO ...對(duì)符合的數(shù)據(jù)進(jìn)行下一步的處理}
過(guò)了不久,小美害羞的看著你,對(duì)你說(shuō):這個(gè)年齡也要做判斷,必填且不能小于0,不能大于60,改好了請(qǐng)你吃飯。你看著她甜美的笑容,果斷的對(duì)好說(shuō):簡(jiǎn)單。于是你又加上了這樣的代碼:
//判斷學(xué)生的姓名是否符合條件if(Objects.nonNull(stu.getStName()) && stu.getStName().length() < 20) {if(Objects.nonNull(stu.getStAge()) && stu.getStAge() > 0 && stu.getStAge() < 60) {//TODO ...對(duì)符合的數(shù)據(jù)進(jìn)行下一步的處理}}
又過(guò)了幾天,你又看到小美跑過(guò)來(lái),你滿(mǎn)懷期待的覺(jué)得她是通知你今天下班了一起共進(jìn)晚餐。但是她卻支支吾吾的想說(shuō)又說(shuō)不出,這里你心想,完蛋了,會(huì)不會(huì)共進(jìn)晚餐的機(jī)會(huì)泡湯了。這時(shí)她說(shuō):這個(gè)學(xué)生的性別也要做校驗(yàn),且只能是“男”或“女”,加上之前的都要校驗(yàn)通過(guò)了才能在到系統(tǒng)里面。聽(tīng)到這里,你不由得松了一口氣,共進(jìn)晚餐的機(jī)會(huì)還有。于是你說(shuō):沒(méi)問(wèn)題。于是你又在原先的代碼里面進(jìn)行了迭代:
//判斷學(xué)生的姓名是否符合條件if(Objects.nonNull(stu.getStName()) && stu.getStName().length() < 20) {if(Objects.nonNull(stu.getStAge()) && stu.getStAge() > 0 && stu.getStAge() < 60) {if(Object.notNull(stu.getGender()) && ('男'.equest(stu.getGender()) || '女'.equest(stu.getGender()))) {//TODO ...對(duì)符合的數(shù)據(jù)進(jìn)行下一步的處理}}}
你很快的改完了,但是在檢查的時(shí)候,看著這個(gè)代碼,總感覺(jué)有總說(shuō)不出來(lái)的問(wèn)題,但是又好像沒(méi)有什么問(wèn)題。實(shí)然,你想到以后小美肯定還會(huì)經(jīng)常來(lái)找你,如果再繼續(xù)的這樣if的寫(xiě)下去,以后越來(lái)越難維護(hù)了,以后和小美吃飯的時(shí)間都沒(méi)有了。想到這,你不由得直冒冷汗。于是你閉關(guān)半天,終于把這個(gè)潛在的阻止你和小美吃飯的的攔路虎給解決了。
三、系統(tǒng)對(duì)數(shù)據(jù)的校驗(yàn)要求 stName(學(xué)生姓名):不能為空,不能超過(guò)20個(gè)字符。 age(學(xué)生年齡):只能為整數(shù),且小于60。 gender(學(xué)生性別):只能是'男'或'女'。 stNo(學(xué)生編號(hào)):要求唯一,不能為空,不能超過(guò)20個(gè)字符,且在數(shù)據(jù)庫(kù)中不能已經(jīng)存在。四、新建一個(gè)抽象類(lèi)在這個(gè)抽象類(lèi)中,包含有有一個(gè)它自身屬性,和一個(gè)set方法,此外還要有一個(gè)抽象方法,給不同的子類(lèi)來(lái)實(shí)現(xiàn)不同的邏輯,詳細(xì)說(shuō)明如下:
//抽象的父類(lèi)public abstract class AbsCheckStudent {//包含有自身的一個(gè)屬性,其作業(yè)主要是下一個(gè)對(duì)數(shù)據(jù)的處理者protected AbsCheckStudent absCheckStudent;//設(shè)定下一個(gè)處理者public void setAbsCheckStudent(AbsCheckStudent absCheckStudent) {this.absCheckStudent = absCheckStudent;}//此方法是業(yè)務(wù)層調(diào)用的方法,即業(yè)務(wù)調(diào)用此方法,把學(xué)生的集合做參數(shù)傳進(jìn)來(lái)即可。public void handleCheck(List<Student> studentList) {if (Objects.nonNull(studentList) && !studentList.isEmpty()) {List<Student> checkIsOk = checkStudent(studentList);//判斷下一個(gè)處理者是不是null,即還有沒(méi)有下一個(gè)處理者,且判斷數(shù)據(jù)是否為空if (Objects.nonNull(absCheckStudent) && Objects.nonNull(checkIsOk) && !checkIsOk.isEmpty()) {//調(diào)用下一個(gè)處理者的業(yè)務(wù)處理方法absCheckStudent.handleCheck(checkIsOk);}}}//此方法是由不同的子類(lèi)來(lái)進(jìn)行不同的處理實(shí)現(xiàn)public abstract List<Student> checkStudent(List<Student> studentList);}五、子類(lèi)的實(shí)現(xiàn)
首先實(shí)現(xiàn)的是學(xué)生姓名的校驗(yàn)的子類(lèi)
public class StNameCheck extends AbsCheckStudent{@Overridepublic List<Student> checkStudent(List<Student> studentList) {//獲取學(xué)生名稱(chēng)不符合條件的學(xué)生對(duì)象List<Student> stNameIsNotOk = studentList.stream().filter(stu -> {String stName = stu.getStName();return Objects.isNull(stName) || ''.equals(stName);}).collect(Collectors.toList());System.out.println('名字校驗(yàn)不通過(guò)的數(shù)據(jù)有:'+ stNameIsNotOk.toString());//在原有的集合中移除不符合學(xué)生姓名的對(duì)象集合studentList.removeAll(stNameIsNotOk);System.out.println('名字校驗(yàn)通過(guò)的數(shù)據(jù):' + studentList.toString());//返回通過(guò)學(xué)生姓名校驗(yàn)的學(xué)生的集合return studentList; }}
然后再實(shí)現(xiàn)的是學(xué)生年齡的校驗(yàn)子類(lèi)
public class StAgeCheck extends AbsCheckStudent{@Overridepublic List<Student> checkStudent(List<Student> studentList) {//獲取學(xué)生年齡不符合條件的學(xué)生對(duì)象List<Student> stAgeIsNotOk = studentList.stream().filter(stu -> {Integer stAge = stu.getAge();return Objects.isNull(stAge) || stAge <= 0 || stAge >= 60;}).collect(Collectors.toList());System.out.println('年齡校驗(yàn)不通過(guò)的數(shù)據(jù)有:' + stAgeIsNotOk.toString());//在原有的集合中移除不符合學(xué)生年齡的對(duì)象集合studentList.removeAll(stAgeIsNotOk);System.out.println('年齡校驗(yàn)通過(guò)的數(shù)據(jù):' + studentList.toString());//返回通過(guò)學(xué)生姓名校驗(yàn)的學(xué)生的集合return studentList; }}
最后實(shí)現(xiàn)的是學(xué)生性別的校驗(yàn)的子類(lèi)
public class StGenderCheck extends AbsCheckStudent{@Overridepublic List<Student> checkStudent(List<Student> studentList) {//獲取學(xué)生年齡不符合條件的學(xué)生對(duì)象List<Student> stGenderIsNotOk = studentList.stream().filter(stu -> {String gender = stu.getGender();return Objects.isNull(gender) || !('男'.equals(gender) || '女'.equals(gender));}).collect(Collectors.toList());System.out.println('性別校驗(yàn)沒(méi)有通過(guò)的數(shù)據(jù):' + stGenderIsNotOk.toString());//在原有的集合中移除不符合學(xué)生年齡的對(duì)象集合studentList.removeAll(stGenderIsNotOk);System.out.println('性別校驗(yàn)通過(guò)的數(shù)據(jù):' + studentList.toString());//返回通過(guò)學(xué)生姓名校驗(yàn)的學(xué)生的集合return studentList; }}六、構(gòu)建責(zé)任鏈和調(diào)用
好了,現(xiàn)在,校驗(yàn)姓名的子類(lèi)、校驗(yàn)?zāi)挲g的子類(lèi)、校驗(yàn)性別的子類(lèi)都已經(jīng)實(shí)現(xiàn)了。不同職責(zé)的子類(lèi)校驗(yàn)有了,現(xiàn)在我們需要構(gòu)建一條責(zé)任鏈。即,先通過(guò)了姓名校驗(yàn)的數(shù)據(jù)才能進(jìn)行下一步的年齡校驗(yàn),通過(guò)了年齡校驗(yàn)的數(shù)據(jù)才能到性別校驗(yàn),性別校驗(yàn)通過(guò)了,就可以保存數(shù)據(jù)到數(shù)據(jù)庫(kù)了。現(xiàn)在我們構(gòu)建如下的責(zé)任鏈:
public class Chain {public static AbsCheckStudent getStudentCheck() {//校驗(yàn)姓名AbsCheckStudent stNameCheck = new StNameCheck();//校驗(yàn)?zāi)挲gAbsCheckStudent stAgeCheck = new StAgeCheck();//校驗(yàn)性別AbsCheckStudent stGenderCheck = new StGenderCheck();//設(shè)置好責(zé)任鏈的順序,把校驗(yàn)?zāi)挲g的子類(lèi)當(dāng)作StNameCheck中的下一個(gè)處理者stNameCheck.setAbsCheckStudent(stAgeCheck);//把校驗(yàn)性別的子類(lèi)當(dāng)作StAgeCheck中的下一個(gè)處理者stAgeCheck.setAbsCheckStudent(stGenderCheck);}public static void main(String[] args) {AbsCheckStudent studentCheck = getStudentCheck();List<Student> studentList = getStudents();studentCheck.handleCheck(studentList);}public static List<Student> getStudents() {List<Student> result = new ArrayList<>();Student s1 = new Student();s1.setAge(12);s1.setGender('男');s1.setStName('張三');s1.setStNo('');Student s2 = new Student();s2.setAge(12);s2.setGender('男1');s2.setStName('張三');s2.setStNo('123');Student s3 = new Student();s3.setAge(12);s3.setGender('男');s3.setStName('張三');s3.setStNo('123');result.add(s1);result.add(s2);result.add(s3);return result; }}
最后的運(yùn)行結(jié)果如下:
你看,這樣的話(huà),我們只要有有最后校驗(yàn)性別的邏輯里面,對(duì)于通過(guò)性別校驗(yàn)的數(shù)據(jù)保存到數(shù)據(jù)庫(kù)里面就行了。
七、可維護(hù)性當(dāng)你閉關(guān)出來(lái)后,小美又過(guò)來(lái)找你了,說(shuō)學(xué)生編號(hào)要求唯一,不能為空,不能超過(guò)20個(gè)字符,且在數(shù)據(jù)庫(kù)中不能已經(jīng)存在。只有當(dāng)編號(hào)的校驗(yàn)通過(guò)了就可以放心的保存到數(shù)據(jù)庫(kù)了。這時(shí)候,你就可以這樣進(jìn)行擴(kuò)展了,先創(chuàng)建一個(gè)子類(lèi)來(lái)繼承AbsCheckStudent
public class StGenderCheck extends AbsCheckStudent{@Overridepublic List<Student> checkStudent(List<Student> studentList) {//獲取學(xué)生年齡不符合條件的學(xué)生對(duì)象List<Student> stNoIsNotOk = studentList.stream().filter(stu -> {String stNo = stu.getStNo();return Objects.isNull(stNo) || ''.equals(stNo) || stNo.length() > 20;}).collect(Collectors.toList());//TODO 做數(shù)據(jù)庫(kù)中的惟一性的校驗(yàn)等System.out.println('編號(hào)校驗(yàn)不通過(guò)的數(shù)據(jù)有:' + stNoIsNotOk.toString());//在原有的集合中移除不符合學(xué)生編號(hào)的對(duì)象集合studentList.removeAll(stNoIsNotOk);System.out.println('通過(guò)了全部的校驗(yàn)的數(shù)據(jù)有:' + studentList);//TODO 全部通過(guò)校驗(yàn)了,保存數(shù)據(jù)到數(shù)據(jù)庫(kù) save(studentList);return null; }}
然后我們?cè)僭谀莻€(gè)責(zé)任鏈上加上這個(gè)新的處理節(jié)點(diǎn):
public class Chain {public static AbsCheckStudent getStudentCheck() {//校驗(yàn)姓名AbsCheckStudent stNameCheck = new StNameCheck();//校驗(yàn)?zāi)挲gAbsCheckStudent stAgeCheck = new StAgeCheck();//校驗(yàn)性別AbsCheckStudent stGenderCheck = new StGenderCheck();//設(shè)置好責(zé)任鏈的順序,把校驗(yàn)?zāi)挲g的子類(lèi)當(dāng)作StNameCheck中的下一個(gè)處理者stNameCheck.setAbsCheckStudent(stAgeCheck);//把校驗(yàn)性別的子類(lèi)當(dāng)作StAgeCheck中的下一個(gè)處理者stAgeCheck.setAbsCheckStudent(stGenderCheck);AbsCheckStudent stNoCheck = new StNoCheck();//把學(xué)生的編號(hào)校驗(yàn)放到性別校驗(yàn)的后面stGenderCheck.setAbsCheckStudent(stNoCheck);}// ......}
運(yùn)行結(jié)果如下:
以上就是詳解Java實(shí)現(xiàn)設(shè)計(jì)模式之責(zé)任鏈模式的詳細(xì)內(nèi)容,更多關(guān)于Java 設(shè)計(jì)模式 責(zé)任鏈模式的資料請(qǐng)關(guān)注好吧啦網(wǎng)其它相關(guān)文章!
相關(guān)文章:
1. JSP數(shù)據(jù)交互實(shí)現(xiàn)過(guò)程解析2. 解決啟動(dòng)django,瀏覽器顯示“服務(wù)器拒絕訪(fǎng)問(wèn)”的問(wèn)題3. JavaMail 1.4 發(fā)布4. vue使用webSocket更新實(shí)時(shí)天氣的方法5. 淺談python出錯(cuò)時(shí)traceback的解讀6. Python importlib動(dòng)態(tài)導(dǎo)入模塊實(shí)現(xiàn)代碼7. Yii2.0引入CSS,JS文件方法8. Nginx+php配置文件及原理解析9. 關(guān)于HTML5的img標(biāo)簽10. 如何使用CSS3畫(huà)出一個(gè)叮當(dāng)貓
