`
wdmcygah
  • 浏览: 60598 次
社区版块
存档分类
最新评论

Mybatis实用Mapper SQL汇总示例

阅读更多

Mybatis作为一个非常好用的持久层框架,相关资料真的是少得可怜,所幸的是官方文档还算详细。本博文主要列举一些个人感觉比较常用的场景及相应的Mapper SQL写法,希望能够对大家有所帮助。

不少持久层框架对动态SQL的支持不足,在SQL需要动态拼接时非常苦恼,而Mybatis很好地解决了这个问题,算是框架的一大亮点。对于常见的场景,例如:批量插入/更新/删除,模糊查询,多条件查询,联表查询,都有非常好的支持。Mybatis的动态SQL生成功能实际使用的是OGNL表达式语言,理解OGNL表达式对动态SQL的使用会有很大程度的帮助。

下面直接上示例:

一、批量插入/更新/删除

批量操作主要使用的是Mybatis的foreach,遍历参数列表执行相应的操作,所以批量插入/更新/删除的写法是类似的,只是SQL略有区别而已。MySql批量操作需要数据库连接配置allowMultiQueries=true才可以。
(1)批量插入 

<insert id="batchInsert" parameterType="java.util.List" useGeneratedKeys="true">
		<foreach close="" collection="list" index="index" item="item" open="" separator=";">
			insert into user (name, age,dept_code) values
			(#{item.name,jdbcType=VARCHAR},
			#{item.age,jdbcType=INTEGER},
			 #{item.deptCode,jdbcType=VARCHAR}
			)
		</foreach>
	</insert>

 

上面演示的是MySql的写法(表主键自增的写法),因为MySql支持主键自增,所以直接设置useGeneratedKeys=true,即可在插入数据时自动实现主键自增;不需要自增时就不需要设置useGeneratedKeys,而且插入SQL包含所有字段即可。实际Mysql还有另外一种写法,就是拼接values的写法,这种方法我测试过比多条insert语句执行的效率会高些。不过需要注意一次批量操作的数量做一定的限制。具体写法如下: 

<insert id="batchInsert" parameterType="java.util.List" useGeneratedKeys="true">
		insert into user (name, age,dept_code) values
		<foreach collection="list" index="index" item="item" open="" close="" separator=",">
			(#{item.name,jdbcType=VARCHAR},
			#{item.age,jdbcType=INTEGER},
			 #{item.deptCode,jdbcType=VARCHAR}
			)
		</foreach>
	</insert>

 对于Oracle不支持主键自增,需要序列替换,所以在SQL写法上略有不同,需要在insert语句前加个 <selectKey>...</selectKey>告知Mybatis主键如何生成(selectKey中间的内容有省略,实际是生成主键的SQL)。

 

 (2)批量更新 

<update id="batchUpdate" parameterType="java.util.List">
		<foreach close="" collection="list" index="index" item="item" open="" separator=";">
			update user set name=#{item.name,jdbcType=VARCHAR},age=#{item.age,jdbcType=INTEGER}
			where id=#{item.id,jdbcType=INTEGER}
		</foreach>
	</update>

 

(3)批量删除 

<delete id="batchDelete" parameterType="java.util.List">
		<foreach close="" collection="list" index="index" item="item" open="" separator=";">
			delete from user
			where id=#{item.id,jdbcType=INTEGER}
		</foreach>
	</delete>

 

二、模糊查询 

<select id="selectLikeName" parameterType="java.lang.String" resultMap="BaseResultMap">
		select
		<include refid="Base_Column_List" />
		from user
		where name like CONCAT('%',#{name},'%' ) 
	</select>

 上面的模糊查询语句是Mysql数据库的写法示例,用到了Mysql的字符串拼接函数CONCAT,其它数据库使用相应的函数即可。

 

三、多条件查询

多条件查询常用到Mybatis的if判断,这样只有条件满足时,才生成对应的SQL。 

<select id="selectUser" parameterType="map" resultMap="BaseResultMap">
		select
		<include refid="Base_Column_List" />
		from user
		<where>
			<if test="name != null">
				name = #{name,jdbcType=VARCHAR}
			</if>
			<if test="age != null">
				and age = #{age,jdbcType=INTEGER}
			</if>
		</where>
	</select>

 

四、联表查询

联表查询在返回结果集为多张表的数据时,可以通过继承resultMap,简化写法。例如下面的示例,结果集在User表字段的基础上添加了Dept的部门名称。 

<resultMap id="ExtResultMap" type="com.research.mybatis.generator.model.UserExt" extends="BaseResultMap">
   	 <result column="name" jdbcType="VARCHAR" property="deptName" />
  </resultMap>
	
	<select id="selectUserExt" parameterType="map" resultMap="ExtResultMap">
		select
		 	u.*, d.name
		from user u inner join dept d on u.dept_code = d.code
		<where>
			<if test="name != null">
				u.name = #{name,jdbcType=VARCHAR}
			</if>
			<if test="age != null">
				and u.age = #{age,jdbcType=INTEGER}
			</if>
		</where>
	</select>

 

 上述示例皆在MySql数据库下测试通过,这里限于篇幅,相应的代码及测试类我就不贴上来了。完整的代码及测试类,可以查看我的Github项目地址:https://github.com/wdmcygah/research-mybatis。主要是UserService和UserServiceTest两个类,仓库里面还有些与本博文不相关的代码,注意区分。

 

 

4
3
分享到:
评论
4 楼 yin_bp 2015-02-02  
wdmcygah 写道

前辈您好,看了您的bboss项目介绍,功能非常强大,作为中国制造的开源项目必须要赞一个,向您学习。不过这里向您吐个小槽,昨天我本来想把项目clone下来看的,结果在github下了两个小时都没成功,download最后也网络错误了。您的项目真是太大了,不知是否可以考虑按照模块拆成多个Git库,小建议。


项目确实比较大,里面都是完整的eclipse工程,依赖的jar也在里面,下下来直接就可以编译部署和运行的,而且包含了很多demo,后面我会把jar文件统一整理一下,这样会大大降低工程目录的大写。网络比较慢的情况确实不容易下载,可以用svn获取里面的相应工程bestpractice下有很多demo:
https://github.com/bbossgroups/bbossgroups-3.5/tree/master/bestpractice

大家可以选择性地用svn checkout里面的demo工程,例如:
checkout bboss demo eclipse工程的svn地址清单:
https://github.com/bbossgroups/bbossgroups-3.5/trunk/bestpractice/mvc
https://github.com/bbossgroups/bbossgroups-3.5/trunk/bestpractice/persistent
https://github.com/bbossgroups/bbossgroups-3.5/trunk/bestpractice/session
https://github.com/bbossgroups/bbossgroups-3.5/trunk/bestpractice/sessionmonitor
https://github.com/bbossgroups/bbossgroups-3.5/trunk/bestpractice/xmlrequest
https://github.com/bbossgroups/bbossgroups-3.5/trunk/bestpractice/xmlserializable
https://github.com/bbossgroups/bbossgroups-3.5/trunk/bestpractice/easyuidatagrid
https://github.com/bbossgroups/bbossgroups-3.5/trunk/bestpractice/demoproject
https://github.com/bbossgroups/bbossgroups-3.5/trunk/bestpractice/bbossupload
https://github.com/bbossgroups/bbossgroups-3.5/trunk/bestpractice/https://github.com/bbossgroups/bbossgroups-3.5/tree/master/bestpractice/bboss-clientproxy

同样也可通过参考上面的地址用svn checkout bboss的各个核心eclipse工程,例如checkout 持久层工程svn地址:
https://github.com/bbossgroups/bbossgroups-3.5/trunk/bboss-persistent
这些工程导入eclipse即可使用了,直接可以采用ant构建核心工程,ant构建指令参考文档:
http://yin-bp.iteye.com/blog/2142959

下载bboss,网速要给力呢
3 楼 wdmcygah 2015-01-31  
yin_bp 写道
上面的例子都是bboss 持久层or maping的模式,接下来看一个原生sql批处理删除操作的玩法。
sql配置:
<property name="deleteByKey">
		<![CDATA[
			delete from TD_APP_BOM where id=?
		]]>
	</property>

java代码:
/**
	 * 用同样的sql只删一条记录,根据主键删除台账信息
	 * @throws AppBomException 
	 */
	public boolean delete(String id) throws AppBomException {
		try {
			executor.delete("deleteByKey", id);
		} catch (Throwable e) {
			throw new AppBomException("delete appbom failed::id="+id,e);
		}
		return true ;
	}

	/**
	 * 用同样的sql批量删除记录,受事务管理控制,如果有记录删除失败,则全部回滚之前的记录,当然也可以不要事务,也就是出错后
	 * 前面的记录成功、后面的记录不入库
	 * @param beans
	 * @return
	 * @throws AppBomException 
	 */
	public boolean deletebatch(String ... ids) throws AppBomException {
		TransactionManager tm = new TransactionManager();
		try {
			tm.begin();
			executor.deleteByKeys("deleteByKey", ids);
			tm.commit();
		} catch (Throwable e) {
			
			throw new AppBomException("batch delete appbom failed::ids="+ids,e);
		}
		finally
		{
			tm.release();
		}

		return true;
	}
//不要事务的模式
	public boolean deletebatch(String ... ids) throws AppBomException {
		
		try {
			
			executor.deleteByKeys("deleteByKey", ids);
			
		} catch (Throwable e) {
			
			throw new AppBomException("batch delete appbom failed::ids="+ids,e);
		}
		

		return true;
	}



前辈您好,看了您的bboss项目介绍,功能非常强大,作为中国制造的开源项目必须要赞一个,向您学习。不过这里向您吐个小槽,昨天我本来想把项目clone下来看的,结果在github下了两个小时都没成功,download最后也网络错误了。您的项目真是太大了,不知是否可以考虑按照模块拆成多个Git库,小建议。
2 楼 yin_bp 2015-01-30  
上面的例子都是bboss 持久层or maping的模式,接下来看一个原生sql批处理删除操作的玩法。
sql配置:
<property name="deleteByKey">
		<![CDATA[
			delete from TD_APP_BOM where id=?
		]]>
	</property>

java代码:
/**
	 * 用同样的sql只删一条记录,根据主键删除台账信息
	 * @throws AppBomException 
	 */
	public boolean delete(String id) throws AppBomException {
		try {
			executor.delete("deleteByKey", id);
		} catch (Throwable e) {
			throw new AppBomException("delete appbom failed::id="+id,e);
		}
		return true ;
	}

	/**
	 * 用同样的sql批量删除记录,受事务管理控制,如果有记录删除失败,则全部回滚之前的记录,当然也可以不要事务,也就是出错后
	 * 前面的记录成功、后面的记录不入库
	 * @param beans
	 * @return
	 * @throws AppBomException 
	 */
	public boolean deletebatch(String ... ids) throws AppBomException {
		TransactionManager tm = new TransactionManager();
		try {
			tm.begin();
			executor.deleteByKeys("deleteByKey", ids);
			tm.commit();
		} catch (Throwable e) {
			
			throw new AppBomException("batch delete appbom failed::ids="+ids,e);
		}
		finally
		{
			tm.release();
		}

		return true;
	}
//不要事务的模式
	public boolean deletebatch(String ... ids) throws AppBomException {
		
		try {
			
			executor.deleteByKeys("deleteByKey", ids);
			
		} catch (Throwable e) {
			
			throw new AppBomException("batch delete appbom failed::ids="+ids,e);
		}
		

		return true;
	}


1 楼 yin_bp 2015-01-30  
楼主可以对比一下bboss持久层类似功能的做法:
1.批量增删改-真正采用jdbc的预编译批处理来实现,性能杠杠的,sql语句只需要配置一条,无需foreach

这里以更新为实例,新增和删除类似:
<property name="updateuser">
		<![CDATA[ update user set name=#[name],age=#[age]where id=#[id]
]]>
	</property>


执行批处理操作
List<User> users =...;
executor.updateBeans("updateuser",users);//在bboss默认数据源上执行 ,多个用户的更新操作以预编译批处理的模式执行  
executor.updateBeans(dbname,"updateuser",users);//在指定的bboss数据源上执行 

参考文档:
bboss预编译批处理api使用介绍

2.模糊查询
以下是mysql数据库依赖模式:
<property name="selectGroupInfoByNames">
		<![CDATA[ select *  
from user   
where name like CONCAT('%',#[name]'%' )    
]]>
	</property>


Map<String, String> condition = new HashMap<String, String>();//定义包含变量的map对象,key为变量名称,value为变量值   
condition.put("name", "john");  
List<CandidateGroup> list = executor.queryListBean(CandidateGroup.class,   
               "selectGroupInfoByNames", condition _);//执行查询 


以下是数据库无关模式
<property name="selectGroupInfoByNames">
		<![CDATA[ select *  
from user   
where name like #[name]
]]>
	</property>


Map<String, String> condition = new HashMap<String, String>();//定义包含变量的map对象,key为变量名称,value为变量值   
condition.put("name", "%john%");  
List<CandidateGroup> list = executor.queryListBean(CandidateGroup.class,   
               "selectGroupInfoByNames", condition _);//执行查询 


3.动态sql
bboss 动态sql使用foreach循环示例  
bboss持久层框架动态sql语句配置和使用
bboss 持久层sql语句中一维/多维数组类型变量、list变量、map变量、bean对象变量使用说明

相关推荐

    MyBatis-Plus 的官方示例(mybatis-plus-samples-master.zip)

    本工程为 MyBatis-Plus 的官方示例,项目结构如下: mybatis-plus-sample-quickstart: 快速开始示例 mybatis-plus-sample-quickstart-springmvc: 快速开始...mybatis-plus-sample-execution-analysis: Sql执行分析示例

    springboot+springmvc+mybatis+mysql示例项目

    springboot+springmvc+mybatis+mysql示例项目,其中mybatis可以用工具自动生成dao,mapper和vo,并且有spring的配置,springboot启动方式

    mabatis的mapper示例文件(MySQL)

    mabatis的mapper示例文件,包含select、update、insert、delete、Sql标签、resultMap标签、where标签、set标记等标签的使用,已经一些SQL语法,可覆盖项目实际开发中95%的使用情况,这份文件是在三个项目中总结而来...

    mybatisGenerator配置文件示例

    mybatisGenerator配置文件示例, java开发利器, 使用mybatisGenerator可以快速根据数据库表定义生成java实体类mapper.xml映射文件或者注解sql代码, 强烈推荐.

    springboot+mybatis+内置tomcat示例.rar

    mybatis.mapper-locations=classpath:mapping/*Mapper.xml mybatis.type-aliases-package=entity #spring.profiles=development #server.address=127.0.0.1 #端口设置 server.port=8092 #日志配置 ...

    mybatis-log-plugin:将Mybatis SQL日志还原到原始的整个可执行SQL

    还原MyBatis输出的日志为完整的SQL语句。 把SQL日志里面的?替换为真正的参数值。 选中要还原的MyBatis日志,右键点击菜单Restore Sql,还原SQL语句. Java接口方法与Mapper xml文件互相跳转。 按钮作用 Text: 从文本...

    SSM框架的学习与应用-Java EE企业级应用开发学习记录(第三天)Mybatis的深入学习(动态sql的操作)

    本资源主要介绍了MyBatis动态SQL的使用。通过对MyBatis的动态SQL各个元素进行详细说明,并且有测试类帮助理解。...综上,本资源主要通过大量示例详细介绍了MyBatis各种动态SQL元素的用法,可以作为学习动态SQL的参考。

    vertx-mybatis:使用mybatis NON-BLOCK&ASYNCHRONOUS的vertx sqlclient模板

    然后,您组织项目,以便在不放弃SQL Mapper框架(MyBatis)的情况下易于使用ORM框架。描述概要纯vertx sql客户端下面的代码块是vertx提供的纯sql客户端代码的示例。 我想使用(不放弃)像mybatis这样SQL Mapper框架...

    springboot-mybatis-druid-sqlserver-maven:一个基于springboot+mybatis+druid+sqlserver+maven的demo

    springboot-mybatis-druid-sqlserver-maven 一个基于springboot+mybatis+druid+sqlserver+maven的...只是一个示例项目,关于数据查询 有注解 和 mapper.xml 两种实现,CityMapper.xml 字段未完全设置正确,但不影响。

    springmybatis

    mybatis实战教程mybatis in action之八mybatis 动态sql语句 mybatis实战教程mybatis in action之九mybatis 代码生成工具的使用 mybatis SqlSessionDaoSupport的使用附代码下载 转自:...

    Mybatis编程示例:基于注解的实现

    Mybatis编程示例:基于注解的实现 使用mysql数据库,配置好数据库后即可直接使用 mybatis基于注解的实现有以下步骤: 步骤1、定义sql映射接口(在src/main/java下创建mapper包) 步骤2、将sql映射接口添加到Mybatis...

    Mybatis编程示例:基于XML的实现

    Mybatis编程:基于XML的实现 使用mysql数据库,配置下数据库就可以直接运行 Mybatis编程步骤分为: 步骤1、创建好MySQL数据库 步骤2、在pom.xml中添加mysql-connector-java和Mybatis依赖 步骤3、创建实体类User...

    mybatis统计每条SQL的执行时间的方法示例

    此方案主要是通过环绕切面的方式将mapper包下的接口方法,然后前后计算时间差即可。这就是典型的AOP知识,不过这种计算比较粗糙,但是也是个办法。具体方法如下: @Aspect @Component @Slf4j public class ...

    freemarker代码生成器示例

    源码 用于Mybatis框架下生成Model.java和Mapper.xml文件, 内含采用mysql数据库建表文件, 分别在ModelHandler类和MapperXMLHandler类中的main方法中执行。 文件存放路径D://generatordata。

    MyBatis3.2.3帮助文档(中文版).zip

    mappers 元素则是包含一组 mapper 映射器(这些 mapper 的 XML 文件包含了 SQL 代码和映射定义信息)。 不使用 XML 构建 SqlSessionFactory 如果你更愿意直接从 Java 程序而不是 XML 文件中创建 configuration,...

    基于Spring + Spring MVC + Mybatis

    封装了常用的CURD,配合mybatis-generator 自动生成dao、model、mapper层,减少重复劳动,提高生产力,实现快速、平稳的开发 实现Mybatis的分页查询模块,支持MySQL、PostgreSQL、SQLServer等数据库分页查询 通用的...

    mybatisIntercept.zip

    3、代码的调整示例 接口实现类 需要把Page对象参数传入 ,SQL拦截器检测到 page 对象就会进行拦截 并且附值 public List, Object&gt;&gt; getConditionsPage(Page page, String condition , String recognitionresult, ...

    tkmybatis-tempate:tkmybatis自动生成模板

    Tkmybatis是基于Mybatis框架开发的一个工具,通过调用它提供的方法实现对单表的数据操作,不需要写任何sql语句,这极大地提高了项目开发效率。 代码说明 tk-mybatis为spring boot集成的tkmybatis ,在配置好文件后,...

    mybatis-plus-generator-ui:对mybatis-plus-generator进行封装,通过Web UI快速生成兼容的Spring boot,mybatis-plus框架的各类业务代码

    提供一致的Web UI用于生成兼容mybatis-plus框架的相关功能代码,包括Entity,Mapper,Mapper.xml,Service,Controller等,可以自定义模板以及各种输出参数,也可以通过SQL查询语句直接生成代码。 使用方法 ♡maven...

Global site tag (gtag.js) - Google Analytics