Mybatis Map保存到數據庫,Mybatis Map動態同步表,Mybatis Map Foreach插入數據庫java
Mybatis 保存Map<String, Object>正則表達式
================================spring
©Copyright 蕃薯耀 2021-01-29數據庫
https://www.cnblogs.com/fanshuyao/apache
1、情景描述數組
後臺接口方式同步表數據。mybatis
爲了讓表數據同步自動化,避免後面重複開發,採起經過接收List<Map<String, Object>>對象的方式,將數據保存到相應的表中去。app
後臺程序代碼自動將Map<String, Object>轉換成insert語句,將數據保存到數據庫中。其中Map<String, Object>的鍵爲字段名,值爲字段的內容。ide
2、相關代碼邏輯和配置工具
一、mybatis mapper.xml配置
${tableName}爲$符號
#{item}爲#符號,使用#符號,不要使用statementType="STATEMENT"聲明
<?xml version="1.0" encoding="UTF-8"?> <!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd"> <mapper namespace="com.xxx.dao.SyncDataDao"> <!-- 使用#{}點位符時, 不要使用statementType="STATEMENT"聲明 --> <insert id="saveData" parameterType="map"> insert into ${tableName} <foreach collection="tableColumns" index="index" item="item" open="(" close=")" separator=","> ${index} </foreach> values <foreach collection="tableColumns" index="index" item="item" open="(" close=")" separator=","> #{item} </foreach> </insert> <insert id="deleteAllData" parameterType="string"> delete from ${tableName} </insert> </mapper>
foreach 元素的功能很是強大,它容許你指定一個集合,聲明能夠在元素體內使用的集合項(item)和索引(index)變量。
它也容許你指定開頭與結尾的字符串以及集合項迭代之間的分隔符。
提示:你能夠將任何可迭代對象(如 List、Set 等)、Map 對象或者數組對象做爲集合參數傳遞給 foreach。
當使用可迭代對象或者數組時,index 是當前迭代的序號,item 的值是本次迭代獲取到的元素。
當使用 Map 對象(或者 Map.Entry 對象的集合)時,index 是鍵,item 是值。
二、Dao類
import java.util.Map; import com.baomidou.mybatisplus.core.mapper.BaseMapper; /** * <p> * 數據同步的配置表 Mapper 接口 * </p> * * @since 2021-01-27 */ public interface SyncDataDao extends BaseMapper<Object> { public void saveData(Map<String, Object> dataMap); public void deleteAllData(String tableName); }
注:此處saveData方法的參數Map<String, Object> dataMap沒有用@Param註解。若是使用了@Param("dataMap"),則Mapper配置文件也要增長
<insert id="saveData" parameterType="map"> insert into ${dataMap.tableName} <foreach collection="dataMap.tableColumns" index="index" item="item" open="(" close=")" separator=","> ${index} </foreach> values <foreach collection="dataMap.tableColumns" index="index" item="item" open="(" close=")" separator=","> #{item} </foreach> </insert>
當不加@Param("dataMap")時,其實默認是:_parameter,但能夠忽略。
三、Service類
import java.util.List; import java.util.Map; import org.apache.commons.lang3.StringUtils; import org.springframework.stereotype.Service; import org.springframework.transaction.annotation.Transactional; import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl; import com.szpl.cu.biz.st.dao.SyncDataDao; import com.szpl.cu.biz.st.service.SyncDataService; import com.szpl.cu.security.exception.RunException; /** * <p> * 數據同步的配置表 服務實現類 * </p> * * @since 2021-01-27 */ @Service public class SyncDataServiceImpl extends ServiceImpl<SyncDataDao, Object> implements SyncDataService { @Transactional @Override public void saveData(Map<String, Object> dataMap) { this.baseMapper.saveData(dataMap); } @Transactional @Override public void deleteAllData(String tableName) { if(StringUtils.isBlank(tableName)) { RunException.run("參數錯誤"); } this.baseMapper.deleteAllData(tableName); } @Transactional @Override public void saveAllDataByDeleteTable(String tableName, List<Map<String, Object>> datas) { //先刪除,再新增 this.deleteAllData(tableName); for (Map<String, Object> dataMap : datas) { this.saveData(dataMap); } } }
四、調用方法(相似Controller)
//遠程接口獲取的數據 @SuppressWarnings("unchecked") List<Map<String, Object>> datas = (List<Map<String, Object>>) result.getDatas(); //要保存的數據,經過datas轉換 List<Map<String, Object>> newDatas = new ArrayList<Map<String,Object>>(datas.size()); //datas轉換 for (Map<String, Object> data : datas) { LinkedHashMap<String, Object> dataMap = new LinkedHashMap<String, Object>(); //處理時間類型的字段,由於返回的時間字段被轉成字符串,不轉換Mybatis識別不了 Map<String, Object> newData = MapDataFormatUtil.format(data); //設置保存的表名 dataMap.put("tableName", syncTableConfig.getTableName()); //設置保存的字段 dataMap.put("tableColumns", newData); newDatas.add(dataMap); } //保存同步數據 syncDataService.saveAllDataByDeleteTable(syncTableConfig.getTableName(), newDatas);
tableColumns、tableColumns對應Mapper文件的屬性。
MapDataFormatUtil工具類:
import java.util.Map; import org.apache.commons.lang3.StringUtils; import com.szpl.cu.util.DateUtils; import com.szpl.cu.util.RegUtils; public class MapDataFormatUtil { public static Map<String, Object> format(Map<String, Object> dataMap) throws Exception{ if(dataMap == null || dataMap.size() < 1) { return dataMap; } for (Map.Entry<String, Object> entry : dataMap.entrySet()) { Object value = entry.getValue(); if(value instanceof String) { String tempValue = (String) entry.getValue(); if(!StringUtils.isBlank(tempValue) && RegUtils.isDateTime(tempValue)) { dataMap.put(entry.getKey(), DateUtils.parseDateTime(tempValue)); } } } return dataMap; } }
RegUtils工具類:
import java.util.regex.Matcher; import java.util.regex.Pattern; import org.apache.commons.lang.StringUtils; /** * 正則表達式工具類 * */ public class RegUtils { /** * 郵箱 */ public static final String EMAIL = "^\\w+((-\\w+)|(\\.\\w+))*\\@[A-Za-z0-9]+((\\.|-)[A-Za-z0-9]+)*\\.[A-Za-z0-9]+$"; /** * 手機號碼 */ public static final String PHONE = "^(1[3-9]([0-9]{9}))$"; /** * 僅中文 */ public static final String CHINESE = "^[\\u4E00-\\u9FA5\\uF900-\\uFA2D]+$"; /** * 整數 */ public static final String INTEGER = "^-?[1-9]\\d*$"; /** * 數字 */ public static final String NUMBER = "^([+-]?)\\d*\\.?\\d+$"; /** * 正整數 */ public static final String INTEGER_POS = "^[1-9]\\d*$"; /** * 浮點數 */ public static final String FLOAT = "^([+-]?)\\d*\\.\\d+$"; /** * 正浮點數 */ public static final String FLOAT_POS = "^[1-9]\\d*.\\d*|0.\\d*[1-9]\\d*$"; /** * 是否爲正整數數字,包括0(00,01非數字) */ public static final String INTEGER_WITH_ZERO_POS = "^(([0-9])|([1-9]([0-9]+)))$"; /** * 是否爲整數數字,包括正、負整數,包括0(00,01非數字) */ public static final String NUMBER_WITH_ZERO = "^((-)?(([0-9])|([1-9]([0-9]+))))$"; /** * 是否爲數字字符串 */ public static final String NUMBER_TEXT = "^([0-9]+)$"; /** * 數字(整數、0、浮點數),能夠判斷是否金額,也能夠是負數 */ public static final String NUMBER_ALL = "^((-)?(([0-9])|([1-9][0-9]+))(\\.([0-9]+))?)$"; /** * QQ,5-14位 */ public static final String QQ = "^[1-9][0-9]{4,13}$"; /** * IP地址 */ public static final String IP = "((?:(?:25[0-5]|2[0-4]\\d|[01]?\\d?\\d)\\.){3}(?:25[0-5]|2[0-4]\\d|[01]?\\d?\\d))"; /** * 郵編 */ public static final String POST_CODE = "[1-9]\\d{5}(?!\\d)"; /** * 普通日期 */ public static final String DATE = "^[1-9]\\d{3}-((0[1-9])|(1[0-2]))-((0[1-9])|([1-2][0-9])|(3[0-1]))$"; /** * 複雜日期,不區分閏年的2月 * 日期格式:2017-10-19 * 或2017/10/19 * 或2017.10.19 * 或2017年10月19日 * 最大31天的月份:(((01|03|05|07|08|10|12))-((0[1-9])|([1-2][0-9])|(3[0-1]))) * 最大30天的月份:(((04|06|11))-((0[1-9])|([1-2][0-9])|(30))) * 最大29天的月份:(02-((0[1-9])|([1-2][0-9]))) */ public static final String DATE_COMPLEX = "^(([1-2]\\d{3})(-|/|.|年)((((01|03|05|07|08|10|12))(-|/|.|月)((0[1-9])|([1-2][0-9])|(3[0-1])))|(((04|06|11))(-|/|.|月)((0[1-9])|([1-2][0-9])|(30)))|(02-((0[1-9])|([1-2][0-9]))))(日)?)$"; /** * 複雜的日期,區分閏年的2月 * 這個日期校驗能區分閏年的2月,格式以下:2017-10-19 * (見:http://www.jb51.net/article/50905.htm) * ^((?!0000)[0-9]{4}-((0[1-9]|1[0-2])-(0[1-9]|1[0-9]|2[0-8])|(0[13-9]|1[0-2])-(29|30)|(0[13578]|1[02])-31)|([0-9]{2}(0[48]|[2468][048]|[13579][26])|(0[48]|[2468][048]|[13579][26])00)-02-29)$ */ public static final String DATE_COMPLEX_LEAP_YEAR = "^(?:(?!0000)[0-9]{4}-(?:(?:0[1-9]|1[0-2])-(?:0[1-9]|1[0-9]|2[0-8])|(?:0[13-9]|1[0-2])-(?:29|30)|(?:0[13578]|1[02])-31)|(?:[0-9]{2}(?:0[48]|[2468][048]|[13579][26])|(?:0[48]|[2468][048]|[13579][26])00)-02-29)$"; /** * 普通日期,帶時間 */ public static final String DATE_TIME = "^[1-9]\\d{3}-((0[1-9])|(1[0-2]))-((0[1-9])|([1-2][0-9])|(3[0-1])) \\d{2}:\\d{2}:\\d{2}$"; /** * 正則表達式校驗,符合返回True * @param regex 正則表達式 * @param content 校驗的內容 * @return */ public static boolean isMatch(String regex, CharSequence content){ return Pattern.matches(regex, content); } /** * 校驗手機號碼 * @param mobile * @return */ public static final boolean isMoblie(String mobile){ boolean flag = false; if (null != mobile && !mobile.trim().equals("") && mobile.trim().length() == 11) { Pattern pattern = Pattern.compile(PHONE); Matcher matcher = pattern.matcher(mobile.trim()); flag = matcher.matches(); } return flag; } /** * 校驗郵箱 * @param value * @return */ public static final boolean isEmail(String value){ boolean flag = false; if (null != value && !value.trim().equals("")) { Pattern pattern = Pattern.compile(EMAIL); Matcher matcher = pattern.matcher(value.trim()); flag = matcher.matches(); } return flag; } /** * 校驗密碼 * @param password * @return 長度符合返回true,不然爲false */ public static final boolean isPassword(String password){ boolean flag = false; if (null != password && !password.trim().equals("")) { password = password.trim(); if(password.length() >= 6 && password.length() <= 30){ return true; } } return flag; } /** * 校驗手機驗證碼 * @param value * @return 符合正則表達式返回true,不然返回false */ public static final boolean isPhoneValidateCode(String value){ boolean flag = false; if (null != value && !value.trim().equals("")) { Pattern pattern = Pattern.compile("^8\\d{5}$"); Matcher matcher = pattern.matcher(value.trim()); flag = matcher.matches(); } return flag; } /** * 判斷是否所有大寫字母 * @param str * @return */ public static boolean isUpperCase(String str){ if(StringUtils.isEmpty(str)){ return false; } String reg = "^[A-Z]$"; return isMatch(reg,str); } /** * 判斷是否所有小寫字母 * @param str * @return */ public static boolean isLowercase(String str){ if(StringUtils.isEmpty(str)){ return false; } String reg = "^[a-z]$"; return isMatch(reg,str); } /** * 是否ip地址 * @param str * @return */ public static boolean isIP(String str){ if(StringUtils.isEmpty(str)){ return false; } return isMatch(IP, str); } /** * 符合返回true,區分30、31天和閏年的2月份(最嚴格的校驗),格式爲2017-10-19 * @param str * @return */ public static boolean isDate(String str){ if(StringUtils.isEmpty(str)){ return false; } return isMatch(DATE_COMPLEX_LEAP_YEAR, str); } /** * 簡單日期校驗,不那麼嚴格 * @param str * @return */ public static boolean isDateSimple(String str){ if(StringUtils.isEmpty(str)){ return false; } return isMatch(DATE, str); } /** * 區分30、31天,但沒有區分閏年的2月份 * @param str * @return */ public static boolean isDateComplex(String str){ if(StringUtils.isEmpty(str)){ return false; } return isMatch(DATE_COMPLEX, str); } /** * 簡單日期-帶時間 * @param str * @return */ public static boolean isDateTime(String str){ if(StringUtils.isEmpty(str)){ return false; } return isMatch(DATE_TIME, str); } /** * 判斷是否爲數字字符串,如0011,10101,01 * @param str * @return */ public static boolean isNumberText(String str){ if(StringUtils.isEmpty(str)){ return false; } return isMatch(NUMBER_TEXT, str); } /** * 判斷全部類型的數字,數字(整數、0、浮點數),能夠判斷是否金額,也能夠是負數 * @param str * @return */ public static boolean isNumberAll(String str){ if(StringUtils.isEmpty(str)){ return false; } return isMatch(NUMBER_ALL, str); } /** * 是否爲正整數數字,包括0(00,01非數字) * @param str * @return */ public static boolean isIntegerWithZeroPos(String str){ if(StringUtils.isEmpty(str)){ return false; } return isMatch(INTEGER_WITH_ZERO_POS, str); } /** * 是否爲整數,包括正、負整數,包括0(00,01非數字) * @param str * @return */ public static boolean isIntegerWithZero(String str){ if(StringUtils.isEmpty(str)){ return false; } return isMatch(NUMBER_WITH_ZERO, str); } /** * 符合返回true,QQ,5-14位 * @param str * @return */ public static boolean isQQ(String str){ if(StringUtils.isEmpty(str)){ return false; } return isMatch(QQ, str); } public static void main(String[] args) { System.out.println(isMoblie("13430800244")); System.out.println(isMoblie("17730800244")); System.out.println(isMoblie("17630800244")); System.out.println(isMoblie("14730800244")); System.out.println(isMoblie("18330800244")); System.out.println(isMoblie("19330800244")); System.out.println(isMoblie("1333000244")); } }
================================
©Copyright 蕃薯耀 2021-01-29
https://www.cnblogs.com/fanshuyao/