mybatis以map为参,动态生成表并插入数据

本文介绍使用MyBatis通过传递Map参数的方式,在PostgreSQL中动态创建表并插入数据的方法。该方法允许根据传入的数据结构自动生成表结构。
mybatis以map为参,动态生成表并插入数据,目前仅在PostgreSQL上做了测试
标签:  MyBatis  iBATIS

代码片段(2)[全屏查看所有代码]

1. [代码]mybatis javabean 映射文件(部分)     跳至 [1] [2] [全屏预览]

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
< insert id = "creatTableAndInsert" parameterType = "java.util.Map" statementType = "STATEMENT" >
         create table if not exists ${tablename}
         < foreach collection = "keys" item = "k" index = "index" open = "(" separator = "," close = ");" >
             ${k} varchar(30) not null
         </ foreach >
         insert into ${tablename}
         < foreach collection = "keys" item = "k" index = "index" open = "(" separator = "," close = ")" >
             ${k}
         </ foreach >
         values
         < foreach collection = "keys" item = "k" index = "index" open = "(" separator = "," close = ")" >
             '${params[k]}'
         </ foreach >
     </ insert >

2. [代码]mybatis中传入的map结构     

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
Map<String, Object> params = new HashMap<>();
             params.put( "a" , "safd" );
             params.put( "b" , "dsjaf" );
             params.put( "c" , "123456" );
             params.put( "d" , "风景的撒娇房顶上经费等快速拉进房间空间上" );
             
             String[] keys = new String[params.size()];
             Set<String> sset = params.keySet();
             int i = 0 ;
             for (String os : sset) {
                 keys[i++] = os;
             }
 
             Map map = new HashMap<>();
             map.put( "tablename" , "pay_o_tb" );
             map.put( "keys" , keys);
             map.put( "params" , params);
 
             mapper.creatTableAndInsert(map);

