经验首页 前端设计 程序设计 Java相关 移动开发 数据库/运维 软件/图像 大数据/云计算 其他经验
当前位置:技术经验 » 数据库/运维 » MyBatis » 查看文章
tk.mybatis扩展通用接口使用详解
来源:jb51  时间:2019/8/21 10:54:05  对本文有异议

 一.tk.mybatis已经为我们封装好了许多拆箱即用的通用mapper,但在实际的项目开发中想必不少小伙伴在数据库设计中都会采用逻辑删除这种方案,再去使用通用的mapper接口就不行了。

这时候就需要我们封装一些扩展的通用Mapper接口。

二.项目中提供了大量现成的方法,这些方法可以作为扩展时的参考。

例如 selectAll 方法。

首先定义接口:

  1. @RegisterMapper
  2. public interface SelectAllMapper<T> {
  3. /**
  4. * 查询全部结果
  5. *
  6. * @return
  7. */
  8. @SelectProvider(type = MySelectProvider.class, method = "dynamicSQL")
  9. List<T> selectAll();
  10. }

其中 MySelectProvider 是你要实现的一个类,该类需要继承 MapperTemplate。@RegisterMapper 注解可以避免 mappers 参数配置,通用 Mapper 检测到该接口被继承时,会自动注册。

  1. import org.apache.ibatis.mapping.MappedStatement;
  2. import tk.mybatis.mapper.mapperhelper.MapperHelper;
  3. import tk.mybatis.mapper.mapperhelper.MapperTemplate;
  4. import tk.mybatis.mapper.mapperhelper.SqlHelper;
  5. public class MySelectProvider extends MapperTemplate {
  6. public BaseSelectProvider(Class<?> mapperClass, MapperHelper mapperHelper) {
  7. super(mapperClass, mapperHelper);
  8. }
  9. /**
  10. * 查询全部结果
  11. *
  12. * @param ms
  13. * @return
  14. */
  15. public String selectAll(MappedStatement ms) {
  16. final Class<?> entityClass = getEntityClass(ms);
  17. //修改返回值类型为实体类型
  18. setResultType(ms, entityClass);
  19. StringBuilder sql = new StringBuilder();
  20. sql.append(SqlHelper.selectAllColumns(entityClass));
  21. sql.append(SqlHelper.fromTable(entityClass, tableName(entityClass)));
  22. sql.append(SqlHelper.orderByDefault(entityClass));
  23. return sql.toString();
  24. }
  25. }

其中 selectAll 方法名要和接口中定义的方法名一致。其次就是该方法的参数为 MappedStatement类型。

在 selectAll 方法中,首先是获取了当前接口的实体类型:

final Class<?> entityClass = getEntityClass(ms);

因为接口返回值类型为 List<T>,MyBatis 会认为返回值类型为 List<Object>,这和我们想要的实体类型不一样,所以下一行代码就是设置返回值类型:

setResultType(ms, entityClass);

注意,只有返回 T 或者 List 时需要设置,返回 int 类型时不需要设置。

接下来就是纯粹的拼接 XML 形式的 SQL 了。

  1. /select col1,col2...
  2. sql.append(SqlHelper.selectAllColumns(entityClass));
  3. //from tablename - 支持动态表名
  4. sql.append(SqlHelper.fromTable(entityClass, tableName(entityClass)));
  5. //order by xxx
  6. sql.append(SqlHelper.orderByDefault(entityClass));

当你想要实现某种方法时,可以从已有的例子中找一个最接近的方法,在此基础上进行修改

三.例:根据主键查询单个实体对象(过滤掉逻辑删除的实体,注:我的数据表逻辑删除字段定义为enabled_status)

首先定义mapper

  1. @RegisterMapper
  2. public interface SelectByKeyAndNotDeletedMapper<T> {
  3. /**
  4. * 根据主键查询没有被逻辑删除的实体
  5. *
  6. * @return
  7. */
  8. @SelectProvider(type = SelectByKeyNotDeletedProvider.class, method = "dynamicSQL")
  9. T selectByKeyNotDeleted(Object key);
  10. }

其次定义SelectByKeyNotDeletedProvider

  1. public class SelectByKeyNotDeletedProvider extends MapperTemplate {
  2. public SelectByKeyNotDeletedProvider(Class<?> mapperClass, MapperHelper mapperHelper) {
  3. super(mapperClass, mapperHelper);
  4. }
  5.  
  6. public String selectByKeyNotDeleted(MappedStatement ms) {
  7. final Class<?> entityClass = getEntityClass(ms);
  8. //将返回值修改为实体类型
  9. setResultType(ms, entityClass);
  10. StringBuilder sql = new StringBuilder();
  11. sql.append(SqlHelper.selectAllColumns(entityClass));
  12. sql.append(SqlHelper.fromTable(entityClass, tableName(entityClass)));
  13. sql.append(wherePKColumns(entityClass, false));
  14. return sql.toString();
  15. }
  16. private String wherePKColumns(Class<?> entityClass, boolean useVersion) {
  17. StringBuilder sql = new StringBuilder();
  18. sql.append("<where>");
  19. //获取全部列
  20. Set<EntityColumn> columnSet = EntityHelper.getPKColumns(entityClass);
  21. //当某个列有主键策略时,不需要考虑他的属性是否为空,因为如果为空,一定会根据主键策略给他生成一个值
  22. for (EntityColumn column : columnSet) {
  23. sql.append(" AND " + column.getColumnEqualsHolder());
  24. }
  25. if (useVersion) {
  26. sql.append(whereVersion(entityClass));
  27. }
  28. //过滤被逻辑删除的数据
  29. sql.append(" AND enabled_status = 1 ");
  30. sql.append("</where>");
  31. return sql.toString();
  32. }
  33. }

然后定义BasicMapper,让其继承通用Mapper接口以及上面自定义的SelectByKeyAndNotDeletedMapper接口。

@tk.mybatis.mapper.annotation.RegisterMapper public interface BasicMapper<T> extends Mapper<T>, SelectByKeyAndNotDeletedMapper<T> { }

最后在通用service中引入,

@Autowired private BasicMapper<T> mapper;

自定义*.Mapper都去继承BasicMapper<T>即可

public interface UserPOMapper extends BasicMapper<UserPO> {}

注意:最新版本的tk已经支持根据注解@LogicDelete实现逻辑删除

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持w3xue。

 友情链接:直通硅谷  点职佳  北美留学生论坛

本站QQ群:前端 618073944 | Java 606181507 | Python 626812652 | C/C++ 612253063 | 微信 634508462 | 苹果 692586424 | C#/.net 182808419 | PHP 305140648 | 运维 608723728

W3xue 的所有内容仅供测试,对任何法律问题及风险不承担任何责任。通过使用本站内容随之而来的风险与本站无关。
关于我们  |  意见建议  |  捐助我们  |  报错有奖  |  广告合作、友情链接(目前9元/月)请联系QQ:27243702 沸活量
皖ICP备17017327号-2 皖公网安备34020702000426号