日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

Guns二次开发(四):重构字典管理模块

發(fā)布時間:2024/3/24 编程问答 57 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Guns二次开发(四):重构字典管理模块 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

?

關于博客中使用的Guns版本問題請先閱讀? ?Guns二次開發(fā)目錄? ? ? ?

? ? ? ?Guns中的字段管理是一個很不錯的功能,我其實也是第一次發(fā)現還能這么玩,因為以前一直使用的方法是:后端直接將數據庫保存的狀態(tài)值返回給前端,然后在文檔中加上字段不同值的解釋,前端針對這個字段做不同的 if else 判斷來解釋其含義,并回顯到頁面中。某種意義上來講,這其實是后端開發(fā)人員將自己的工作量拋給前端開發(fā)人員,以此減輕自己的工作量。但是對于mvc項目下的后臺管理系統(tǒng),此時前端代碼和后端代碼都是由Java開發(fā)包圓了,于是后端拋出去的球又回到了自己手中,在這種情況下,字典管理便應運而生了。當然了,以上都是我個人的想法,大家當個段子看便好。

? ? ? ?現在切入正題。上一篇博客中,我們一起修改了字典管理的性別字段顯示錯誤的問題,相信大家對字典管理的都有了一定的認識。今天我決定再加大挑戰(zhàn)的難度,我準備重構這個字典功能。之所以重構,并非本人閑的蛋疼,而是字典管理這個模塊的確存在一定的設計缺陷。

?

1、保證字典名稱的唯一性

? ? ? 需求: 準確點描述就是,在sys_dict 表中,對于pid為0的記錄,要保證name字段的唯一性(也就是保證字典名稱的唯一性)。對于pid不為0的同級別記錄(即pid相等但pid不為0的記錄),要保證name和code字段的唯一性。

?

更為形象的表述:

以上圖中的A、B兩組數據為例來說明,A、B兩組數據代表兩個字典,

(1)pid為0的記錄的name字段代表的是字典名稱,code字段表示表中的字段名;

(2)pid不為0的記錄的name字段代表的就是字段值的釋義,code字段代表的是字段值。

?

要求:

(1)當pid都為0時,此時必須保證name字段的唯一性,而code字段反而可以重復;

(2)當pid不為0,但相等時,比如pid是53,此時A組數據的id為54和55的記錄就是同級記錄,此時既要保證name字段的唯一性,也要保證code字段的唯一性。

(3)當pid不同時,比如id為54和129的兩條記錄,因為不屬于同一組,所以不管是name重復還是code重復都是允許的。

?

?

而現在的情況是:

?

?

?

?

?

所以,需要修改添加字典和修改字典接口的邏輯,保證字典名稱(即pid=0的記錄的name字段)的唯一性,之所以要保證字典名稱的唯一性,而不是code字段的唯一性,是因為實際開發(fā)中,不同表中出現相同字段的情況是時常發(fā)生的,有可能我商品分類表中有一個字段status,然后我商品表中也有一個status字段,如果此時這兩個字段都要做字典,此時如何保證唯一性?一開始我想到要不要在字典表中新增一個代表表名的字段來用于保證記錄唯一性,但后來一想,這反而增加修改的成本,我們直接在給name字段命名的時候規(guī)范一下,其實也正好可以達到這個效果。比如商品表的status字段的字典命名為【商品狀態(tài)】,商品分類表的status字段的字典命名為【商品分類狀態(tài)】,這樣既保證了name的唯一性,又不用修改表結構,而且反而還更具可讀性。

?

下面把修改后的代碼貼出來,要修改的接口有兩個,一個是添加接口,另一個是修改接口,另外我將公共的添加方法抽取出來了。下面只粘貼需要修改的部分代碼,方便閱讀,文章最后會有完整的代碼:


?

DictController.java

/*** 新增字典** @param dictValues 格式例如 "1:啟用;2:禁用;3:凍結"*/@BussinessLog(value = "添加字典記錄", key = "dictName,dictValues", dict = DictMap.class)@RequestMapping(value = "/add")@Permission(Const.ADMIN_NAME)@ResponseBodypublic Object add(@RequestParam(required = false,defaultValue = "0",value = "dictNum")Integer dictNum,String dictCode, String dictTips, String dictName, String dictValues) {if (ToolUtil.isOneEmpty(dictCode, dictValues,dictName)) {throw new ServiceException(BizExceptionEnum.REQUEST_NULL);}this.dictService.addDict(dictCode, dictName, dictTips, dictValues,dictNum);return SUCCESS_TIP;}/*** 修改字典*/@BussinessLog(value = "修改字典", key = "dictName,dictValues", dict = DictMap.class)@RequestMapping(value = "/update")@Permission(Const.ADMIN_NAME)@ResponseBodypublic Object update(@RequestParam(required = false,defaultValue = "0",value = "dictNum")Integer dictNum,Integer dictId, String dictCode, String dictName, String dictTips, String dictValues) {if (ToolUtil.isOneEmpty(dictId, dictCode, dictName, dictValues)) {throw new ServiceException(BizExceptionEnum.REQUEST_NULL);}dictService.editDict(dictId, dictCode, dictName, dictTips, dictValues,dictNum);return SUCCESS_TIP;}

?

DictServiceImpl.java

@Override@Transactional//為了保證事務一致性,必須加上這個注解public void addDict(String dictCode, String dictName, String dictTips, String dictValues, Integer dictNum) {//添加字典insertDict(dictCode, dictName, dictTips, dictValues, dictNum);}/*** 公共的添加字典業(yè)務*/private void insertDict(String dictCode, String dictName, String dictTips, String dictValues, Integer dictNum){//解析dictValuesList<Dict> list = getDictList(dictValues);//添加字典Dict dict = new Dict();dict.setName(dictName);dict.setCode(dictCode);dict.setTips(dictTips);dict.setNum(dictNum);//此處傳入設置的numdict.setPid(0);//執(zhí)行字典名的添加操作int count = dictMapper.insert(dict);if(count==0) {//FAIL_ADD_RECORD(500,"數據庫中新增數據失敗"),throw new ServiceException(BizExceptionEnum.FAIL_ADD_RECORD);}//因為要保證字典名的唯一性,所以需要再查詢一遍,判斷是否沖突Wrapper<Dict> wrapper = new EntityWrapper<>();wrapper.eq("pid",0).eq("name",dictName);count = dictMapper.selectCount(wrapper);if(count>1){//DICT_NAME_EXISTED(400,"字典名稱已經存在"),throw new ServiceException(BizExceptionEnum.DICT_NAME_EXISTED);}//添加字典條目Dict item = null;Integer pid = dict.getId();//獲取父類idfor(int i=0; i<list.size(); i++ ){item = list.get(i);item.setPid(pid);}//直接調用批量插入語句insertBatch(list);}@Override@Transactionalpublic void editDict(Integer dictId, String dictCode, String dictName, String dictTips, String dicts,Integer dictNum) {//刪除之前的字典this.delteDict(dictId);//添加字典insertDict(dictCode, dictName, dictTips, dicts, dictNum);}@Override@Transactionalpublic void delteDict(Integer dictId) {//刪除這個字典的子詞典Wrapper<Dict> dictEntityWrapper = new EntityWrapper<>();dictEntityWrapper = dictEntityWrapper.eq("pid", dictId);dictMapper.delete(dictEntityWrapper);//刪除這個詞典dictMapper.deleteById(dictId);}/*** 解析一個組合字符串(例如: "1:啟用;2:禁用;3:凍結" 這樣的字符串)* 本方法是將字符串直接轉換成對應的Dict實體類集合,* 如果前端傳入參數的方式錯誤* 或者用戶輸入的參數不合法,此處都會拋出異常,終止程序*/private List<Dict> getDictList(String mutiString) {List<Dict> list = new ArrayList<>();//判斷分隔后是否為空數組List<String> split = StringUtil.split(mutiString,";");if(split.isEmpty()){ // ERROR_ITEM_EMPTY(500, "字典詳情不能為空"),throw new ServiceException(BizExceptionEnum.ERROR_ITEM_EMPTY);}//用來封裝已經存在的code參數,如果有重復的值,說明前端傳入的值錯誤了,//此時也應該拋出異常Set<String> codeSet = new HashSet<>();Set<String> nameSet = new HashSet<>();Dict dict = null;List<String> attrs = null;String attr = null;for (String item : split) {dict = new Dict();//如果數組長度不為3,說明前端的傳入數據錯誤,此時請求應終止attrs = StringUtil.split(item,":");if(attrs.size() !=3){//DICT_PARAMS_ILLEGAL(400,"請?zhí)顚懲暾淖值湓斍樾畔?#34;),throw new ServiceException(BizExceptionEnum.DICT_PARAMS_ILLEGAL);}//保證code值的唯一attr = attrs.get(0);if(codeSet.contains(attr)){throw new ServiceException(400,"值【"+attr+"】重復");}codeSet.add(attr);dict.setCode(attr);//為了字典的可讀性,我這邊也要保證name字段的一致attr = attrs.get(1);if(nameSet.contains(attr)){throw new ServiceException(400,"名稱【"+attr+"】重復");}nameSet.add(attr);dict.setName(attr);//保存排序編號if(!NumberUtils.isDigits(attrs.get(2))){//ERROR_NUM_TYPE(500, "字典詳情序號只能是數字"),throw new ServiceException(BizExceptionEnum.DICT_MUST_BE_NUMBER);}dict.setNum(new Integer(attrs.get(2)));list.add(dict);}return list;}

