脚本sql
XML配置方式的动态SQL我就不讲了,有兴趣可以自己了解,下面是用<script>的方式把它照搬过来,用注解来实现。适用于xml配置转换到注解配置
@Select("<script>select * from user <if test=\"id !=null \">where id = #{id} </if></script>")
public List<User> findUserById(User user);
很明显,在java中写xml可读性和维护性太差,尤其当SQL很长时,这样写是很痛苦的。在这里就不多说了。
在方法中构建sql
@SelectProvider(type=xxxx.class,method=”xxxx”)
个人理解:
用自定义的provider类构造SQL语句
属性详解:
type 属性用于指定获取sql语句的指定类 method 属性用于指定类中要执行获取sql语句的方法
下面就给个例子吧
package cn.et.demo04.annotation.mapper;
import cn.et.demo04.annotation.model.Student;
import org.apache.ibatis.annotations.Param;
import org.apache.ibatis.annotations.Result;
import org.apache.ibatis.annotations.Results;
import org.apache.ibatis.annotations.SelectProvider;
import java.util.List;
public interface StudentMapper {
/**
* 用字符串的方式拼接sql返回
* @return
*/
@Results({
@Result(column = "sid",property = "id"),
@Result(column = "sname",property = "name"),
@Result(column = "sage",property = "age"),
@Result(column = "ssex",property = "sex")
})
@SelectProvider(type =StudentProvider.class ,method = "getStudent")
List<Student> getStudent(@Param("name") String name, @Param("age") String age, @Param("sex") String sex);
}
这种方式拼接sql只能用键值对的方式来接收数据,这是mybatis内部机制造成的,其参数需要是key/value结构,当遇到这里不是key/value结构时,mybatis会自己把它转换成key/value结构,key就是他的名字"name",value就是他的值,要正确传参需要使用key/value结构的map,如下
package cn.et.demo04.annotation.mapper;
import org.apache.ibatis.jdbc.SQL;
import java.util.Map;
public class StudentProvider {
/**
* 用字符串的方式拼接sql返回
* @param map
* @return
*/
public String getStudent(Map map){
String sql ="select * from student where 1=1";
if (map.get("name") != null && !map.get("name").equals("")){
String name ="'%"+map.get("name")+"%'";
sql += " and sname like " +name;
}
if (map.get("age") != null && !map.get("age").equals("")){
String age ="'%"+map.get("age")+"%'";
sql += " and sage like " +age;
}
if (map.get("sex") != null && !map.get("sex").equals("")){
String sex ="'%"+map.get("sex")+"%'";
sql += " and ssex like " +sex;
}
return sql;
}
}
这比<script>更加清晰,适用于查询语句不是很长、条件不多的场景,SQL很直观。但是在写很长的SQL时,这样拼接SQL同样会很痛苦
结构化SQL
package cn.et.demo04.annotation.mapper;
import cn.et.demo04.annotation.model.Student;
import org.apache.ibatis.annotations.Param;
import org.apache.ibatis.annotations.Result;
import org.apache.ibatis.annotations.Results;
import org.apache.ibatis.annotations.SelectProvider;
import java.util.List;
public interface StudentMapper {
/**
* 通过SQL这个类文成动态拼接
* @return
*/
@Results({
@Result(column = "sid",property = "id"),
@Result(column = "sname",property = "name"),
@Result(column = "sage",property = "age"),
@Result(column = "ssex",property = "sex")
})
@SelectProvider(type =StudentProvider.class ,method = "sqlStudent")
List<Student> sqlStudent(@Param("name") String name, @Param("age") String age, @Param("sex") String sex);
}
package cn.et.demo04.annotation.mapper;
import org.apache.ibatis.jdbc.SQL;
import java.util.Map;
public class StudentProvider {
/**
* 方案二 通过SQL这个类文成动态拼接
* @param map
* @return
*/
public String sqlStudent(Map map){
SQL sql =new SQL();
sql.SELECT("*").FROM("student");
if (map.get("name") != null && !map.get("name").equals("")){
String name ="'%"+map.get("name")+"%'";
sql.WHERE(" sname like "+name);
}
if (map.get("age") != null && !map.get("age").equals("")){
String age ="'"+map.get("age")+"'";
sql.WHERE(" sage= "+age);
}
if (map.get("sex") != null && !map.get("sex").equals("")){
String sex ="'"+map.get("sex")+"'";
sql.WHERE(" ssex="+sex);
}
return sql.toString();
}
}
这是把前面的内部类改造一下
SELECT:表示要查询的字段,如果一行写不完,可以在第二行再写一个SELECT,这两个SELECT会智能的进行合并而不会重复
FROM和WHERE:跟SELECT一样,可以写多个参数,也可以在多行重复使用,最终会智能合并而不会报错
这样语句适用于写很长的SQL时,能够保证SQL结构清楚。便于维护,可读性高。但是这种自动生成的SQL和HIBERNATE一样,在实现一些复杂语句的SQL时会束手无策。所以需要根据现实场景,来考虑使用哪一种动态SQL
上面的例子只是最基本的用法:更多详细用法,可以参考mybatis中文网的专门介绍
http://www.mybatis.org/mybatis-3/zh/statement-builders.html