?
<think>我们正在讨论如何在MyBatis XML中实现批量插入List<Map>类型的数据根据用户提供的引用,特别是引用[2]和引用[3],我们可以总结出两种主要方法:一种是全动态的名和列名(引用[2]),另一种是固定列名但动态值的批量插入(引用[3])。下面将分别介绍这两种方法,给出示例。 ### 方法一:全动态批量插入名、列名、值均动态) 这种方法适用于名和列名都是动态传入的情况。例如,在运行时确定插入到哪张,以及插入哪些列。 #### XML配置示例(引用[2]): ```xml <insert id="insertTableListDynamic" parameterType="java.util.HashMap"> INSERT INTO ${map.tableName} ( <foreach collection="map.mapHead" index="key" item="value" separator=","> `${key}` </foreach> ) VALUES <foreach collection="map.list" item="line" separator=","> ( <foreach collection="line" index="key" item="value" separator=","> #{value} </foreach> ) </foreach> </insert> ``` #### Java调用代码(引用[2]): ```java private void insertBatchList(String tableName, Map<String, Object> mapHead, List<Map<String, Object>> list) { Map<String, Object> batchList = new HashMap<>(); batchList.put("tableName", tableName); batchList.put("mapHead", mapHead); // mapHead的键是列名,值可以是类型信息或其他,但这里只用到键(列名) batchList.put("list", list); // list中的每个Map对应一行数据,键为列名,值为要插入数据 insertTableListDynamic(batchList); // 调用MyBatis插入方法 } ``` **说明**: 1. `${map.tableName}`: 直接使用字符串替换名(注意:需防止SQL注入,确保名安全)。 2. `<foreach>`遍历`map.mapHead`(列名集合):生成列名部分,用反引号包裹列名(防止列名是关键字)。 3. 外层`<foreach>`遍历`map.list`(数据行列),内层`<foreach>`遍历每行数据键值对,生成值列。 4. 注意:这里假设`mapHead`的键集合与`list`中每个Map的键集合一致(顺序和数量),否则会出现列与值不匹配。 ### 方法二:固定列名(或部分固定)的批量插入 如果列名是固定的,或者部分固定(如主键固定,其他列动态),则可以采用更简单的方式。 #### XML配置示例(引用[3]): ```xml <insert id="insert"> INSERT INTO table_name (uid, data0, data1, data2) VALUES <foreach collection="data" separator="," item="map"> (#{uid}, #{map.data0}, #{map.data1}, #{map.data2}) </foreach> </insert> ``` #### Java接口: ```java void insert(@Param("uid") String uid, @Param("data") List<Map<String, String>> data); ``` **说明**: 1. 列名在SQL中固定(如`uid, data0, data1, data2`)。 2. `foreach`遍历`data`(List类型),每个`map`代一行数据,通过`#{map.data0}`等方式取出对应值。 3. 注意:这里假设每个Map中都有`data0`、`data1`、`data2`这些键。 ### 注意事项 1. **SQL注入风险**:使用`${}`直接拼接名或列名存在风险,应确保这些值来自可信来源(如内部配置,非用户输入)。 2. **批量插入限制**:MySQL等数据库对单条SQL的长度有限制,如果List过大,可能需要分批插入。 3. **性能**:批量插入比单条插入效率高,但也要注意数据库的承受能力。 4. **数据类型**:Map中的值类型需与数据库列类型匹配,否则可能报错。MyBatis会尝试自动转换,但复杂类型(如日期)可能需要特殊处理。 ### 完整示例(固定列名) 假设有一个用户`t_user`,需要批量插入用户名(`username`)、年龄(`age`)、邮箱(`email`)。 #### Java代码: ```java // 构造数据 List<Map<String, Object>> userList = new ArrayList<>(); Map<String, Object> user1 = new HashMap<>(); user1.put("username", "张三"); user1.put("age", 25); user1.put("email", "zhangsan@example.com"); userList.add(user1); Map<String, Object> user2 = new HashMap<>(); user2.put("username", "李四"); user2.put("age", 30); user2.put("email", "lisi@example.com"); userList.add(user2); // 调用MyBatis userMapper.batchInsertUsers(userList); ``` #### Mapper接口: ```java void batchInsertUsers(@Param("list") List<Map<String, Object>> userList); ``` #### XML配置: ```xml <insert id="batchInsertUsers"> INSERT INTO t_user (username, age, email) VALUES <foreach collection="list" item="userMap" separator=","> (#{userMap.username}, #{userMap.age}, #{userMap.email}) </foreach> </insert> ``` ### 总结 - 使用`List<Map>`作为入,结合MyBatis的`foreach`标签,可以灵活实现批量插入。 - 动态名/列名场景需谨慎处理SQL注入问题。 - 固定列名的场景更简单直接,推荐使用。 §§相关问题§§ 1. MyBatis批量插入List<Map>时如何防止SQL注入? 2. MyBatis批量插入的最大数据量限制是多少? 3. MyBatis中如何实现动态列名的查询操作?</think>### MyBatis XML批量插入List<Map>数据的方法 在MyBatis中,通过XML配置实现`List<Map>`的批量插入主要有两种方式:**固定列名插入**和**全动态列名插入**。以下是具体实现方法和示例: --- #### 方法一:固定列名插入(推荐) 适用于列名固定的场景,通过`<foreach>`标签遍历List中的每个Map。 **XML配置示例**: ```xml <insert id="batchInsertUsers"> INSERT INTO user_table (uid, name, email, age) VALUES <foreach collection="list" item="map" separator=","> (#{map.uid}, #{map.name}, #{map.email}, #{map.age}) </foreach> </insert> ``` **Java调用代码**: ```java // 1. 准备数据 List<Map<String, Object>> dataList = new ArrayList<>(); Map<String, Object> map1 = new HashMap<>(); map1.put("uid", "1001"); map1.put("name", "张三"); map1.put("email", "zhangsan@example.com"); map1.put("age", 25); Map<String, Object> map2 = new HashMap<>(); map2.put("uid", "1002"); map2.put("name", "李四"); map2.put("email", "lisi@example.com"); map2.put("age", 30); dataList.add(map1); dataList.add(map2); // 2. 调用Mapper方法 userMapper.batchInsertUsers(dataList); ``` **关键点说明**: 1. `collection="list"`:固定写法,示处理List类型数 2. `item="map"`:List中的每个元素(Map对象)的别名 3. `separator=","`:每行数据用逗号分隔 4. 列名需明确写在SQL中(如`uid, name, email, age`) --- #### 方法二:全动态列名插入 适用于名和列名都动态传入的场景(需注意SQL注入风险)。 **XML配置示例**: ```xml <insert id="dynamicBatchInsert"> INSERT INTO ${tableName} ( <foreach collection="columns" item="col" separator=","> ${col} </foreach> ) VALUES <foreach collection="dataList" item="row" separator=","> ( <foreach collection="row.values" item="val" separator=","> #{val} </foreach> ) </foreach> </insert> ``` **Java调用代码**: ```java Map<String, Object> params = new HashMap<>(); params.put("tableName", "user_table"); // 动态名 params.put("columns", Arrays.asList("uid", "name", "email", "age")); // 动态列名 params.put("dataList", dataList); // 数据 userMapper.dynamicBatchInsert(params); ``` **关键点说明**: 1. `${tableName}`:动态名(使用`$`有注入风险,确保数安全) 2. 双层`<foreach>`嵌套:外层遍历行,内层遍历列值 3. 要求所有Map的键顺序必须与`columns`列一致 --- #### 注意事项 1. **SQL长度限制**:批量插入时单条SQL长度不超过数据库限制(MySQL默认4MB) 2. **性能优化**: ```xml <!-- 添加rewriteBatchedStatements数提升性能 --> <property name="url" value="jdbc:mysql://localhost:3306/db?rewriteBatchedStatements=true"/> ``` 3. **事务管理**:在Service层添加`@Transactional`注解保证原子性 4. **防注入措施**:动态名/列名需白名单校验,避免使用用户输入直接拼接 --- ### 两种方法对比 | **特性** | 固定列名法 | 全动态列名法 | |------------------|------------------------------|-----------------------------| | 安全性 | ⭐⭐⭐⭐⭐ (无注入风险) | ⭐⭐ (需防范`$`注入风险) | | 灵活性 | ⭐⭐ (列名固定) | ⭐⭐⭐⭐⭐ (/列全动态) | | 性能 | ⭐⭐⭐⭐ (优化简单) | ⭐⭐⭐ (需额外解析) | | 代码可读性 | ⭐⭐⭐⭐ (直观) | ⭐⭐ (较复杂) | | 适用场景 | 固定结构 | 动态结构(如多租户系统) | > 推荐优先使用固定列名法,仅在必要场景使用动态列名法[^2][^3]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值