?

?


?

?

修改好代碼后,重新運行項目,測試效果:

測試添加重復的字典名稱:

?

測試添加相同釋義的記錄:

?

添加列表顯示的字段,提高可讀性:

?

?

?

?

?

2、修改編輯字典的內部邏輯

? ? ? ?不知道大家還記不記得,上一篇博客?Guns二次開發(fā)(三):解決用戶性別字典顯示錯誤?中,我們在解決用戶性別字典顯示錯誤的時候,有讀到這么一段源碼:

?

性別字典的名稱在代碼中被寫死了,也就是我要想正常使用【性別】這個字典的值,就必須保證性別字典的名稱是字符串“性別”二字:

?

只有這樣,性別字典才能正常使用。

?

一旦我們修改了【性別】字典的名稱,如下圖:

?

那么,性別字典就會失效,如下圖:

?

?

?

要想讓性別字典生效,只能去改動代碼中的字典名稱,但這樣一來,字典功能就顯得沒有價值了。還有一種方法就是在修改字典的時候,前端禁用字典名稱字段所在的文本框:

?

? ? ? ?但是這樣一來,就降低了用戶體驗,因為有時候,用戶修改字典名稱的需求還是很大的,不能因為設計者的無能而降低了用戶的體驗。所以,給字典定義一個不變且唯一的字段也就變得很有必要,一說到唯一且不變,我們應該很下意識就能想到數據表的主鍵id,主鍵id因為是自增的,所以能保證唯一,又因為是主鍵,所以更新操作通常都不會改變它。不過呢,只是正常情況下如此,我們只要去查看字典修改接口 /dict/update 的內部實現邏輯,就會發(fā)現它的邏輯是:先刪除掉原來的字典,然后再重新添加一遍。如此一來,修改之后,主鍵id也會跟著改變,如下圖源碼:

?

? ? ? ?看到這里,內心是不是有一種無法言明的難受感,事實上,這樣寫代碼是不行的。因為用戶(不管是后臺客服還是前臺用戶)的刪除操作,都不能使用邏輯刪除,除了DBA(數據庫管理員)外,任何人都不應該有刪除數據庫數據的權限,因為大數據時代來了,大數據時代,沒有無效的數據,只有無用的大數據開發(fā)者。當然這些都只是我個人的理解,言歸正傳,鑒于前面的情況,我決定還是修改后端的修改字典接口來達到我們的目的。

? ? ? ?我原本是想將物理刪除修改為邏輯刪除(sys_dict表中加個狀態(tài)字段),但是考慮到這樣修改的成本太大,而且我的時間也有限,自己項目的進度也很緊,所以最終還是以 “盡量減少改動原代碼”的原則,做出如下的修改思路:

? ? ? ?我在編輯的時候,對于字典名稱那條記錄,不做刪除操作,而做修改操作,這樣一來,字典名稱所在記錄的id就是唯一且一直有效的,然后字典條目(字典釋義)的記錄(即pid不為0的記錄)則依舊是物理刪除后再重新添加。接著是字典包裝那一段,我不再使用字典名稱作為定位的標志,而是改為使用字典名稱記錄的id字段,如此一來,即便我將定位標志寫死了,后面無論我如何修改字典名稱,我的字典都能正常使用。

前面的話都太抽象,讓我換個形象點的說法:?

?

?

?

?

?

?

?

?

3、源碼

?

?

?

?

?

?

?

?

(1)DictController.java

