批量插入是一种高效的数据库操作方式,可以显著提高数据插入的性能。在MyBatis中,有多种方法可以实现批量插入,下面是其中的几种:
使用foreach标签可以将多条SQL语句合并成一条,从而实现批量插入。具体步骤如下: - 在Mapper.xml文件中编写SQL语句,并使用foreach标签包裹插入语句。 - 在Java代码中调用Mapper接口中的批量插入方法,传入一个List集合作为参数,该集合中存放待插入的数据。 以下是示例代码: Mapper.xml文件:
<insert id="batchInsert" parameterType="com.cy.dao.model.TestUser">
insert into test_user (id, login_name, user_name
)
values
<foreach collection="list" item="item" separator=",">
(#{item.id,jdbcType=INTEGER}, #{item.loginName,jdbcType=VARCHAR}, #{item.userName,jdbcType=VARCHAR})
foreach>
insert>
Java代码:
List<MyObject> list = new ArrayList<>();
// 添加待插入的数据
int result = mapper.batchInsert(list);
SqlSession提供了多种批量操作方法,例如 insert()
、 update()
和 delete()
等。具体步骤如下: - 调用SqlSession的批量操作方法,传入一个Mapper接口和方法名,以及一个Collection集合作为参数,该集合中存放待插入的数据。 以下是示例代码:
List<MyObject> list = new ArrayList<>();
// 添加待插入的数据
int result = sqlSession.insert("MyMapper.batchInsert", list);
sqlSession.commit();
需要注意的是,以上两种方法都可以实现批量插入,但其性能和效率可能会有所不同。具体使用哪种方法,需要根据具体情况进行选择。
import java.util.Collection;
public interface IBatchMapper {
<T> int batchInsert(Collection<T> list , String statement);
<T> int batchUpdate(Collection<T> list , String statement);
}
注意该种方式需要加入事务,本人撤销事务注解,执行时间超长,应该是没有进行批量执行。
import cn.hutool.json.JSONUtil;
import lombok.extern.slf4j.Slf4j;
import org.apache.ibatis.session.ExecutorType;
import org.apache.ibatis.session.SqlSession;
import org.apache.ibatis.session.SqlSessionFactory;
import org.springframework.stereotype.Component;
import org.springframework.transaction.annotation.Transactional;
import javax.annotation.Resource;
import java.util.Collection;
@Slf4j
@Component
public class BatchMapper implements IBatchMapper{
@Resource
private SqlSessionFactory sqlSessionFactory;
@Override
@Transactional(rollbackFor = Exception.class)
public <T> int batchInsert(Collection<T> list , String statement){
int result = 0;
SqlSession sqlSession = null;
try{
sqlSession = sqlSessionFactory.openSession(ExecutorType.BATCH);
for (T obj : list) {
result += sqlSession.insert(statement, obj);
}
sqlSession.commit();
} catch (Exception e) {
log.error("批插入失败:{}" , JSONUtil.toJsonStr(list) , e);
throw new RuntimeException(e);
} finally {
if (sqlSession != null) {
sqlSession.close(); // 关闭SqlSession,释放资源
}
}
return result;
}
@Override
@Transactional(rollbackFor = Exception.class)
public <T> int batchUpdate(Collection<T> list, String statement) {
int result = 0;
SqlSession sqlSession = null;
try{
sqlSession = sqlSessionFactory.openSession(ExecutorType.BATCH);
for (T obj : list) {
result += sqlSession.update(statement, obj);
}
sqlSession.commit();
} catch (Exception e) {
log.error("批更新失败:{}" , JSONUtil.toJsonStr(list) , e);
throw new RuntimeException(e);
} finally {
if (sqlSession != null) {
sqlSession.close(); // 关闭SqlSession,释放资源
}
}
return result;
}
}
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.ApplicationContext;
import javax.annotation.Resource;
import java.lang.reflect.ParameterizedType;
import java.util.ArrayList;
import java.util.List;
public class BaseEventBiz<M , T> {
@Autowired
private ApplicationContext applicationContext;
@Resource
private BatchMapper batchMapper;
protected int batchCount = 2000;
@SuppressWarnings("unchecked")
Class<M> getMapperClass() {
ParameterizedType parameterizedType = (ParameterizedType) getClass().getGenericSuperclass();
return (Class<M>) parameterizedType.getActualTypeArguments()[0];
}
public M getMapper (){
return applicationContext.getBean(getMapperClass());
}
public int batchInsert(List<T> list) {
String statement = getMapperClass().getName() + ".insert";
int result = 0;
List<T> l = new ArrayList<>();
for (int i = 0; i < list.size(); i++) {
l.add(list.get(i));
if (i % batchCount == 0) {
result += batchMapper.batchInsert(l, statement);
l.clear();
}
}
if (l.size()>0) {
result += batchMapper.batchInsert(l , statement);
}
return result;
}
public int batchUpdate(List<T> list) {
String statement = getMapperClass().getName() + ".update";
int result = 0;
List<T> l = new ArrayList<>();
for (int i = 0; i < list.size(); i++) {
l.add(list.get(i));
if (i % batchCount == 0) {
result += batchMapper.batchUpdate(l, statement);
l.clear();
}
}
if (l.size()>0) {
result += batchMapper.batchUpdate(l , statement);
}
return result;
}
}
package com.cy.dao.model;
public class TestUser {
/**
*
* This field was generated by MyBatis Generator.
* This field corresponds to the database column test_user.id
*
* @mbg.generated
*/
private Integer id;
/**
*
* This field was generated by MyBatis Generator.
* This field corresponds to the database column test_user.login_name
*
* @mbg.generated
*/
private String loginName;
/**
*
* This field was generated by MyBatis Generator.
* This field corresponds to the database column test_user.user_name
*
* @mbg.generated
*/
private String userName;
/**
* This method was generated by MyBatis Generator.
* This method returns the value of the database column test_user.id
*
* @return the value of test_user.id
*
* @mbg.generated
*/
public Integer getId() {
return id;
}
/**
* This method was generated by MyBatis Generator.
* This method sets the value of the database column test_user.id
*
* @param id the value for test_user.id
*
* @mbg.generated
*/
public void setId(Integer id) {
this.id = id;
}
/**
* This method was generated by MyBatis Generator.
* This method returns the value of the database column test_user.login_name
*
* @return the value of test_user.login_name
*
* @mbg.generated
*/
public String getLoginName() {
return loginName;
}
/**
* This method was generated by MyBatis Generator.
* This method sets the value of the database column test_user.login_name
*
* @param loginName the value for test_user.login_name
*
* @mbg.generated
*/
public void setLoginName(String loginName) {
this.loginName = loginName == null ? null : loginName.trim();
}
/**
* This method was generated by MyBatis Generator.
* This method returns the value of the database column test_user.user_name
*
* @return the value of test_user.user_name
*
* @mbg.generated
*/
public String getUserName() {
return userName;
}
/**
* This method was generated by MyBatis Generator.
* This method sets the value of the database column test_user.user_name
*
* @param userName the value for test_user.user_name
*
* @mbg.generated
*/
public void setUserName(String userName) {
this.userName = userName == null ? null : userName.trim();
}
}
import com.cy.dao.model.TestUser;
import com.cy.dao.model.TestUserExample;
import org.apache.ibatis.annotations.Param;
import java.util.List;
public interface TestUserMapper {
/**
* This method was generated by MyBatis Generator.
* This method corresponds to the database table test_user
*
* @mbg.generated
*/
long countByExample(TestUserExample example);
/**
* This method was generated by MyBatis Generator.
* This method corresponds to the database table test_user
*
* @mbg.generated
*/
int deleteByExample(TestUserExample example);
/**
* This method was generated by MyBatis Generator.
* This method corresponds to the database table test_user
*
* @mbg.generated
*/
int deleteByPrimaryKey(Integer id);
/**
* This method was generated by MyBatis Generator.
* This method corresponds to the database table test_user
*
* @mbg.generated
*/
int insert(TestUser record);
/**
* This method was generated by MyBatis Generator.
* This method corresponds to the database table test_user
*
* @mbg.generated
*/
int insertSelective(TestUser record);
/**
* This method was generated by MyBatis Generator.
* This method corresponds to the database table test_user
*
* @mbg.generated
*/
List<TestUser> selectByExample(TestUserExample example);
/**
* This method was generated by MyBatis Generator.
* This method corresponds to the database table test_user
*
* @mbg.generated
*/
TestUser selectByPrimaryKey(Integer id);
/**
* This method was generated by MyBatis Generator.
* This method corresponds to the database table test_user
*
* @mbg.generated
*/
int updateByExampleSelective(@Param("record") TestUser record, @Param("example") TestUserExample example);
/**
* This method was generated by MyBatis Generator.
* This method corresponds to the database table test_user
*
* @mbg.generated
*/
int updateByExample(@Param("record") TestUser record, @Param("example") TestUserExample example);
/**
* This method was generated by MyBatis Generator.
* This method corresponds to the database table test_user
*
* @mbg.generated
*/
int updateByPrimaryKeySelective(TestUser record);
/**
* This method was generated by MyBatis Generator.
* This method corresponds to the database table test_user
*
* @mbg.generated
*/
int updateByPrimaryKey(TestUser record);
}
DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.cy.dao.mapper.TestUserMapper">
<resultMap id="BaseResultMap" type="com.cy.dao.model.TestUser">
<id column="id" jdbcType="INTEGER" property="id" />
<result column="login_name" jdbcType="VARCHAR" property="loginName" />
<result column="user_name" jdbcType="VARCHAR" property="userName" />
resultMap>
<sql id="Example_Where_Clause">
<where>
<foreach collection="oredCriteria" item="criteria" separator="or">
<if test="criteria.valid">
<trim prefix="(" prefixOverrides="and" suffix=")">
<foreach collection="criteria.criteria" item="criterion">
<choose>
<when test="criterion.noValue">
and ${criterion.condition}
when>
<when test="criterion.singleValue">
and ${criterion.condition} #{criterion.value}
when>
<when test="criterion.betweenValue">
and ${criterion.condition} #{criterion.value} and #{criterion.secondValue}
when>
<when test="criterion.listValue">
and ${criterion.condition}
<foreach close=")" collection="criterion.value" item="listItem" open="(" separator=",">
#{listItem}
foreach>
when>
choose>
foreach>
trim>
if>
foreach>
where>
sql>
<sql id="Update_By_Example_Where_Clause">
<where>
<foreach collection="example.oredCriteria" item="criteria" separator="or">
<if test="criteria.valid">
<trim prefix="(" prefixOverrides="and" suffix=")">
<foreach collection="criteria.criteria" item="criterion">
<choose>
<when test="criterion.noValue">
and ${criterion.condition}
when>
<when test="criterion.singleValue">
and ${criterion.condition} #{criterion.value}
when>
<when test="criterion.betweenValue">
and ${criterion.condition} #{criterion.value} and #{criterion.secondValue}
when>
<when test="criterion.listValue">
and ${criterion.condition}
<foreach close=")" collection="criterion.value" item="listItem" open="(" separator=",">
#{listItem}
foreach>
when>
choose>
foreach>
trim>
if>
foreach>
where>
sql>
<sql id="Base_Column_List">
id, login_name, user_name
sql>
<select id="selectByExample" parameterType="com.cy.dao.model.TestUserExample" resultMap="BaseResultMap">
select
<if test="distinct">
distinct
if>
<include refid="Base_Column_List" />
from test_user
<if test="_parameter != null">
<include refid="Example_Where_Clause" />
if>
<if test="orderByClause != null">
order by ${orderByClause}
if>
select>
<select id="selectByPrimaryKey" parameterType="java.lang.Integer" resultMap="BaseResultMap">
select
<include refid="Base_Column_List" />
from test_user
where id = #{id,jdbcType=INTEGER}
select>
<delete id="deleteByPrimaryKey" parameterType="java.lang.Integer">
delete from test_user
where id = #{id,jdbcType=INTEGER}
delete>
<delete id="deleteByExample" parameterType="com.cy.dao.model.TestUserExample">
delete from test_user
<if test="_parameter != null">
<include refid="Example_Where_Clause" />
if>
delete>
<insert id="insert" parameterType="com.cy.dao.model.TestUser">
insert into test_user (id, login_name, user_name
)
values (#{id,jdbcType=INTEGER}, #{loginName,jdbcType=VARCHAR}, #{userName,jdbcType=VARCHAR}
)
insert>
<insert id="insertSelective" parameterType="com.cy.dao.model.TestUser">
insert into test_user
<trim prefix="(" suffix=")" suffixOverrides=",">
<if test="id != null">
id,
if>
<if test="loginName != null">
login_name,
if>
<if test="userName != null">
user_name,
if>
trim>
<trim prefix="values (" suffix=")" suffixOverrides=",">
<if test="id != null">
#{id,jdbcType=INTEGER},
if>
<if test="loginName != null">
#{loginName,jdbcType=VARCHAR},
if>
<if test="userName != null">
#{userName,jdbcType=VARCHAR},
if>
trim>
insert>
<select id="countByExample" parameterType="com.cy.dao.model.TestUserExample" resultType="java.lang.Long">
select count(*) from test_user
<if test="_parameter != null">
<include refid="Example_Where_Clause" />
if>
select>
<update id="updateByExampleSelective" parameterType="map">
update test_user
<set>
<if test="record.id != null">
id = #{record.id,jdbcType=INTEGER},
if>
<if test="record.loginName != null">
login_name = #{record.loginName,jdbcType=VARCHAR},
if>
<if test="record.userName != null">
user_name = #{record.userName,jdbcType=VARCHAR},
if>
set>
<if test="_parameter != null">
<include refid="Update_By_Example_Where_Clause" />
if>
update>
<update id="updateByExample" parameterType="map">
update test_user
set id = #{record.id,jdbcType=INTEGER},
login_name = #{record.loginName,jdbcType=VARCHAR},
user_name = #{record.userName,jdbcType=VARCHAR}
<if test="_parameter != null">
<include refid="Update_By_Example_Where_Clause" />
if>
update>
<update id="updateByPrimaryKeySelective" parameterType="com.cy.dao.model.TestUser">
update test_user
<set>
<if test="loginName != null">
login_name = #{loginName,jdbcType=VARCHAR},
if>
<if test="userName != null">
user_name = #{userName,jdbcType=VARCHAR},
if>
set>
where id = #{id,jdbcType=INTEGER}
update>
<update id="updateByPrimaryKey" parameterType="com.cy.dao.model.TestUser">
update test_user
set login_name = #{loginName,jdbcType=VARCHAR},
user_name = #{userName,jdbcType=VARCHAR}
where id = #{id,jdbcType=INTEGER}
update>
mapper>
import com.cy.dao.model.TestUser;
import java.util.List;
public interface TestUserCustomMapper {
int batchInsert(List<TestUser> list);
}
DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.cy.dao.mapper.custom.TestUserCustomMapper">
<resultMap id="BaseResultMap" type="com.cy.dao.model.TestUser">
<id column="id" jdbcType="INTEGER" property="id" />
<result column="login_name" jdbcType="VARCHAR" property="loginName" />
<result column="user_name" jdbcType="VARCHAR" property="userName" />
resultMap>
<sql id="Base_Column_List">
id, login_name, user_name
sql>
<insert id="batchInsert" parameterType="com.cy.dao.model.TestUser">
insert into test_user (id, login_name, user_name
)
values
<foreach collection="list" item="item" separator=",">
(#{item.id,jdbcType=INTEGER}, #{item.loginName,jdbcType=VARCHAR}, #{item.userName,jdbcType=VARCHAR})
foreach>
insert>
mapper>
import org.springframework.stereotype.Service;
import javax.annotation.Resource;
import java.util.ArrayList;
import java.util.List;
@Service
public class TestUserEventBiz extends BaseEventBiz<TestUserMapper, TestUser> {
@Resource
private TestUserCustomMapper testUserCustomMapper;
public int customBatchInsert(List<TestUser> list){
int result = 0;
List<TestUser> l = new ArrayList<>();
for (int i = 0; i < list.size(); i++) {
l.add(list.get(i));
if (i % batchCount == 0) {
testUserCustomMapper.batchInsert(l);
l.clear();
}
}
if (l.size()>0) {
testUserCustomMapper.batchInsert(l);
}
return result;
}
public int insert(TestUser testUser) {
return getMapper().insert(testUser);
}
}
测试类
import com.cy.biz.TestUserEventBiz;
import com.cy.dao.model.TestUser;
import lombok.extern.slf4j.Slf4j;
import org.junit.Before;
import org.junit.FixMethodOrder;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.MethodSorters;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.test.context.junit4.SpringRunner;
import org.springframework.util.StopWatch;
import javax.annotation.Resource;
import java.util.ArrayList;
import java.util.Collection;
import java.util.List;
@Slf4j
@RunWith(SpringRunner.class)
@SpringBootTest(classes = RedisLockApplication.class)
@FixMethodOrder(MethodSorters.NAME_ASCENDING)
public class UserTest {
@Resource
private TestUserEventBiz testUserEventBiz;
private List<TestUser> list1W;
private List<TestUser> list10W;
private List<TestUser> list100W;
@Before
public void init(){
list1W = new ArrayList<>();
list10W = new ArrayList<>();
list100W = new ArrayList<>();
int count = 1000000;
while(count > 0){
String s = String.valueOf(count);
TestUser testUser = new TestUser();
testUser.setUserName(s);
testUser.setLoginName(s);
if (count > 990000) {
list1W.add(testUser);
}
if (count > 900000) {
list10W.add(testUser);
}
list100W.add(testUser);
count -- ;
}
}
@Test
public void aBatchInsert() {
StopWatch stopWatch = new StopWatch("sqlSession批插入");
stopWatch.start("1W");
testUserEventBiz.batchInsert(list1W);
stopWatch.stop();
stopWatch.start("10W");
testUserEventBiz.batchInsert(list10W);
stopWatch.stop();
stopWatch.start("100W");
testUserEventBiz.batchInsert(list100W);
stopWatch.stop();
log.info(stopWatch.prettyPrint());
}
@Test
public void bCustomBatchInsert() {
StopWatch stopWatch = new StopWatch("foreach批插入");
stopWatch.start("1W");
testUserEventBiz.customBatchInsert(list1W);
stopWatch.stop();
stopWatch.start("10W");
testUserEventBiz.customBatchInsert(list10W);
stopWatch.stop();
stopWatch.start("100W");
testUserEventBiz.customBatchInsert(list100W);
stopWatch.stop();
log.info(stopWatch.prettyPrint());
}
}
2023-06-11 17:50:32.351 INFO 31172 --- [main] com.cy.UserTest - StopWatch 'sqlSession批插入': running time = 41413780000 ns
---------------------------------------------
ns % Task name
---------------------------------------------
923910700 002% 1W
3991819400 010% 10W
36498049900 088% 100W
2023-06-11 17:50:47.562 INFO 31172 --- [main] com.cy.UserTest - StopWatch 'foreach批插入': running time = 15136264900 ns
---------------------------------------------
ns % Task name
---------------------------------------------
271883200 002% 1W
1653087200 011% 10W
13211294500 087% 100W
以下是使用不同方法进行批量插入的性能效率比较表格,具体数据可能因环境和数据量不同而有所不同,仅供参考:
方法 | 数据量 | 耗时 |
---|---|---|
foreach标签 | 1W | 271ms |
foreach标签 | 10W | 1653ms |
foreach标签 | 100W | 13211ms |
SqlSession批量操作方法 | 1W | 924ms |
SqlSession批量操作方法 | 10W | 3992ms |
SqlSession批量操作方法 | 100W | 36498ms |
从上表可以看出,使用foreach标签的性能效率较高。以上数据仅供参考,实际效率可能会受到多种因素的影响。在实际使用中,需要根据具体情况进行选择。