当前环境下,mybatis是使用很频繁的一个数据持久层框架。我们很多时候使用xml的方式来配置mybatis的sql,这种方式也称之为mybatis的动态SQL。but,本篇要说的是另一种方式。日常业务中我们可能会遇到很多有关于动态SQL的问题。我们就需要在代码中来编写SQL。
这个时候有的人可能就会想到直接 String sql = "select * from XXX where XXX";对于这种方式我是不推荐的,1、既然是代码,就要有代码的易读性。2、既然是动态sql就要有足够的扩展性。。。。
进入正题:mybatis针对上述问题,提供了一套API来解决我们这个尴尬的窘态。
3.2版本之前:
SqlBuilder 和 SelectBuilder
用法:
public String selectBlogsSql() {
BEGIN(); // Clears ThreadLocal variable
SELECT("*");
FROM("BLOG");
return SQL();
}
private String selectPersonSql() {
BEGIN(); // Clears ThreadLocal variable
SELECT("P.ID, P.USERNAME, P.PASSWORD, P.FULL_NAME");
SELECT("P.LAST_NAME, P.CREATED_ON, P.UPDATED_ON");
FROM("PERSON P");
FROM("ACCOUNT A");
INNER_JOIN("DEPARTMENT D on D.ID = P.DEPARTMENT_ID");
INNER_JOIN("COMPANY C on D.COMPANY_ID = C.ID");
WHERE("P.ID = A.ID");
WHERE("P.FIRST_NAME like ?");
OR();
WHERE("P.LAST_NAME like ?");
GROUP_BY("P.ID");
HAVING("P.LAST_NAME like ?");
OR();
HAVING("P.FIRST_NAME like ?");
ORDER_BY("P.ID");
ORDER_BY("P.FULL_NAME");
return SQL();
}
具体相关文档:http://www.mybatis.org/mybatis-3/zh/statement-builders.html
在3.2版本之前,通过实现ThreadLocal变量来掩盖一些导致Java DSL麻烦的语言限制。但这种方式已经废弃了,现代的框架都欢迎人们使用构建器类型和匿名内部类的想法。因此,SelectBuilder 和 SqlBuilder 类都被废弃了。
3.2版本之后:
实例:
private String selectPersonSql() {
return new SQL() {{
SELECT("P.ID, P.USERNAME, P.PASSWORD, P.FULL_NAME");
SELECT("P.LAST_NAME, P.CREATED_ON, P.UPDATED_ON");
FROM("PERSON P");
FROM("ACCOUNT A");
INNER_JOIN("DEPARTMENT D on D.ID = P.DEPARTMENT_ID");
INNER_JOIN("COMPANY C on D.COMPANY_ID = C.ID");
WHERE("P.ID = A.ID");
WHERE("P.FIRST_NAME like ?");
OR();
WHERE("P.LAST_NAME like ?");
GROUP_BY("P.ID");
HAVING("P.LAST_NAME like ?");
OR();
HAVING("P.FIRST_NAME like ?");
ORDER_BY("P.ID");
ORDER_BY("P.FULL_NAME");
}}.toString();
}
————————————————————————————————————————————————————————————————————————————————————————
当然上述都是这些Mybatis本身提供的一些API,要达到动态SQL,我们还差一步,所以来个简单的demo。
public class MybatisSQLTemplate {
/**
* 插入数据
* @param params
* @return
*/
public String insert(Map<String,Object> params){
return new SQL()
.INSERT_INTO(getTableName(params))
.VALUES(buildFiled(params),buildValue(params))
.toString();
}
/**
* 更新数据
* @param params
* @return
*/
public String update(Map<String,Object> params){
return new SQL()
.UPDATE(getTableName(params))
.SET(buildSet(params))
.WHERE(buildWhere(params))
.toString();
}
/**
* 获取表名
* @param params
* @return
*/
private String getTableName(Map<String,Object> params){
if(params == null || params.get(Condition.TABLE_NAME.name()) == null){
throw new IllegalArgumentException("Table name is required..");
}else {
return params.get(Condition.TABLE_NAME.name()).toString();
}
}
/**
* 构建插入字段
* @param params
* @return
*/
private String buildFiled(Map<String,Object> params){
params.remove(Condition.TABLE_NAME.name());
if(params.isEmpty()){
throw new IllegalArgumentException("Parameter name is required..");
}
String columns = params.keySet().toString();
return columns.substring(1,columns.length()-1);
}
/**
* 构建插入value值
* @param params
* @return
*/
private String buildValue(Map<String,Object> params){
params.remove(Condition.TABLE_NAME.name());
if(params.isEmpty()){
throw new IllegalArgumentException("Parameter value is required..");
}
String columns = params.keySet().stream().map(i->"#{"+i+"}").collect(Collectors.toList()).toString();
return columns.substring(1,columns.length()-1);
}
/**
* 构建更新set数据
* @param params
* @return
*/
private String buildSet(Map<String,Object> params){
params.remove(Condition.TABLE_NAME.name());
if(params.isEmpty()){
throw new IllegalArgumentException("The value to be modified cannot be empty..");
}
Object temp = params.remove(Condition.WHERE.name());
String sets = params.keySet().stream().map(i -> i + "=#{" + i + "}").collect(Collectors.toList()).toString();
params.put(Condition.WHERE.name(),temp);
return sets.substring(1,sets.length()-1);
}
/**
* 构建where条件(目前只支持单个条件)
* @param params
* @return
*/
private String buildWhere(Map<String,Object> params){
params.remove(Condition.TABLE_NAME.name());
if(params.isEmpty() || params.get(Condition.WHERE.name()) == null){
throw new IllegalArgumentException("Missing where condition..");
}
Object where = params.get(Condition.WHERE.name());
//可以做多类型判断以兼容更复杂where条件
if(where instanceof String){
String temp = (String) where;
return temp + "= #{" + where +"}";
}else {
throw new IllegalArgumentException("Where argument is illegal..");
}
}
}
说明一下,上述的代码中使用的参数都是Map。一是为了测试方便,而是降低复杂度。当然你也可以传入自己的db对象。通过反射机制来读取属性,或者通过自定义注解来使整个代码结构更合理。
接下来说明一下,如何使用。
首先了解一下注解@SelectProvider、@InsertProvider、@UpdateProvide、@DeleteProvider,这四个注解分别用于查、插、改、删,都拥有一个type属性以及一个method属性。type属性用来指明生成SQL的类是那个,method用来表明该类下的那个方法。
public interface OrderInfoMapper {
@InsertProvider(type = MybatisSQLTemplate.class,method = "insert")
int insert(Map<String,Object> params);
@UpdateProvider(type = MybatisSQLTemplate.class,method = "update")
int update(Map<String,Object> params);
}
到这里动态SQL就算完成了。这里的Mapper与XML方式对应的Mapper无异。
写在后面:当然MyBatis不仅提供了API方式的SQL,结果集参数等的映射都是可以做到的。而且如果想要自定义Mybatis的一些行为,可以通过编写插件的方式来实现 @Intercepts。