/*** Copyright 2018-2020 stylefeng & fengshuonan (https://gitee.com/stylefeng)* <p>* Licensed under the Apache License, Version 2.0 (the "License");* you may not use this file except in compliance with the License.* You may obtain a copy of the License at* <p>* http://www.apache.org/licenses/LICENSE-2.0* <p>* Unless required by applicable law or agreed to in writing, software* distributed under the License is distributed on an "AS IS" BASIS,* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.* See the License for the specific language governing permissions and* limitations under the License.*/ package cn.stylefeng.guns.modular.system.controller;import cn.stylefeng.guns.core.common.annotion.BussinessLog; import cn.stylefeng.guns.core.common.annotion.Permission; import cn.stylefeng.guns.core.common.constant.Const; import cn.stylefeng.guns.core.common.constant.dictmap.DictMap; import cn.stylefeng.guns.core.common.constant.factory.ConstantFactory; import cn.stylefeng.guns.core.common.exception.BizExceptionEnum; import cn.stylefeng.guns.core.log.LogObjectHolder; import cn.stylefeng.guns.modular.system.model.Dict; import cn.stylefeng.guns.modular.system.service.IDictService; import cn.stylefeng.guns.modular.system.warpper.DictWarpper; import cn.stylefeng.roses.core.base.controller.BaseController; import cn.stylefeng.roses.core.util.ToolUtil; import cn.stylefeng.roses.kernel.model.exception.ServiceException; import com.baomidou.mybatisplus.mapper.EntityWrapper; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Controller; import org.springframework.ui.Model; import org.springframework.web.bind.annotation.*;import java.util.List; import java.util.Map;/*** 字典控制器** @author fengshuonan* @Date 2017年4月26日 12:55:31*/ @Controller @RequestMapping("/dict") public class DictController extends BaseController {private String PREFIX = "/system/dict/";@Autowiredprivate IDictService dictService;/*** 跳轉到字典管理首頁*/@RequestMapping("")public String index() {return PREFIX + "dict.html";}/*** 跳轉到添加字典*/@RequestMapping("/dict_add")public String deptAdd() {return PREFIX + "dict_add.html";}/*** 跳轉到修改字典*/@Permission(Const.ADMIN_NAME)@RequestMapping("/dict_edit/{dictId}")public String deptUpdate(@PathVariable Integer dictId, Model model) {Dict dict = dictService.selectById(dictId);model.addAttribute("dict", dict);List<Dict> subDicts = dictService.selectList(new EntityWrapper<Dict>().eq("pid", dictId));model.addAttribute("subDicts", subDicts);LogObjectHolder.me().set(dict);return PREFIX + "dict_edit.html";}/*** 新增字典** @param dictValues 格式例如 "1:啟用;2:禁用;3:凍結"*/@BussinessLog(value = "添加字典記錄", key = "dictName,dictValues", dict = DictMap.class)@RequestMapping(value = "/add")@Permission(Const.ADMIN_NAME)@ResponseBodypublic Object add(@RequestParam(required = false,defaultValue = "0",value = "dictNum")Integer dictNum,String dictCode, String dictTips, String dictName, String dictValues) {if (ToolUtil.isOneEmpty(dictCode, dictValues,dictName)) {throw new ServiceException(BizExceptionEnum.REQUEST_NULL);}this.dictService.addDict(dictCode, dictName, dictTips, dictValues,dictNum);return SUCCESS_TIP;}/*** 修改字典*/@BussinessLog(value = "修改字典", key = "dictName,dictValues", dict = DictMap.class)@RequestMapping(value = "/update")@Permission(Const.ADMIN_NAME)@ResponseBodypublic Object update(@RequestParam(required = false,defaultValue = "0",value = "dictNum")Integer dictNum,Integer dictId, String dictCode, String dictName, String dictTips, String dictValues) {if (ToolUtil.isOneEmpty(dictId, dictCode, dictName, dictValues)) {throw new ServiceException(BizExceptionEnum.REQUEST_NULL);}dictService.editDict(dictId, dictCode, dictName, dictTips, dictValues,dictNum);return SUCCESS_TIP;}/*** 獲取所有字典列表*/@RequestMapping(value = "/list")@Permission(Const.ADMIN_NAME)@ResponseBodypublic Object list(String condition) {List<Map<String, Object>> list = this.dictService.list(condition);return super.warpObject(new DictWarpper(list));}/*** 字典詳情*/@RequestMapping(value = "/detail/{dictId}")@Permission(Const.ADMIN_NAME)@ResponseBodypublic Object detail(@PathVariable("dictId") Integer dictId) {return dictService.selectById(dictId);}/*** 刪除字典記錄*/@BussinessLog(value = "刪除字典記錄", key = "dictId", dict = DictMap.class)@RequestMapping(value = "/delete")@Permission(Const.ADMIN_NAME)@ResponseBodypublic Object delete(@RequestParam Integer dictId) {//緩存被刪除的名稱LogObjectHolder.me().set(ConstantFactory.me().getDictName(dictId));this.dictService.delteDict(dictId);return SUCCESS_TIP;}}

?

(2)DictServiceImpl.java

/*** Copyright 2018-2020 stylefeng & fengshuonan (https://gitee.com/stylefeng)* <p>* Licensed under the Apache License, Version 2.0 (the "License");* you may not use this file except in compliance with the License.* You may obtain a copy of the License at* <p>* http://www.apache.org/licenses/LICENSE-2.0* <p>* Unless required by applicable law or agreed to in writing, software* distributed under the License is distributed on an "AS IS" BASIS,* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.* See the License for the specific language governing permissions and* limitations under the License.*/ package cn.stylefeng.guns.modular.system.service.impl;import cn.stylefeng.guns.core.common.exception.BizExceptionEnum; import cn.stylefeng.guns.elephish.utils.StringUtil; import cn.stylefeng.guns.modular.system.dao.DictMapper; import cn.stylefeng.guns.modular.system.model.Dict; import cn.stylefeng.guns.modular.system.service.IDictService; import cn.stylefeng.roses.kernel.model.exception.ServiceException; import com.baomidou.mybatisplus.mapper.EntityWrapper; import com.baomidou.mybatisplus.mapper.Wrapper; import com.baomidou.mybatisplus.service.impl.ServiceImpl; import org.apache.commons.lang3.StringUtils; import org.apache.commons.lang3.math.NumberUtils; import org.springframework.stereotype.Service; import org.springframework.transaction.annotation.Transactional;import javax.annotation.Resource; import java.util.*;/*** 字典服務** @author fengshuonan* @Date 2018/10/15 下午11:39*/ @Service public class DictServiceImpl extends ServiceImpl<DictMapper, Dict> implements IDictService {@Resourceprivate DictMapper dictMapper;@Override@Transactional//為了保證事務一致性,必須加上這個注解public void addDict(String dictCode, String dictName, String dictTips, String dictValues, Integer dictNum) {//添加字典Dict dict = new Dict();dict.setName(dictName);dict.setCode(dictCode);dict.setTips(dictTips);dict.setNum(dictNum);//此處傳入設置的numdict.setPid(0);//執(zhí)行字典名的添加操作int count = dictMapper.insert(dict);if(count==0) {//FAIL_ADD_RECORD(500,"數據庫中新增數據失敗"),throw new ServiceException(BizExceptionEnum.FAIL_ADD_RECORD);}//因為要保證字典名的唯一性,所以需要再查詢一遍,判斷是否沖突Wrapper<Dict> wrapper = new EntityWrapper<>();wrapper.eq("pid",0).eq("name",dictName);count = dictMapper.selectCount(wrapper);if(count>1){//DICT_NAME_EXISTED(400,"字典名稱已經存在"),throw new ServiceException(BizExceptionEnum.DICT_NAME_EXISTED);}//添加字典條目addChildrenDict(dictValues,dict.getId());}/*** 公共的添加字典條目*/private void addChildrenDict( String dictValues, Integer pid){//解析dictValuesList<Dict> list = getDictList(dictValues);//添加字典條目Dict item = null;for(int i=0; i<list.size(); i++ ){item = list.get(i);item.setPid(pid);}//直接調用批量插入語句insertBatch(list);}@Override@Transactionalpublic void editDict(Integer dictId, String dictCode, String dictName, String dictTips, String dicts,Integer dictNum) {//更新當前這個字典名稱Dict dict = new Dict();dict.setName(dictName);dict.setCode(dictCode);dict.setTips(dictTips);dict.setNum(dictNum);dict.setId(dictId);dictMapper.updateById(dict);//刪除這個字典的所有子字典Wrapper<Dict> wrapper = new EntityWrapper<>();wrapper.eq("pid", dictId);dictMapper.delete(wrapper);//添加新的字典條目addChildrenDict(dicts,dictId);}/*** 解析一個組合字符串(例如: "1:啟用;2:禁用;3:凍結" 這樣的字符串)* 本方法是將字符串直接轉換成對應的Dict實體類集合,* 如果前端傳入參數的方式錯誤* 或者用戶輸入的參數不合法,此處都會拋出異常,終止程序*/private List<Dict> getDictList(String mutiString) {List<Dict> list = new ArrayList<>();//判斷分隔后是否為空數組List<String> split = StringUtil.split(mutiString,";");if(split.isEmpty()){ // ERROR_ITEM_EMPTY(500, "字典詳情不能為空"),throw new ServiceException(BizExceptionEnum.ERROR_ITEM_EMPTY);}//用來封裝已經存在的code參數,如果有重復的值,說明前端傳入的值錯誤了,//此時也應該拋出異常Set<String> codeSet = new HashSet<>();Set<String> nameSet = new HashSet<>();Dict dict = null;List<String> attrs = null;String attr = null;for (String item : split) {dict = new Dict();//如果數組長度不為3,說明前端的傳入數據錯誤,此時請求應終止attrs = StringUtil.split(item,":");if(attrs.size() !=3){//DICT_PARAMS_ILLEGAL(400,"請?zhí)顚懲暾淖值湓斍樾畔?#34;),throw new ServiceException(BizExceptionEnum.DICT_PARAMS_ILLEGAL);}//保證code值的唯一attr = attrs.get(0);if(codeSet.contains(attr)){throw new ServiceException(400,"值【"+attr+"】重復");}codeSet.add(attr);dict.setCode(attr);//為了字典的可讀性,我這邊也要保證name字段的一致attr = attrs.get(1);if(nameSet.contains(attr)){throw new ServiceException(400,"名稱【"+attr+"】重復");}nameSet.add(attr);dict.setName(attr);//保存排序編號if(!NumberUtils.isDigits(attrs.get(2))){//ERROR_NUM_TYPE(500, "字典詳情序號只能是數字"),throw new ServiceException(BizExceptionEnum.DICT_MUST_BE_NUMBER);}dict.setNum(new Integer(attrs.get(2)));list.add(dict);}return list;}@Overridepublic List<Map<String, Object>> list(String conditiion) {Wrapper<Dict> wrapper = new EntityWrapper<>();if(StringUtils.isNoneBlank(conditiion)){wrapper.like("name",conditiion.trim());}//此處添加按num字段順序排序wrapper.eq("pid",0).orderBy("num",true);return dictMapper.selectMaps(wrapper);}@Override@Transactionalpublic void delteDict(Integer dictId) {//刪除這個字典的子詞典Wrapper<Dict> dictEntityWrapper = new EntityWrapper<>();dictEntityWrapper = dictEntityWrapper.eq("pid", dictId);dictMapper.delete(dictEntityWrapper);//刪除這個詞典dictMapper.deleteById(dictId);}@Overridepublic List<Dict> selectByCode(String code) {return this.baseMapper.selectByCode(code);}@Overridepublic List<Dict> selectByParentCode(String code) {return this.baseMapper.selectByParentCode(code);}}

?

(3)UserWarpper.java

/*** Copyright 2018-2020 stylefeng & fengshuonan (https://gitee.com/stylefeng)* <p>* Licensed under the Apache License, Version 2.0 (the "License");* you may not use this file except in compliance with the License.* You may obtain a copy of the License at* <p>* http://www.apache.org/licenses/LICENSE-2.0* <p>* Unless required by applicable law or agreed to in writing, software* distributed under the License is distributed on an "AS IS" BASIS,* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.* See the License for the specific language governing permissions and* limitations under the License.*/ package cn.stylefeng.guns.modular.system.warpper;import cn.stylefeng.guns.core.common.constant.factory.ConstantFactory; import cn.stylefeng.guns.elephish.constants.WrapperDictNameConstant; import cn.stylefeng.roses.core.base.warpper.BaseControllerWrapper; import cn.stylefeng.roses.kernel.model.page.PageResult; import com.baomidou.mybatisplus.plugins.Page;import java.util.List; import java.util.Map;/*** 用戶管理的包裝類** @author fengshuonan* @date 2017年2月13日 下午10:47:03*/ public class UserWarpper extends BaseControllerWrapper {public UserWarpper(Map<String, Object> single) {super(single);}public UserWarpper(List<Map<String, Object>> multi) {super(multi);}public UserWarpper(Page<Map<String, Object>> page) {super(page);}public UserWarpper(PageResult<Map<String, Object>> pageResult) {super(pageResult);}/*** 此處自定義添加需要包裝的字段的字典信息*/@Overrideprotected void wrapTheMap(Map<String, Object> map) {//這段代碼表示,我需要在返回值中添加一個屬性 sexName,// 這個seName是字段sex對應的字典解釋信息,比如如果sex=1,那么sexName=男 // map.put("sexName", ConstantFactory.me().getSexName( (Integer)map.get("sex")));map.put("sexName", ConstantFactory.me().getDictById(WrapperDictNameConstant.SYS_USER_SEX_ID,map.get("sex").toString()));map.put("roleName", ConstantFactory.me().getRoleName((String) map.get("roleid")));map.put("deptName", ConstantFactory.me().getDeptName((Integer) map.get("deptid")));map.put("statusName", ConstantFactory.me().getStatusName((Integer) map.get("status")));}}

?

?

(4)ConstantFactory.java

/*** Copyright 2018-2020 stylefeng & fengshuonan (https://gitee.com/stylefeng)* <p>* Licensed under the Apache License, Version 2.0 (the "License");* you may not use this file except in compliance with the License.* You may obtain a copy of the License at* <p>* http://www.apache.org/licenses/LICENSE-2.0* <p>* Unless required by applicable law or agreed to in writing, software* distributed under the License is distributed on an "AS IS" BASIS,* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.* See the License for the specific language governing permissions and* limitations under the License.*/ package cn.stylefeng.guns.core.common.constant.factory;import cn.hutool.core.convert.Convert; import cn.hutool.core.util.StrUtil; import cn.stylefeng.guns.core.common.constant.cache.Cache; import cn.stylefeng.guns.core.common.constant.cache.CacheKey; import cn.stylefeng.guns.core.common.constant.state.ManagerStatus; import cn.stylefeng.guns.core.common.constant.state.MenuStatus; import cn.stylefeng.guns.core.log.LogObjectHolder; import cn.stylefeng.guns.modular.system.dao.*; import cn.stylefeng.guns.modular.system.model.*; import cn.stylefeng.roses.core.util.SpringContextHolder; import cn.stylefeng.roses.core.util.ToolUtil; import com.baomidou.mybatisplus.mapper.EntityWrapper; import com.baomidou.mybatisplus.mapper.Wrapper; import lombok.val; import org.apache.commons.lang3.StringUtils; import org.springframework.cache.annotation.Cacheable; import org.springframework.context.annotation.DependsOn; import org.springframework.stereotype.Component;import java.util.ArrayList; import java.util.List;/*** 常量的生產工廠** @author fengshuonan* @date 2017年2月13日 下午10:55:21*/ @Component @DependsOn("springContextHolder") public class ConstantFactory implements IConstantFactory {private RoleMapper roleMapper = SpringContextHolder.getBean(RoleMapper.class);private DeptMapper deptMapper = SpringContextHolder.getBean(DeptMapper.class);private DictMapper dictMapper = SpringContextHolder.getBean(DictMapper.class);private UserMapper userMapper = SpringContextHolder.getBean(UserMapper.class);private MenuMapper menuMapper = SpringContextHolder.getBean(MenuMapper.class);private NoticeMapper noticeMapper = SpringContextHolder.getBean(NoticeMapper.class);public static IConstantFactory me() {return SpringContextHolder.getBean("constantFactory");}/*** 根據用戶id獲取用戶名稱** @author stylefeng* @Date 2017/5/9 23:41*/@Overridepublic String getUserNameById(Integer userId) {User user = userMapper.selectById(userId);if (user != null) {return user.getName();} else {return "--";}}/*** 根據用戶id獲取用戶賬號** @author stylefeng* @date 2017年5月16日21:55:371*/@Overridepublic String getUserAccountById(Integer userId) {User user = userMapper.selectById(userId);if (user != null) {return user.getAccount();} else {return "--";}}/*** 通過角色ids獲取角色名稱*/@Override@Cacheable(value = Cache.CONSTANT, key = "'" + CacheKey.ROLES_NAME + "'+#roleIds")public String getRoleName(String roleIds) {if (ToolUtil.isEmpty(roleIds)) {return "";}Integer[] roles = Convert.toIntArray(roleIds);StringBuilder sb = new StringBuilder();for (int role : roles) {Role roleObj = roleMapper.selectById(role);if (ToolUtil.isNotEmpty(roleObj) && ToolUtil.isNotEmpty(roleObj.getName())) {sb.append(roleObj.getName()).append(",");}}return StrUtil.removeSuffix(sb.toString(), ",");}/*** 通過角色id獲取角色名稱*/@Override@Cacheable(value = Cache.CONSTANT, key = "'" + CacheKey.SINGLE_ROLE_NAME + "'+#roleId")public String getSingleRoleName(Integer roleId) {if (0 == roleId) {return "--";}Role roleObj = roleMapper.selectById(roleId);if (ToolUtil.isNotEmpty(roleObj) && ToolUtil.isNotEmpty(roleObj.getName())) {return roleObj.getName();}return "";}/*** 通過角色id獲取角色英文名稱*/@Override@Cacheable(value = Cache.CONSTANT, key = "'" + CacheKey.SINGLE_ROLE_TIP + "'+#roleId")public String getSingleRoleTip(Integer roleId) {if (0 == roleId) {return "--";}Role roleObj = roleMapper.selectById(roleId);if (ToolUtil.isNotEmpty(roleObj) && ToolUtil.isNotEmpty(roleObj.getName())) {return roleObj.getTips();}return "";}/*** 獲取部門名稱*/@Override@Cacheable(value = Cache.CONSTANT, key = "'" + CacheKey.DEPT_NAME + "'+#deptId")public String getDeptName(Integer deptId) {Dept dept = deptMapper.selectById(deptId);if (ToolUtil.isNotEmpty(dept) && ToolUtil.isNotEmpty(dept.getFullname())) {return dept.getFullname();}return "";}/*** 獲取菜單的名稱們(多個)*/@Overridepublic String getMenuNames(String menuIds) {Integer[] menus = Convert.toIntArray(menuIds);StringBuilder sb = new StringBuilder();for (int menu : menus) {Menu menuObj = menuMapper.selectById(menu);if (ToolUtil.isNotEmpty(menuObj) && ToolUtil.isNotEmpty(menuObj.getName())) {sb.append(menuObj.getName()).append(",");}}return StrUtil.removeSuffix(sb.toString(), ",");}/*** 獲取菜單名稱*/@Overridepublic String getMenuName(Long menuId) {if (ToolUtil.isEmpty(menuId)) {return "";} else {Menu menu = menuMapper.selectById(menuId);if (menu == null) {return "";} else {return menu.getName();}}}/*** 獲取菜單名稱通過編號*/@Overridepublic String getMenuNameByCode(String code) {if (ToolUtil.isEmpty(code)) {return "";} else {Menu param = new Menu();param.setCode(code);Menu menu = menuMapper.selectOne(param);if (menu == null) {return "";} else {return menu.getName();}}}/*** 獲取字典名稱*/@Overridepublic String getDictName(Integer dictId) {if (ToolUtil.isEmpty(dictId)) {return "";} else {Dict dict = dictMapper.selectById(dictId);if (dict == null) {return "";} else {return dict.getName();}}}/*** 獲取通知標題*/@Overridepublic String getNoticeTitle(Integer dictId) {if (ToolUtil.isEmpty(dictId)) {return "";} else {Notice notice = noticeMapper.selectById(dictId);if (notice == null) {return "";} else {return notice.getTitle();}}}/*** 獲取性別名稱*/@Overridepublic String getSexName(Integer sex) {return getDictsByName("性別", sex);}@Overridepublic String getDictByNameAndCode(String name, String code) {/*** 判斷是否*/if (StringUtils.isBlank(name) || StringUtils.isBlank(code)) {return "";}//通過name字段查詢父類字典Dict temp = new Dict();temp.setName(name);temp.setPid(0);Dict dict = dictMapper.selectOne(temp);if(dict==null){return "";}Wrapper<Dict> wrapper = new EntityWrapper<>();//遍歷操作太過耗費內存資源,不如直接查結果wrapper.setSqlSelect("name")//提高封裝效率,只查一個字段.eq("pid", dict.getId()).eq("code", code.trim());//理論上只會查詢到一條字段,但是為了安全起見,還是查詢集合,并且只取第一條記錄List<Dict> dicts = dictMapper.selectList(wrapper);//mybatis-plus返回的list不會為null,如果沒有查到結果,會返回空集合if (!dicts.isEmpty()) {return dicts.get(0).getName();}return "";}@Overridepublic String getDictById(Integer id, String code) {//判斷是否合法參數if(id==null || StringUtils.isBlank(code)){return "";}Wrapper<Dict> wrapper = new EntityWrapper<>();//遍歷操作太過耗費內存資源,不如直接查結果wrapper.setSqlSelect("name")//提高查詢后的封裝效率,只查一個字段.eq("pid", id).eq("code", code.trim());//理論上只會查詢到一條字段,但是為了安全起見,還是查詢集合,并且只取第一條記錄List<Dict> dicts = dictMapper.selectList(wrapper);//mybatis-plus返回的list不會為null,如果沒有查到結果,會返回空集合if(!dicts.isEmpty()){return dicts.get(0).getName();}return "";}/*** 根據字典名稱和字典中的值獲取對應的名稱*/@Overridepublic String getDictsByName(String name, Integer val) {//判斷是否為nullif(StringUtils.isBlank(name) ||val == null){return "";}//通過name字段查詢父類字典Dict temp = new Dict();temp.setName(name);Dict dict = dictMapper.selectOne(temp);if (dict == null) {return "";} else {Wrapper<Dict> wrapper = new EntityWrapper<>();//遍歷操作太過耗費內存資源,不如直接查結果wrapper.eq("pid", dict.getId()).eq("code",val.toString());//注意,要轉換成字符串后再傳入參數//有可能會查詢到多條記錄,因為添加的時候沒有做code的唯一判斷List<Dict> dicts = dictMapper.selectList(wrapper);//mybatis-plus返回的list不會為null,如果沒有查到結果,會返回空集合if(!dicts.isEmpty()){return dicts.get(0).getName();}return "";// wrapper = wrapper.eq("pid", dict.getId()); // List<Dict> dicts = dictMapper.selectList(wrapper); // for (Dict item : dicts) { // if (item.getNum() != null && item.getNum().equals(val)) { // return item.getName(); // } // }}}/*** 獲取用戶登錄狀態(tài)*/@Overridepublic String getStatusName(Integer status) {return ManagerStatus.valueOf(status);}/*** 獲取菜單狀態(tài)*/@Overridepublic String getMenuStatusName(Integer status) {return MenuStatus.valueOf(status);}/*** 查詢字典*/@Overridepublic List<Dict> findInDict(Integer id) {if (ToolUtil.isEmpty(id)) {return null;} else {EntityWrapper<Dict> wrapper = new EntityWrapper<>();List<Dict> dicts = dictMapper.selectList(wrapper.eq("pid", id));if (dicts == null || dicts.size() == 0) {return null;} else {return dicts;}}}/*** 獲取被緩存的對象(用戶刪除業(yè)務)*/@Overridepublic String getCacheObject(String para) {return LogObjectHolder.me().get().toString();}/*** 獲取子部門id*/@Overridepublic List<Integer> getSubDeptId(Integer deptid) {Wrapper<Dept> wrapper = new EntityWrapper<>();wrapper = wrapper.like("pids", "%[" + deptid + "]%");List<Dept> depts = this.deptMapper.selectList(wrapper);ArrayList<Integer> deptids = new ArrayList<>();if (depts != null && depts.size() > 0) {for (Dept dept : depts) {deptids.add(dept.getId());}}return deptids;}/*** 獲取所有父部門id*/@Overridepublic List<Integer> getParentDeptIds(Integer deptid) {Dept dept = deptMapper.selectById(deptid);String pids = dept.getPids();String[] split = pids.split(",");ArrayList<Integer> parentDeptIds = new ArrayList<>();for (String s : split) {parentDeptIds.add(Integer.valueOf(StrUtil.removeSuffix(StrUtil.removePrefix(s, "["), "]")));}return parentDeptIds;}}

?

?

(5)IConstantFactory.java

/*** Copyright 2018-2020 stylefeng & fengshuonan (https://gitee.com/stylefeng)* <p>* Licensed under the Apache License, Version 2.0 (the "License");* you may not use this file except in compliance with the License.* You may obtain a copy of the License at* <p>* http://www.apache.org/licenses/LICENSE-2.0* <p>* Unless required by applicable law or agreed to in writing, software* distributed under the License is distributed on an "AS IS" BASIS,* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.* See the License for the specific language governing permissions and* limitations under the License.*/ package cn.stylefeng.guns.core.common.constant.factory;import cn.stylefeng.guns.modular.system.model.Dict; import io.swagger.models.auth.In;import java.util.List;/*** 常量生產工廠的接口** @author fengshuonan* @date 2017-06-14 21:12*/ public interface IConstantFactory {/*** 根據用戶id獲取用戶名稱** @author stylefeng* @Date 2017/5/9 23:41*/String getUserNameById(Integer userId);/*** 根據用戶id獲取用戶賬號** @author stylefeng* @date 2017年5月16日21:55:371*/String getUserAccountById(Integer userId);/*** 通過角色ids獲取角色名稱*/String getRoleName(String roleIds);/*** 通過角色id獲取角色名稱*/String getSingleRoleName(Integer roleId);/*** 通過角色id獲取角色英文名稱*/String getSingleRoleTip(Integer roleId);/*** 獲取部門名稱*/String getDeptName(Integer deptId);/*** 獲取菜單的名稱們(多個)*/String getMenuNames(String menuIds);/*** 獲取菜單名稱*/String getMenuName(Long menuId);/*** 獲取菜單名稱通過編號*/String getMenuNameByCode(String code);/*** 獲取字典名稱*/String getDictName(Integer dictId);/*** 獲取通知標題*/String getNoticeTitle(Integer dictId);/*** 根據字典名稱和字典中的值獲取對應的名稱*/String getDictsByName(String name, Integer val); // String getDictsByName(String name, String val);/*** 根據字典名稱和字典中的值獲取對應的名稱*/String getDictByNameAndCode(String name,String code);/*** 根據字典父類id和對應的子類字典的code獲取對應的子類字典名稱* @param id 字典的父類id* @param code 字典的子類id* @return*/String getDictById(Integer id,String code);/*** 獲取性別 */String getSexName(Integer sex); // String getSexName(String sex);/*** 獲取用戶登錄狀態(tài)*/String getStatusName(Integer status);/*** 獲取菜單狀態(tài)*/String getMenuStatusName(Integer status);/*** 查詢字典*/List<Dict> findInDict(Integer id);/*** 獲取被緩存的對象(用戶刪除業(yè)務)*/String getCacheObject(String para);/*** 獲取子部門id*/List<Integer> getSubDeptId(Integer deptid);/*** 獲取所有父部門id*/List<Integer> getParentDeptIds(Integer deptid);}

?

(6)WrapperDictNameConstant.java

package cn.stylefeng.guns.elephish.constants;/**** 字典包裝類的字典名稱常量** Created by hqq on 2020/4/15.*/ public interface WrapperDictNameConstant {/*** sys_user 表中 sex 字段對應的字典名稱 的ID*/int SYS_USER_SEX_ID = 127;}

?

(7)StringUtil.java

package cn.stylefeng.guns.elephish.utils;import org.apache.commons.lang3.StringUtils; import org.apache.commons.lang3.math.NumberUtils; import org.apache.velocity.runtime.directive.Foreach;import java.util.ArrayList; import java.util.List;/*** 自定義的字符串工具類* Created by hqq on 2020/4/15.*/ public class StringUtil {/*** 自定義字符串split方法,并去除每一個元素的首尾空格* 如果傳入的參數是null或空串,容量為0的字符串集合* 如果分隔后的結果為null或空數組,返回容量為0的集合,防止調用者空指針異常** 示例如:split("aa, bb ,c c," , ",");* 得到結果: ["aa","bb","c c"]** @param mutiString 需要進行分割的字符串* @param separatorChars 分割標志* @return*/public static List<String> split(String mutiString,final String separatorChars){//定義一個list封裝最后的結果List<String> result = new ArrayList<>();if(StringUtils.isBlank(mutiString)){return result;}//使用性能更好的StringUtils.split方法String[] split = StringUtils.split(mutiString,separatorChars);if(split==null || split.length<1){return result;}//去除首尾空格for (int i = 0; i<split.length; i++) {if(StringUtils.isBlank(split[i])){continue;}result.add(split[i].trim());}return result;}}

?

?

至此,分享結束!

?

? ? ? ? 寫了這么多,目的只有一個,就是通過帶領大家修改guns源碼來加深對guns的理解,以此幫助大家更高效的對Guns進行二次開發(fā),我們對guns上手的越快,那我們就能更多的將時間花在我們自己的業(yè)務上。另外,有些人可能會抗拒這種修改,我想說的是,? ultimate v2.5 版本的Guns,也對字典模塊進行了較大的改變,所以,就連guns作者也在進步,因為不進步,就只能被社會淘汰,哪怕你曾經很牛逼!

?

該系列更多文章請前往?Guns二次開發(fā)目錄

?

總結

以上是生活随笔為你收集整理的Guns二次开发(四):重构字典管理模块的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

欧美日韩久久久 | 一区二区三区精品在线视频 | 久久亚洲视频 | 中文字幕.av.在线 | 97av精品 | 99久久电影 | 五月激情综合婷婷 | h视频日本 | 黄色91在线| 在线观看av小说 | 日本韩国欧美在线观看 | 日韩av片无码一区二区不卡电影 | 果冻av在线 | 国产精品毛片一区二区在线看 | 国产成人一区二区精品非洲 | 亚洲自拍av在线 | 精品视频一区在线 | 成人黄色电影免费观看 | 国产一区精品在线观看 | 国产高潮久久 | 人人爽人人爽人人爽 | 欧美激情在线看 | 在线只有精品 | japanesefreesexvideo高潮 | 国产xxxxx在线观看 | 久久激情片 | 久久久久一区二区三区 | 五月婷婷免费 | 日韩动态视频 | 国产不卡av在线播放 | 欧美成人精品欧美一级乱 | 国产一区二区高清不卡 | 国产一级片一区二区三区 | 国产精品成人久久久久久久 | 成年人免费看av | 69av网| 国产精品久久久久久久久久尿 | 欧美二区在线播放 | 一级黄色电影网站 | 天堂中文在线视频 | 中文字幕视频免费观看 | 波多野结衣电影久久 | 日日碰狠狠躁久久躁综合网 | 亚洲视频在线免费观看 | 国产字幕av| 成人网大片 | 99色免费视频 | 亚洲免费精彩视频 | 不卡的av中文字幕 | 五月婷香蕉久色在线看 | 97在线观看视频 | 国产一区在线观看视频 | 日本精品久久久久影院 | 国产美女被啪进深处喷白浆视频 | 欧美另类调教 | 色姑娘综合天天 | 日本二区三区在线 | 成人在线视频免费看 | 国产伦理一区二区三区 | 丁香电影小说免费视频观看 | 久久久久久黄色 | 国产精品白浆视频 | 日韩视频a | 久久精品观看 | 色视频在线免费 | 国产精品成人国产乱 | 国产精品美女久久久久久久久久久 | 国产玖玖在线 | 夜夜夜草| 欧美日韩不卡一区二区三区 | 久久精品视频在线 | 综合色中文| 在线观看日韩专区 | av高清网站在线观看 | 亚洲一级电影在线观看 | 国产专区视频在线观看 | 91精品一区国产高清在线gif | 免费在线黄色av | 字幕网av| 一区二区av | 国内精品国产三级国产aⅴ久 | 特级西西444www大胆高清无视频 | 夜添久久精品亚洲国产精品 | 91成熟丰满女人少妇 | 精品福利网站 | 久久精品美女 | 久久综合久久综合这里只有精品 | 成年人黄色免费视频 | japanesexxxhd奶水| 欧美aa在线观看 | 色资源在线 | 久久不射电影网 | 亚洲一区动漫 | 免费看成人片 | 国产美女精品久久久 | 成人av网址大全 | 欧美另类成人 | 日韩视频中文字幕 | 久久 在线 | 欧美性色黄大片在线观看 | 不卡的av电影在线观看 | 99久久夜色精品国产亚洲96 | 久久精品99久久 | 91丨九色丨国产在线观看 | 国产九九在线 | 久久不射电影网 | 激情视频综合网 | 最近日本韩国中文字幕 | 亚洲欧洲精品久久 | 国产精品mm| 韩日av在线 | 在线观看免费中文字幕 | 97福利在线观看 | 久久国产精品久久国产精品 | av在线观 | 在线 国产 日韩 | 韩日成人av | 亚洲一区二区黄色 | 国产一区二区在线观看免费 | 中文字幕在线影视资源 | 欧美久久电影 | 精品久久一级片 | 日韩免费网址 | 成人污视频在线观看 | 亚洲午夜av久久乱码 | av线上免费观看 | 久久人91精品久久久久久不卡 | 久久精品视频国产 | 一区二区三区在线不卡 | 97超碰在线久草超碰在线观看 | 国产剧情一区在线 | 免费久久99精品国产婷婷六月 | 国产91精品一区二区麻豆亚洲 | 在线 欧美 日韩 | 亚洲h视频在线 | 2022国产精品视频 | 99精品久久只有精品 | 国产精品专区h在线观看 | 国产涩图| 91精品国产欧美一区二区 | 成人久久18免费网站图片 | 国产成人亚洲在线电影 | 婷婷激情5月天 | 999久久久久久久久 69av视频在线观看 | 日韩精品一区电影 | 亚洲不卡av一区二区三区 | 亚洲午夜久久久综合37日本 | 99久久久国产精品免费观看 | 久久手机在线视频 | 精品视频99| 伊人午夜视频 | 中文字幕免费高清在线 | 欧美激情视频在线免费观看 | 不卡的av在线播放 | 91久久电影| 久久午夜精品影院一区 | 国产福利小视频在线 | 久久久久久毛片精品免费不卡 | 久久免费的精品国产v∧ | 免费日韩 精品中文字幕视频在线 | 欧美日韩在线精品一区二区 | 国产精品一区二区免费在线观看 | 国模视频一区二区三区 | 日日夜夜狠狠操 | 丁香花在线视频观看免费 | 欧美久久久久久久久久久久久 | 91视频最新网址 | 天堂av一区二区 | 开心激情婷婷 | 99 色| 久久久国产精品成人免费 | 在线免费亚洲 | 亚洲成人资源 | 碰超在线观看 | 国产伦理久久精品久久久久_ | 国产小视频福利在线 | 日韩视频一区二区三区 | 91av视频在线免费观看 | 亚洲国产欧美在线看片xxoo | 国产精品一区在线观看 | 久久一级电影 | 天天射夜夜爽 | 成片视频免费观看 | 最近中文字幕在线中文高清版 | 国产日韩欧美精品在线观看 | 成人国产综合 | 久久久久久久久免费视频 | 久久免费视频精品 | 日韩在线免费播放 | 久久毛片视频 | 国产一区二区三区 在线 | 91色亚洲| 成人午夜av电影 | 久久久久亚洲精品成人网小说 | 国产精国产精品 | 亚洲精品h | 色播亚洲婷婷 | 一级黄色大片在线观看 | 亚洲精品视频网 | 国产精品久久久久三级 | 91传媒免费在线观看 | 亚洲一区二区视频 | 视频二区在线视频 | 在线观看电影av | av大片免费 | 激情av在线资源 | 日本成人黄色片 | 久久免费激情视频 | 久久草精品 | 亚洲精品国偷拍自产在线观看蜜桃 | 精品国产电影一区二区 | 91精品国产高清自在线观看 | 四虎成人精品永久免费av | 91在线精品视频 | 免费观看9x视频网站在线观看 | 91 在线视频 | 2023年中文无字幕文字 | 九九热免费在线观看 | 久久电影网站中文字幕 | 久久久久久亚洲精品 | 日韩毛片精品 | 少妇bbw搡bbbb搡bbbb | 亚洲涩涩涩 | www.夜夜爱| 在线观看国产中文字幕 | 精品免费视频. | 国产精品96久久久久久吹潮 | 久99精品 | 精品一区二区综合 | 成年人免费在线观看网站 | 国产丝袜 | 日本在线精品视频 | 国产成人av在线影院 | 日韩精品三区四区 | 麻豆传媒视频在线免费观看 | 日本不卡视频 | 免费国产黄线在线观看视频 | 欧美精品一区二区在线播放 | 国产99在线 | 国产高清视频免费观看 | 91污在线 | 精品美女在线视频 | 欧美资源在线观看 | 在线观看免费色 | 国产精品电影在线 | 91看成人| 4438全国亚洲精品在线观看视频 | www.五月天| 九九热在线精品视频 | 亚洲国产精品第一区二区 | www.色五月.com | 亚洲第一久久久 | 91激情小视频 | 伊人宗合网| 精品乱码一区二区三四区 | 国产高清视频在线播放一区 | 在线免费观看视频一区 | 丁香六月国产 | 国产九九热 | 日韩午夜电影院 | 99在线精品视频在线观看 | 91av社区 | 天天干夜夜操视频 | 中文字幕av专区 | 日本在线精品视频 | 在线中文字幕av观看 | 成人欧美在线 | 欧美三级在线播放 | 男女啪啪免费网站 | 日韩电影在线观看中文字幕 | 精品国产免费人成在线观看 | 婷婷久久一区 | 日本黄区免费视频观看 | 伊人午夜视频 | 久久久亚洲国产精品麻豆综合天堂 | 99国产一区二区三精品乱码 | 中文字幕4 | 免费视频黄 | 亚洲另类视频在线 | 国内成人精品2018免费看 | 国产精品免费高清 | 色欧美日韩 | 国产91精品一区二区麻豆网站 | 久久在线免费观看视频 | 欧美午夜理伦三级在线观看 | 女人高潮一级片 | 四虎影视成人精品国库在线观看 | 久久免费av | 中文国产在线观看 | 免费一级片观看 | 欧美成人黄| 婷婷五月色综合 | 亚洲一区二区视频在线播放 | 欧美一级大片在线观看 | 久久久视屏 | 黄色一级免费电影 | 一区二区三区久久 | 免费一级特黄毛大片 | 五月婷久久 | 国产精品成久久久久三级 | 黄色a级片在线观看 | 国产成人一区二区精品非洲 | 中文字幕.av.在线 | 一区二区免费不卡在线 | 中文字幕一区2区3区 | 西西4444www大胆视频 | 久久久久激情 | 久久久九色精品国产一区二区三区 | 久久成人免费 | 三级在线视频观看 | 91九色网站 | 一区二区视频欧美 | 开心激情五月网 | 久插视频 | 五月激情在线 | 免费在线视频一区二区 | 久久69精品 | 国产免费黄色 | 91九色视频观看 | 国产高清一级 | 免费精品久久久 | 特级毛片在线 | 高清中文字幕av | 激情欧美一区二区免费视频 | 亚洲最新毛片 | 色婷婷视频网 | 亚洲精品在线视频网站 | 国产系列在线观看 | 日韩在线观看三区 | 激情视频一区二区三区 | 国内视频 | 97免费在线观看视频 | 国产精品视频不卡 | 国产人成看黄久久久久久久久 | 国产一区二区播放 | 91精品麻豆| 婷婷色中文网 | 久久久久国产精品厨房 | 亚洲国产中文字幕在线观看 | 婷婷av网站 | 亚洲欧美在线观看视频 | 日韩手机在线 | 亚洲闷骚少妇在线观看网站 | 激情图片qvod | 欧美坐爱视频 | 亚洲欧洲一区二区在线观看 | 中文字幕精品一区二区三区电影 | 美女视频久久黄 | 97夜夜澡人人爽人人免费 | 99r在线精品| 九九热只有精品 | 久久tv| 婷婷综合影院 | 综合激情久久 | 婷婷伊人五月天 | 色偷偷88888欧美精品久久久 | 日日日日 | 午夜精品三区 | 日日干精品 | 成人在线观看网址 | 丁香婷婷在线 | 精品久久久久久久久久久久久久久久 | 少妇高潮冒白浆 | 精品99在线视频 | 欧美日韩亚洲在线 | 高清国产午夜精品久久久久久 | 在线免费91 | 国产精品亚洲精品 | 亚洲一区免费在线 | 黄色av成人在线观看 | 91精品久久香蕉国产线看观看 | 色综合天天综合在线视频 | 国产一区国产二区在线观看 | 久久伊人免费视频 | 国产99久久久国产精品成人免费 | 亚洲精品免费在线视频 | 久久天天躁狠狠躁亚洲综合公司 | 超碰公开在线观看 | 欧美日韩在线网站 | 91大神免费视频 | 在线视频专区 | 久草资源在线观看 | 91成人在线网站 | 中文字幕亚洲欧美日韩2019 | 国产综合视频在线观看 | 伊人资源视频在线 | 91女人18片女毛片60分钟 | 亚洲精品午夜国产va久久成人 | 狠狠色综合欧美激情 | 在线看v片成人 | 蜜臀av性久久久久av蜜臀三区 | 在线 成人 | 免费av网站在线看 | 亚洲国产精品一区二区尤物区 | 成人v| 久久精品国产一区二区三区 | 国产精品伦一区二区三区视频 | 欧美三级高清 | 91视频在线网址 | 日本久草电影 | 国产精品福利在线播放 | 久久久久夜色 | 一级黄色片在线观看 | 天天插天天干 | 黄色午夜网站 | 欧美激情综合网 | 久久香蕉电影 | 国产xx视频 | 永久免费的啪啪网站免费观看浪潮 | 人人舔人人射 | 色吊丝在线永久观看最新版本 | 日韩精品在线播放 | 天天做日日爱夜夜爽 | 国产精品第 | 久久精品网| 久久手机视频 | 天天草视频 | 色爱区综合激月婷婷 | 天天躁天天狠天天透 | 三级av在线免费观看 | 懂色av一区二区在线播放 | 成人国产网站 | 天天操综合网 | 色综合久久久 | 日韩在线二区 | 久久久国产精品亚洲一区 | 国产午夜精品一区二区三区欧美 | 国产白浆视频 | 国产精品美女久久久网av | 2018好看的中文在线观看 | 亚洲 av网站 | 婷婷久久五月天 | 日韩mv欧美mv国产精品 | 久久99精品久久久久久秒播蜜臀 | 91在线porny国产在线看 | 麻豆传媒在线免费看 | 亚洲黄污 | 日本老少交 | 久久夜色精品国产欧美一区麻豆 | 日本精品xxxx | 人人玩人人添人人 | www.久久久com | 美女网站在线观看 | 一区二区三区在线观看免费 | 日韩av播放在线 | 在线高清一区 | 欧美夫妻性生活电影 | 91精品久久久久久综合乱菊 | 国产精品理论片在线观看 | 久av在线 | 国产精品麻豆欧美日韩ww | 欧美精品在线观看免费 | 中文字幕在线看人 | av福利网址导航大全 | 精品9999| 亚洲精品乱码久久久一二三 | 极品美女被弄高潮视频网站 | 在线国产专区 | 精品国产一区二区三区日日嗨 | 欧美日韩精 | www四虎影院 | 日韩二区在线播放 | 亚洲精品国产精品久久99热 | 欧美在线一 | 国产精品国产三级在线专区 | 狠狠狠狠狠色综合 | www.天天成人国产电影 | 国产午夜精品一区二区三区欧美 | 日本99精品 | 成人在线黄色 | 国产乱码精品一区二区蜜臀 | 九色在线视频 | 99在线精品视频在线观看 | 欧美日韩不卡一区二区 | 国产一区二区免费在线观看 | 国产精品爽爽爽 | 不卡的一区二区三区 | 久免费| 亚洲综合成人在线 | 久久只精品99品免费久23小说 | 欧美日韩国产精品一区 | 美女天天操 | 国产精品女同一区二区三区久久夜 | 久久婷婷五月综合色丁香 | 成人电影毛片 | 在线中文字幕av观看 | 制服丝袜成人在线 | 91视频麻豆 | 黄色在线免费观看网址 | 91成人亚洲| 91精选 | 欧美激情综合五月色丁香小说 | 亚洲一区 影院 | www.888.av| 亚洲精品白浆高清久久久久久 | 国产精品一区二区av麻豆 | 91资源在线 | 日韩久久一区二区 | 亚洲一二三区精品 | 国产精品h在线观看 | 国产精品欧美久久久久久 | 国产精品美乳一区二区免费 | 在线精品视频免费观看 | 久久免费久久 | 国产69精品久久久久99 | 午夜在线免费观看视频 | 天天曰天天干 | 在线播放视频一区 | 97电院网手机版 | 99综合电影在线视频 | 天堂va在线高清一区 | 国产视频欧美视频 | 成人av一区二区三区 | 亚洲精品免费在线观看 | 四虎在线免费观看 | 国产精品自产拍在线观看中文 | 91麻豆精品国产自产在线 | 婷婷色中文网 | 久久久99精品免费观看乱色 | 亚洲资源一区 | av日韩精品 | 91精品对白一区国产伦 | 国产视频精品视频 | 欧美一级在线看 | 久久久久久久久久久久久国产精品 | 高清在线一区二区 | 激情视频免费在线观看 | 超碰免费成人 | 天天久久综合 | 亚洲男模gay裸体gay | 中文字幕日韩高清 | japanesexxxhd奶水 91在线精品一区二区 | 日韩a在线| 久久久免费看视频 | 欧美日韩国产色综合一二三四 | 国产精品高清在线观看 | 97免费在线观看视频 | 精品久久一 | 91污视频在线观看 | 91精品婷婷国产综合久久蝌蚪 | 黄色av一区 | 天海翼一区二区三区免费 | 国产精品女同一区二区三区久久夜 | 91丨九色丨国产在线 | 久久国产手机看片 | 精品国偷自产国产一区 | 黄色影院在线播放 | 午夜久久久久 | 亚洲视频在线观看免费 | 中文字幕久久亚洲 | 91精品福利在线 | 国产裸体无遮挡 | 99热精品国产一区二区在线观看 | 亚洲日本韩国一区二区 | 在线免费观看视频你懂的 | 欧美一区二区三区免费看 | 成年人在线免费视频观看 | 国产精品欧美久久久久天天影视 | 日韩黄色大片在线观看 | 国产视频久久久 | 99久久精品国产一区 | 91色网址 | 亚洲国产欧美一区二区三区丁香婷 | 亚洲第二色 | 国产字幕在线播放 | 久久久久久影视 | 成人免费视频网 | av福利免费 | 青青河边草手机免费 | 在线观看免费av网站 | 亚洲一级黄色大片 | 亚洲日本三级 | 国产精品久久久久久久久久免费看 | av成人在线电影 | 久久69av | 国产成人无码AⅤ片在线观 日韩av不卡在线 | 五月婷婷综合激情网 | 久久成人免费 | 天天曰夜夜爽 | 国产一级高清 | 国产尤物在线视频 | 丝袜制服综合网 | 91精品久久久久久综合五月天 | 一区二区三区日韩在线观看 | 天天色综合三 | 男女拍拍免费视频 | wwwwww色| 久久精品国产亚洲aⅴ | 欧美精品一区二区免费 | 国产在线精品一区二区不卡了 | 久草网站 | 婷婷久久久 | 在线免费观看黄色大片 | 久久69精品 | 国产日韩精品欧美 | 亚洲综合婷婷 | 久久在线免费视频 | 91免费在线 | 亚洲闷骚少妇在线观看网站 | 在线日韩精品视频 | 国产亚洲在线观看 | 精品99在线视频 | 久久久久久综合网天天 | 欧美日韩xx | 免费观看v片在线观看 | 成人永久视频 | 精品婷婷 | 欧美在线一二 | 久久99日韩 | 国内精品久久天天躁人人爽 | 中文字幕视频一区 | 特黄一级毛片 | 欧美激情精品久久久久 | 久久久久久久久久久久99 | 成人av高清在线观看 | 在线视频日韩 | 美女网站久久 | 福利网在线 | 久操视频在线播放 | 日韩中文字幕亚洲一区二区va在线 | 欧美精品小视频 | 欧美一级片 | av不卡免费看 | 97视频在线观看成人 | 一区二区三区在线免费观看 | 天天艹| 91大神免费视频 | 成人a视频片观看免费 | 色播五月婷婷 | 成人av电影免费 | 精品国产一区在线观看 | 在线免费观看视频你懂的 | 亚洲综合色视频 | 亚洲精品一区二区三区四区高清 | 亚洲精品在线免费观看视频 | 91c网站色版视频 | 国产精品你懂的在线观看 | 天天操天天干天天操天天干 | 99这里只有| 婷婷九九 | 国产高清福利在线 | 色综合久久综合网 | av黄色一级片 | 人人草在线观看 | 麻豆网站免费观看 | 天天操天天弄 | 午夜久久网站 | 久久综合久久综合久久 | 久久久久久久久久影视 | 国产伦精品一区二区三区在线 | 国产黄视频在线观看 | 国产一区免费在线 | 日韩一区二区免费在线观看 | 9999精品视频 | 亚洲国产欧洲综合997久久, | www.五月天婷婷.com | 国产亚洲精品女人久久久久久 | 黄色软件网站在线观看 | 国产成人av综合色 | 国产成人精品一区二区三区福利 | 五月天天天操 | 高清av免费看| 久久久久久国产精品999 | 亚欧日韩成人h片 | 亚洲精品在线视频播放 | 国产一区av在线 | 天天摸夜夜添 | 国产永久免费高清在线观看视频 | 成人丁香花 | 免费看黄的视频 | 丁香五月缴情综合网 | 亚洲欧洲精品一区二区精品久久久 | 欧美色888 | 日韩簧片在线观看 | 一区二区三区中文字幕在线观看 | 欧美午夜剧场 | 国产一区二区在线视频观看 | 天堂视频一区 | 九九九九精品九九九九 | 成年人黄色大片在线 | 亚洲一区在线看 | 99久久精品免费视频 | 91亚洲精品国偷拍自产在线观看 | 久久久久久久久久久久国产精品 | 成人免费观看网站 | 午夜精品视频一区二区三区在线看 | 黄色三级免费片 | 日韩欧美高清一区二区三区 | www五月天婷婷 | 中文字幕一区二区三区四区视频 | 免费高清看电视网站 | 久久伊人精品天天 | 久久久九色精品国产一区二区三区 | 黄色三级免费片 | 天天干,天天干 | 国产精品正在播放 | 蜜臀av性久久久久蜜臀aⅴ流畅 | 黄色一级网 | 青草视频网 | 欧美成人高清 | 日韩av电影免费在线观看 | 久久中文字幕在线视频 | 97人人网 | 91久草视频| 欧美日韩国产亚洲乱码字幕 | 永久免费的av电影 | 久草在线最新 | 人人爽人人爽人人爽学生一级 | 亚洲天堂精品 | 色噜噜日韩精品一区二区三区视频 | 最新精品国产 | 91视频最新网址 | 日本免费久久高清视频 | 国产成人精品av在线观 | 超碰公开97| 色综合久久精品 | 色婷丁香| 中文av影院 | 国产精品久久99综合免费观看尤物 | 日韩大片在线免费观看 | 成人av高清在线 | 激情综合网色播五月 | 香蕉久久久久久av成人 | 伊人成人久久 | 色妞色视频一区二区三区四区 | 国产成人久久av | 国产手机在线观看 | 黄色小说网站在线 | 成人午夜精品福利免费 | 国产1区在线 | 午夜国产成人 | 欧美色综合天天久久综合精品 | 亚洲国产精品激情在线观看 | 精品视频在线免费 | 午夜精品视频一区 | 999国内精品永久免费视频 | 去看片| 精品久久久久一区二区国产 | 在线播放 日韩专区 | 69av久久 | 欧美日韩精品影院 | 91九色精品女同系列 | 久久综合导航 | 在线观看午夜av | 欧美日韩国产精品爽爽 | 色视频成人在线观看免 | 亚洲精品99久久久久中文字幕 | 国产区在线看 | 国产精品一区二区久久精品爱微奶 | 中文字幕 影院 | 欧美日韩视频观看 | 天天操天天摸天天射 | 国产精品永久免费观看 | 人人澡超碰碰 | 夜色在线资源 | 欧美日韩精品免费观看 | 国产午夜精品av一区二区 | 天天色天| av色图天堂网 | 一 级 黄 色 片免费看的 | 精品福利在线视频 | 五月天激情综合 | www.日本色 | 高清免费在线视频 | 五月激情姐姐 | 久久午夜国产 | 日韩美女高潮 | 粉嫩av一区二区三区入口 | 欧美日产在线观看 | 国产在线观看你懂的 | 亚洲精品在线免费播放 | 五月丁色| 欧美伦理电影一区二区 | 99亚洲国产 | 久久国产午夜精品理论片最新版本 | 亚洲综合精品视频 | 99久久这里有精品 | 精品国产伦一区二区三区 | 色婷五月天| 午夜精品视频在线 | 国产精品高清在线 | 99热只有精品在线观看 | 在线观看黄色av | 久久黄色影视 | 久久欧美在线电影 | 91精品欧美一区二区三区 | 久久这里有 | 日韩最新av在线 | 国产高清在线免费 | 色综合久久综合中文综合网 | 欧美一级性 | 在线色视频小说 | 黄色资源在线观看 | 尤物97国产精品久久精品国产 | 日韩精品中文字幕在线观看 | 午夜国产福利在线 | 国产女人免费看a级丨片 | 精品在线不卡 | 97视频久久久 | 国产专区日韩专区 | 欧美日韩视频在线观看一区二区 | 亚洲精品在线免费 | 激情网站五月天 | 色综合久久中文综合久久牛 | 久久精品牌麻豆国产大山 | 人人看看人人 | 国产精品96久久久久久吹潮 | 久久一精品 | 中文国产字幕在线观看 | av在线永久免费观看 | 黄色一级动作片 | 久久精品高清视频 | 色综合国产 | 手机成人av在线 | 精品视频在线播放 | 国产在线97 | 91精品国产乱码久久 | 一区二区三区免费 | www.干| 最近中文字幕大全 | 一级黄色片毛片 | 成人国产精品一区 | 三上悠亚一区二区在线观看 | 综合色站导航 | 国产精品av久久久久久无 | 亚洲精品自拍视频在线观看 | 日韩精品中文字幕在线不卡尤物 | 国产视频中文字幕在线观看 | 日韩一二区在线观看 | 一区 二区电影免费在线观看 | 手机在线观看国产精品 | 久久久久区 | 狠狠色丁香久久婷婷综合五月 | 精品久久久久久久久久久久久 | 欧美一区二区精品在线 | 亚洲久草在线视频 | 国产亚洲人 | 久久久精品久久 | 天天射天天干天天爽 | 婷婷综合 | 深爱五月网 | 国产精品美女免费看 | 狠狠综合| 精品xxx | 亚洲天天在线日亚洲洲精 | 久久超碰99 | 免费av观看网站 | 亚洲精品成人网 | 国产 日韩 欧美 在线 | 久久国产精品99久久人人澡 | 国产一级在线 | 久久99亚洲热视 | 九九久久久久久久久激情 | a视频在线播放 | 亚洲韩国一区二区三区 | 免费高清在线观看电视网站 | 91麻豆精品国产91久久久更新时间 | 亚洲激情校园春色 | 午夜在线看 | 成人高清在线 | 99在线热播精品免费 | 一级欧美日韩 | 久精品一区 | 欧美贵妇性狂欢 | 福利视频导航网址 | 99这里只有久久精品视频 | 中文字幕韩在线第一页 | 久久视了| 黄色a三级 | 免费观看91视频大全 | 日韩精品一区二 | 500部大龄熟乱视频 欧美日本三级 | 天天爽天天爽 | 亚洲精区二区三区四区麻豆 | 精品免费一区二区三区 | 日本亚洲国产 | 亚洲午夜电影网 | 日本最大色倩网站www | 久久在线精品 | 天天色天天射天天干 | 亚洲精品小视频 | 亚洲一区二区精品 | 欧美日韩中文在线 | www.超碰97.com| aaa日本高清在线播放免费观看 | 日韩免费精品 | 色网站免费在线观看 | 日韩精品欧美专区 | 啪啪肉肉污av国网站 | 免费在线h| 精品久久99 | 成人午夜剧场在线观看 | 色五月色开心色婷婷色丁香 | 精品久久一区二区 | 免费看的国产视频网站 | 在线超碰av | 亚洲一区 av | 亚洲美女在线一区 | 久久激情视频免费观看 | 久久最新网址 | 国产综合激情 | 99精品一区 | 久久日本视频 | 天天想夜夜操 | 午夜久久电影网 | 国产精品第72页 | 久久免费资源 | 免费在线观看视频a | 精品久久久久久国产91 | 久久激情婷婷 | 欧美精品成人在线 | 伊人色综合久久天天 | 国产三级午夜理伦三级 | 91麻豆精品国产91久久久久久久久 | 中文字幕一二 | 伊人天堂网 | 免费看污黄网站 | av视屏在线 | 欧美成人999 | 99久久激情 | 欧美日韩免费观看一区二区三区 | 日韩中文幕 | 人人精久 | 亚洲综合日韩在线 | 欧美大片在线观看一区 | 国产精品久久久久久久久免费看 | 久久这里有精品 | www.玖玖玖 | 婷婷深爱五月 | 韩国三级一区 | 欧美国产高清 | 在线视频黄 | 一区二区三区日韩视频在线观看 | 51精品国自产在线 | 久艹视频在线免费观看 | 久久热首页 | 亚洲资源在线 | 国产精品视频久久久 | 黄色三级在线 | 91在线区 | 99久久久久久久 | 一区二区三区日韩视频在线观看 | 97人人添人澡人人爽超碰动图 | 黄色三级免费看 | 久久99国产精品免费 | 婷婷久久一区二区三区 | 在线精品视频免费观看 | 天天爽夜夜爽人人爽一区二区 | 中文字幕在线观看视频一区 | 精品国产成人av在线免 | 午夜的福利 | 免费在线91 | 久久久国产99久久国产一 | 青青久视频 | 国产在线播放一区 | 特黄一级毛片 | 色狠狠婷婷 | 国产在线观看午夜 | 久久久久国产一区二区三区四区 | 99精品国产99久久久久久福利 | 四虎在线观看视频 | 狠狠搞,com| av免费看在线| 麻豆激情电影 | 伊人久久在线观看 | 天天干天天操天天拍 | 高清免费在线视频 | 国产精品久久久久久久久久久久午夜 | 日本在线观看一区二区三区 | 九9热这里真品2 | 国产精品一区二区三区视频免费 | 高清色免费 | 国产精彩在线视频 | 天天色天天艹 | 日韩精品高清不卡 | 天天撸夜夜操 | 少妇bbbb| 一本—道久久a久久精品蜜桃 | 97免费视频在线 | 中文不卡视频 | 伊人久久av | av在线一二三区 | 人人艹人人 | 午夜精品视频免费在线观看 | 伊人婷婷色| 久久香蕉电影 | 日韩av片免费在线观看 | 国产精品欧美日韩在线观看 | 亚洲伦理电影在线 | 久99精品| 91成品视频| www.夜色.com | 欧美va电影| 欧亚日韩精品一区二区在线 | 精品国产成人 | 欧美日韩国产二区三区 | 四虎在线免费观看 | 9在线观看免费 |