Java工具类——通过配置XML验证Map
Java工具類——通過配置XML驗證Map
背景
在JavaWeb項目中,接收前端過來的參數(shù)時通常是使用我們的實體類進(jìn)行接收的。但是呢,我們不能去決定已經(jīng)搭建好的框架是怎么樣的,在我接觸的框架中有一種就是通過Map來接收前端過來的所有參數(shù),框架中沒有實體類的說法,從接收參數(shù),驗證參數(shù)到參數(shù)至持久層整個過程都是通過Map來傳遞數(shù)據(jù)。
而在開發(fā)的過程中,減少了實體類的存在,有時是感覺挺方便的,比如:一個系統(tǒng)中有100多個表,這里我們可以減少工作量(雖然對應(yīng)表的實體可以代碼生成),因為我們開發(fā)過程中是需要返回多個表關(guān)聯(lián)后的結(jié)果的,這里可能我們需要創(chuàng)建DTO,這些步驟確實是挺煩人的。但是,前端過來的參數(shù)我們需不需要驗證呢?客戶的輸入不管有意或者是無意,我認(rèn)為都應(yīng)該讓系統(tǒng)的容錯能力更強(qiáng)悍一些。所以,在驗證前端過來的參數(shù)時,使用了Map就著實讓人頭痛。每個需要強(qiáng)制驗證的參數(shù)都需要get,然后判斷類型,強(qiáng)制轉(zhuǎn)型,判斷參數(shù)符不符合期望值邊界等。
所以,我就考慮了,實體類可以通過Spring MVC中Hibernate的Validation使用注解的方式進(jìn)行參數(shù)校驗,那么,少了實體類,我是不是可以通過配置XML的方式來達(dá)到類似有實體類的效果。網(wǎng)上找了類似關(guān)鍵詞的工具類,發(fā)現(xiàn)沒有我所期望的,所以就動手來了一個。
大致的想法
在Web開發(fā)時,有許多if-else語句的出現(xiàn)都是在為了驗證前端參數(shù)合不合法真的是挺無奈的,而且有些代碼雖然長起來類似但是呢要去重構(gòu)成一個公用的方法好像有些困難,時常問自己,要怎么去搞,Java不是JavaScript,語句沒那么靈活。
于是想著通過XML配置試試,大致就是通過配置好的XML代替我們的實體類,并且有個入口將XML中的實體映射,并傳入待驗證的Map,驗證之后傳出一個數(shù)組,如果驗證通過數(shù)組為空,不通過則是我們XML中配置的對應(yīng)錯誤語句。
如何設(shè)計XML格式
動手在這之前,需要想好我們大致的XML結(jié)構(gòu)是怎么樣的。這里,我的想法是,在我們一般遇到的參數(shù)主要就是Integer,String,Double,Date,List了(這里居然沒有考慮Boolean,算了,之后再做補(bǔ)充也行)。所以基于以上,設(shè)計的結(jié)構(gòu)大致如下:
<?xml version="1.0" encoding="UTF-8"?> <map-verifyxmlns="https://www.lger.cn/verify"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="https://www.lger.cn/verify map-verify-util.xsd"><!--在長度的方面分別使用了lt; lte; gt; gte; eq;分別代表<; <=; >; >=; =--><Entity name="User01"><!--castErrMsg為當(dāng)驗證過程中類型解析錯誤時返回提示--><String name="username" castErrMsg="username必須為字符串類型"><length><lt errMsg="當(dāng)前值不能小于2">2</lt></length><notNull errMsg="當(dāng)前值不能為空"/><notBlank errMsg="當(dāng)前值不能為空(去掉首尾空格)"/><pattern><!--如果不匹配則發(fā)出錯誤--><value errMsg="號碼D[0-8]{4}">[0-8]{4}</value><value errMsg="號碼D[0-7]{4}">[0-7]{4}</value></pattern></String><Integer name="age" castErrMsg="必須為整型"><lt errMsg="歲數(shù)必須大于2">2</lt><gt>150</gt><notNull/></Integer><List name="details"><notEmpty/><size><gte>0</gte></size><!-- 遍歷List內(nèi)容,遍歷的Entity映射為name為User02的實體 --><forEach entity="User02" /></List><Date name="birthday"><lt>1900-07-21</lt><notNull/></Date><Double name="money"><lt>1</lt><gt>150</gt><notNull/></Double></Entity><Entity name="User02"><String name="username"><length><lt errMsg="當(dāng)前值不能小于2">2</lt><gt errMsg="當(dāng)前值不能大于10">10</gt></length><notBlank errMsg="當(dāng)前值不能為空(去掉空格)"/></String></Entity> </map-verify>需要寫出一個具有一定格式的XML那么還需要寫出一個約束文件了驗證是否XML格式寫的正確。這里還小學(xué)了一下XSD,這里就不貼出代碼了。
以上的XML大致的說了下怎么使用其替代實體類。寫到了這里其實有時還覺得使用實體類可能更方便,何必使用Map呢?但是,別人框架已經(jīng)寫了,參數(shù)只能接收到Map那我只能屈服了。
類結(jié)構(gòu)設(shè)計
首先,每一個Entity就是一個實體對象,這里我認(rèn)為每個Entity都應(yīng)該包含有一個驗證方法和一個初始化方法,因為在進(jìn)行XML解析時就調(diào)用init,在進(jìn)行Map驗證時就調(diào)用verify方法這樣,那些String節(jié)點也是類似的,解析初始化時就把XML中配置的信息保存起來,等到驗證時就通過之前保存的信息進(jìn)行判斷即可,不必重新解析了。
這里以解析一個IntegerEntity為例,首先是其父類,不論是XML中哪個節(jié)點,都是實現(xiàn)VerifyEntity接口,代碼如下:
public interface VerifyEntity {/*** 一個實體初始化,當(dāng)實體被創(chuàng)建時將由創(chuàng)建方主動調(diào)用* @param currentEle 被創(chuàng)建的實體節(jié)點* @param factory 實體創(chuàng)建工廠,可以通過此工廠創(chuàng)建Entity*/void init(Element currentEle, EntityFactory factory);/*** 驗證當(dāng)前節(jié)點是否匹配XML的配置* @param value 需要驗證的值* @return 異常字符串集*/String[] verify(Object value);/*** 初始化完畢后調(diào)用,傳入包含所有Entity的Map* @param entityMap entityMap*/void finished(Map<String, VerifyEntity> entityMap);}首先解析XML時當(dāng)獲取到Entity節(jié)點下的子節(jié)點將會通過一個工廠類創(chuàng)建子節(jié)點的對應(yīng)實現(xiàn)VerifyEntity,之后調(diào)用init方法對當(dāng)前的子節(jié)點進(jìn)行解析。這里先看下IntegerEntity的源碼:
public class IntegerEntity implements VerifyEntity {private AbstractEquation<Integer> integerEquation;private boolean notNull = false;private String notNullErrMsg;private String castErrMsg;@Overridepublic void init(Element currentEle, EntityFactory factory) {// 開始解析當(dāng)前節(jié)點<Integer/>String name = currentEle.attributeValue("name");// 獲取節(jié)點屬性castErrMsg,看是否存在castErrMsgthis.castErrMsg = currentEle.attributeValue("castErrMsg");if (Util.isEmpty(this.castErrMsg)) {this.castErrMsg = name + ": this is not integer type.";} else {this.castErrMsg = name + ": " + this.castErrMsg;}// 獲取子節(jié)點notNullElement element = currentEle.element("notNull");if (element != null) {this.notNull = true;this.notNullErrMsg = element.attributeValue("errMsg");if (Util.isEmpty(this.notNullErrMsg)) {this.notNullErrMsg = name + ": this is not null";} else {this.notNullErrMsg = name + ": " + this.notNullErrMsg;}}// 這里是新建一個抽象的Equation類,主要是因為lt和lte等等的這些節(jié)點在其他實體中也有,為了代碼復(fù)用所以使用了抽象類來定義//這里實現(xiàn)后與上面解析notNull代碼差異不大integerEquation = new AbstractEquation<Integer>(currentEle, name) {@OverrideInteger valueOf(String value) {try {return Integer.valueOf(value);}catch (NumberFormatException e) {throw new ConvertException(castErrMsg);}}@Overrideboolean lessThan(Integer value, Integer lt) {return value < lt;}@Overrideboolean greaterThan(Integer value, Integer gt) {return value > gt;}@Overrideboolean lessThanOrEquals(Integer value, Integer lte) {return value <= lte;}@Overrideboolean greaterThanOrEquals(Integer value, Integer gte) {return value <= gte;}@Overrideboolean equals(Integer value, Integer eq) {return value.equals(eq);}};}@Overridepublic String[] verify(Object value) {//正式驗證參數(shù)是否合法if (value == null) {//如果之前解析包含notNull,則這里為true,那么將返回解析的notNullErrMsgif (this.notNull) {return new String[]{this.notNullErrMsg};}return null;}try {//使用上面實現(xiàn)的抽象類進(jìn)行驗證return integerEquation.verify(Integer.valueOf(value.toString()));}catch (NumberFormatException e) {return new String[]{this.castErrMsg};}}@Overridepublic void finished(Map<String, VerifyEntity> entityMap) {//這里是在Entity解析完畢后調(diào)用,并將保存Entity的Map傳入}}其實根據(jù)以上的代碼可以看出,就是XML的節(jié)點對應(yīng)著一個VerifyEntity實現(xiàn),每一個實現(xiàn)都保存著其中定義的<notNull/>等信息。
實現(xiàn)后運行效果
這里我們的測試代碼如下,其中demo.xml為設(shè)計XML所示的代碼:
//初始化MapVerify,將定義好xml以數(shù)組方式傳入final String[] xmls = {"demo.xml"};MapVerify.init(xmls);final Map<String, Object> map = new HashMap<>(5);map.put("username", "12312371237123778");map.put("age", 1);map.put("birthday", "2000-07-21");map.put("money", 149);List<Map<String, Object>> list = new ArrayList<>(2);Map<String, Object> map1 = new HashMap<>(1);map1.put("username", "abcab");Map<String, Object> map2 = new HashMap<>(1);map2.put("username", "cc");list.add(map1);list.add(map2);map.put("details", list);System.out.println(Arrays.toString(MapVerify.verifyByReturnArr("User01", map)));運行的結(jié)果如下:
[age: 歲數(shù)必須大于2]總結(jié)
此想法是半年前的了,現(xiàn)在利用了空余的時間實現(xiàn)了自己的想法,趁熱打鐵寫下博客,希望有這個需求的小伙伴能節(jié)省自己的時間(注:這里Boolean的判斷沒有實現(xiàn),如果有需要就需要自己動手了:-))。以后也希望自己如果有什么小想法盡量的抽時間去做,不論做的好不好。
源碼已經(jīng)上傳至 GitHub,包含demo。
jar包請點擊 此鏈接,使用jar包時需要多引入dom4j依賴包
轉(zhuǎn)載于:https://www.cnblogs.com/lger/p/10645453.html
總結(jié)
以上是生活随笔為你收集整理的Java工具类——通过配置XML验证Map的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: centos7 kafka2.3.1单机
- 下一篇: smartadmin官网_smartad