詳解Java實現設計模式之責任鏈模式
假設我們現在需要在我們的系統中導入一批關于學生信息的Excel的數據,其主要的信息有:學號、姓名、年齡、性別等等,在導入系統的時候,我們肯定不能直接的保存到數據庫,我們肯定是先要對這個Excel的數據進行校驗,看是否符合系統的要求,只有都符合了系統的要求了,我們把這些數據保存到數據庫中去。假如我們的學生對應的實體類如下:
@Datapublic class Student {/** * 學生編號 */private String stNo;/** * 學生姓名 */private String stName;/** * 學生年齡 */private Integer age;/** * 性別 */private String gender;}
那么假設我們現在的需求是:在我們的StudentServiceImpl業務實現類里面已經接收到了一個List studentList集合的數據,這個集合的數據就是剛剛從Excel里導進來的學生的數據信息,但是集合里面的每個Student對象的屬性都沒有進行過校驗,現要求你對這些屬性進行校驗完全通過后再把這些學生的信息保存到數據庫中去。
二、小步小跑的迭代開發好,一開始,業務那邊的小姑娘小美說這些學生的數據沒有什么重要的,只要校驗這個學生的姓名不能為空就行且不超過20個字就行了,這個對于你來說就是個小意思,于是你可能在業務代碼中先對集合進行遍歷然后寫下這樣的判斷:
//判斷學生的姓名是否符合條件if(Objects.nonNull(stu.getStName()) && stu.getStName().length() < 20) {//TODO ...對符合的數據進行下一步的處理}
過了不久,小美害羞的看著你,對你說:這個年齡也要做判斷,必填且不能小于0,不能大于60,改好了請你吃飯。你看著她甜美的笑容,果斷的對好說:簡單。于是你又加上了這樣的代碼:
//判斷學生的姓名是否符合條件if(Objects.nonNull(stu.getStName()) && stu.getStName().length() < 20) {if(Objects.nonNull(stu.getStAge()) && stu.getStAge() > 0 && stu.getStAge() < 60) {//TODO ...對符合的數據進行下一步的處理}}
又過了幾天,你又看到小美跑過來,你滿懷期待的覺得她是通知你今天下班了一起共進晚餐。但是她卻支支吾吾的想說又說不出,這里你心想,完蛋了,會不會共進晚餐的機會泡湯了。這時她說:這個學生的性別也要做校驗,且只能是“男”或“女”,加上之前的都要校驗通過了才能在到系統里面。聽到這里,你不由得松了一口氣,共進晚餐的機會還有。于是你說:沒問題。于是你又在原先的代碼里面進行了迭代:
//判斷學生的姓名是否符合條件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 ...對符合的數據進行下一步的處理}}}
你很快的改完了,但是在檢查的時候,看著這個代碼,總感覺有總說不出來的問題,但是又好像沒有什么問題。實然,你想到以后小美肯定還會經常來找你,如果再繼續的這樣if的寫下去,以后越來越難維護了,以后和小美吃飯的時間都沒有了。想到這,你不由得直冒冷汗。于是你閉關半天,終于把這個潛在的阻止你和小美吃飯的的攔路虎給解決了。
三、系統對數據的校驗要求 stName(學生姓名):不能為空,不能超過20個字符。 age(學生年齡):只能為整數,且小于60。 gender(學生性別):只能是'男'或'女'。 stNo(學生編號):要求唯一,不能為空,不能超過20個字符,且在數據庫中不能已經存在。四、新建一個抽象類在這個抽象類中,包含有有一個它自身屬性,和一個set方法,此外還要有一個抽象方法,給不同的子類來實現不同的邏輯,詳細說明如下:
//抽象的父類public abstract class AbsCheckStudent {//包含有自身的一個屬性,其作業主要是下一個對數據的處理者protected AbsCheckStudent absCheckStudent;//設定下一個處理者public void setAbsCheckStudent(AbsCheckStudent absCheckStudent) {this.absCheckStudent = absCheckStudent;}//此方法是業務層調用的方法,即業務調用此方法,把學生的集合做參數傳進來即可。public void handleCheck(List<Student> studentList) {if (Objects.nonNull(studentList) && !studentList.isEmpty()) {List<Student> checkIsOk = checkStudent(studentList);//判斷下一個處理者是不是null,即還有沒有下一個處理者,且判斷數據是否為空if (Objects.nonNull(absCheckStudent) && Objects.nonNull(checkIsOk) && !checkIsOk.isEmpty()) {//調用下一個處理者的業務處理方法absCheckStudent.handleCheck(checkIsOk);}}}//此方法是由不同的子類來進行不同的處理實現public abstract List<Student> checkStudent(List<Student> studentList);}五、子類的實現
首先實現的是學生姓名的校驗的子類
public class StNameCheck extends AbsCheckStudent{@Overridepublic List<Student> checkStudent(List<Student> studentList) {//獲取學生名稱不符合條件的學生對象List<Student> stNameIsNotOk = studentList.stream().filter(stu -> {String stName = stu.getStName();return Objects.isNull(stName) || ''.equals(stName);}).collect(Collectors.toList());System.out.println('名字校驗不通過的數據有:'+ stNameIsNotOk.toString());//在原有的集合中移除不符合學生姓名的對象集合studentList.removeAll(stNameIsNotOk);System.out.println('名字校驗通過的數據:' + studentList.toString());//返回通過學生姓名校驗的學生的集合return studentList; }}
然后再實現的是學生年齡的校驗子類
public class StAgeCheck extends AbsCheckStudent{@Overridepublic List<Student> checkStudent(List<Student> studentList) {//獲取學生年齡不符合條件的學生對象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('年齡校驗不通過的數據有:' + stAgeIsNotOk.toString());//在原有的集合中移除不符合學生年齡的對象集合studentList.removeAll(stAgeIsNotOk);System.out.println('年齡校驗通過的數據:' + studentList.toString());//返回通過學生姓名校驗的學生的集合return studentList; }}
最后實現的是學生性別的校驗的子類
public class StGenderCheck extends AbsCheckStudent{@Overridepublic List<Student> checkStudent(List<Student> studentList) {//獲取學生年齡不符合條件的學生對象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('性別校驗沒有通過的數據:' + stGenderIsNotOk.toString());//在原有的集合中移除不符合學生年齡的對象集合studentList.removeAll(stGenderIsNotOk);System.out.println('性別校驗通過的數據:' + studentList.toString());//返回通過學生姓名校驗的學生的集合return studentList; }}六、構建責任鏈和調用
好了,現在,校驗姓名的子類、校驗年齡的子類、校驗性別的子類都已經實現了。不同職責的子類校驗有了,現在我們需要構建一條責任鏈。即,先通過了姓名校驗的數據才能進行下一步的年齡校驗,通過了年齡校驗的數據才能到性別校驗,性別校驗通過了,就可以保存數據到數據庫了。現在我們構建如下的責任鏈:
public class Chain {public static AbsCheckStudent getStudentCheck() {//校驗姓名AbsCheckStudent stNameCheck = new StNameCheck();//校驗年齡AbsCheckStudent stAgeCheck = new StAgeCheck();//校驗性別AbsCheckStudent stGenderCheck = new StGenderCheck();//設置好責任鏈的順序,把校驗年齡的子類當作StNameCheck中的下一個處理者stNameCheck.setAbsCheckStudent(stAgeCheck);//把校驗性別的子類當作StAgeCheck中的下一個處理者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; }}
最后的運行結果如下:
你看,這樣的話,我們只要有有最后校驗性別的邏輯里面,對于通過性別校驗的數據保存到數據庫里面就行了。
七、可維護性當你閉關出來后,小美又過來找你了,說學生編號要求唯一,不能為空,不能超過20個字符,且在數據庫中不能已經存在。只有當編號的校驗通過了就可以放心的保存到數據庫了。這時候,你就可以這樣進行擴展了,先創建一個子類來繼承AbsCheckStudent
public class StGenderCheck extends AbsCheckStudent{@Overridepublic List<Student> checkStudent(List<Student> studentList) {//獲取學生年齡不符合條件的學生對象List<Student> stNoIsNotOk = studentList.stream().filter(stu -> {String stNo = stu.getStNo();return Objects.isNull(stNo) || ''.equals(stNo) || stNo.length() > 20;}).collect(Collectors.toList());//TODO 做數據庫中的惟一性的校驗等System.out.println('編號校驗不通過的數據有:' + stNoIsNotOk.toString());//在原有的集合中移除不符合學生編號的對象集合studentList.removeAll(stNoIsNotOk);System.out.println('通過了全部的校驗的數據有:' + studentList);//TODO 全部通過校驗了,保存數據到數據庫 save(studentList);return null; }}
然后我們再在那個責任鏈上加上這個新的處理節點:
public class Chain {public static AbsCheckStudent getStudentCheck() {//校驗姓名AbsCheckStudent stNameCheck = new StNameCheck();//校驗年齡AbsCheckStudent stAgeCheck = new StAgeCheck();//校驗性別AbsCheckStudent stGenderCheck = new StGenderCheck();//設置好責任鏈的順序,把校驗年齡的子類當作StNameCheck中的下一個處理者stNameCheck.setAbsCheckStudent(stAgeCheck);//把校驗性別的子類當作StAgeCheck中的下一個處理者stAgeCheck.setAbsCheckStudent(stGenderCheck);AbsCheckStudent stNoCheck = new StNoCheck();//把學生的編號校驗放到性別校驗的后面stGenderCheck.setAbsCheckStudent(stNoCheck);}// ......}
運行結果如下:
以上就是詳解Java實現設計模式之責任鏈模式的詳細內容,更多關于Java 設計模式 責任鏈模式的資料請關注好吧啦網其它相關文章!
相關文章:
