Mybatis单表自动生成SQL

1. 为什么要写单表SQL自动生成呢?

我们在使用Mybatis做dao层开发时,对单表的操作大部分都是一样的,每次都写着一样的代码既浪费时间也得不到任何乐趣,我们为何不通过程序解决这个问题呢?

2. Mybatis中Provider介绍

在使用Mybatis开发过程中,可以通过Mybatis定义的xml文件进行SQL的配置,我们比较少使用其提供的Provider进行SQL编写。Mybatis提供了InsertProviderDeleteProviderUpdateProviderSelectProvider四个注解来进行SQL的配置,以及提供了ResultsResultMapResultType几个注解进行结果映射。至于这几个注解该如何使用我这边就不做详细介绍了,可以参考这篇博客,该文章有比较详细的例子介绍Provider如何使用。下面我们就来介绍如何通过Mybatis提供的Provider来编写SQL自动生成。

3. 通过Provider进行SQL自动生成

3.1. 总体思路

通过定义一个基础接口,在基础接口中定义一系列CRUD操作,在调用的过程中通过反射获取数据库实体(Entity)上定义的表名、列名等元数据,然后根据表名、列名生成要操作的SQL。

3.2 代码实现

  • BaseMapper
public interface BaseMapper<T> {    
    @SelectProvider(type = SQLProvider.class, method = "selectById")    
    T selectById(T item);    
    @SelectProvider(type = SQLProvider.class, method = "selectOne")    
    T selectOne(T item);    
    @SelectProvider(type = SQLProvider.class, method = "select")    
    List<T> select(T item);    
    @InsertProvider(type = SQLProvider.class, method = "insert")    
    int insert(T item);    
    @DeleteProvider(type = SQLProvider.class, method = "delete")    
    int delete(T item);    
    @DeleteProvider(type = SQLProvider.class, method = "deleteById")    
    int deleteById(T item);    
    @UpdateProvider(type = SQLProvider.class, method = "updateById")   
    int updateById(T item);    
    @UpdateProvider(type = SQLProvider.class, method =     "updateSelectiveById")    
    int updateSelectiveById(T item);
}
  • SQLProvider
public class SQLProvider {    
    private static final Logger LOGGER = LoggerFactory.getLogger(SQLProvider.class);    
    private SQLBuilderHelper sqlHelper = SQLBuilderHelper.getInstance();
    public <T> String selectById(T item) throws AutoSQLException {    
    String table = sqlHelper.getTableName(item.getClass());    
    ColumnMapping columnMapping = getIdColumn(item.getClass());    
    if (columnMapping == null) {        
        throw new AutoSQLException(String.format("There is no identity column in table[%s]", table));    
    }    
    SelectSQLBuilder sqlBuilder = new SelectSQLBuilder();    sqlBuilder.table(table)            .condition(columnMapping.getColumnName(), columnMapping.getField().getName(), "=");    
    return sqlBuilder.toSqlString();
}
public <T> String selectOne(T item) throws AutoSQLException {    
    String table = sqlHelper.getTableName(item.getClass());    
    List<ColumnValue> values = sqlHelper.getColumnValue(item); 
    SelectSQLBuilder sqlBuilder = new SelectSQLBuilder();        
    sqlBuilder.table(table);   
    for (ColumnValue value : values) {        
        if (!value.isValueNull()) {    
            sqlBuilder.condition(value.getColumnName(), value.getFieldName(), "=");       
       }   
    }    
    return sqlBuilder.toSqlString();
}
public <T> String select(T item) throws AutoSQLException {    
     return selectOne(item);
}
public <T> String delete(T item) throws AutoSQLException {    
      String table = sqlHelper.getTableName(item.getClass()); 
      List<ColumnValue> values = sqlHelper.getColumnValue(item);  
      DeleteSQLBuilder sqlBuilder = new DeleteSQLBuilder();   
      sqlBuilder.table(table);    
      for (ColumnValue value : values) { 
           if (!value.isValueNull()) {                  
              sqlBuilder.condition(value.getColumnName(), value.getFieldName(), "=");        
            }   
       }    
       return sqlBuilder.toSqlString();
}
public <T> String deleteById(T item) throws AutoSQLException {    
      ColumnMapping mapping = getIdColumn(item.getClass());    
      String table = sqlHelper.getTableName(item.getClass());   
      if (mapping == null) {       
         throw new AutoSQLException(String.format("There is no identity column in table[%s]", table));    
      }   
      DeleteSQLBuilder sqlBuilder = new DeleteSQLBuilder();    
      sqlBuilder.table(table).condition(mapping.getColumnName(), mapping.getFieldName(), "=");    
      return sqlBuilder.toSqlString();
}
public <T> String updateById(T item) throws AutoSQLException {    
      String table = sqlHelper.getTableName(item.getClass());    
      List<ColumnValue> values = sqlHelper.getColumnValue(item);          
      UpdateSQLBuilder sqlBuilder = new UpdateSQLBuilder();   
      sqlBuilder.table(table);    
      for (ColumnValue value : values) {        
          if (!value.getColumnMapping().isId()) {        
              sqlBuilder.set(value.getColumnName(), value.getFieldName());          
          }       
         if (value.getColumnMapping().isId()) {  
            sqlBuilder.condition(value.getColumnName(), value.getFieldName(), "=");        
        }    
      }    
      return sqlBuilder.toSqlString();
}
public <T> String updateSelectiveById(T item) throws AutoSQLException {    
      String table = sqlHelper.getTableName(item.getClass());    
      List<ColumnValue> values = sqlHelper.getColumnValue(item);   
      UpdateSQLBuilder sqlBuilder = new UpdateSQLBuilder();    
      sqlBuilder.table(table);   
      for (ColumnValue value : values) { 
         if (!value.isValueNull() && !value.getColumnMapping().isId()) { 
             sqlBuilder.set(value.getColumnName(), value.getFieldName());      
         }          
        if (value.getColumnMapping().isId()) {          
            sqlBuilder.condition(value.getColumnName(), value.getFieldName(), "=");        
        }  
      }   
      return sqlBuilder.toSqlString();
}
private ColumnMapping getIdColumn(Class<?> clazz) {   
     List<ColumnMapping> mappings = sqlHelper.getColumnMapping(clazz);        
     for (ColumnMapping mapping : mappings) {      
         if (mapping.isId()) {            
            return mapping;       
         }    
     }    
     return null;
}
public <T> String  insert(T item) throws AutoSQLException {  
      String tableName = sqlHelper.getTableName(item.getClass());        
      List<ColumnMapping> values =       
      sqlHelper.getColumnMapping(item.getClass());  
      if (values.isEmpty()) {        
          throw new AutoSQLException(String.format("Table[%s] has no column"));    
      }   
      InsertSQLBuilder sqlBuilder =new InsertSQLBuilder();        
      sqlBuilder.table(tableName);   
     for (ColumnMapping mapping : values) {          
          sqlBuilder.column(mapping.getColumnName());            
          sqlBuilder.field(mapping.getField().getName());   
     }    
     return sqlBuilder.toSqlString();
}
  • SQLBuilderHelper
    SQLBuilderHelper主要是通过反射获取实体中的注解,这里采用javax.persistence Api,表名通过Table注解定义,默认采用实体类的简单名称,主键通过Id注解定义,通过Column注解定义列名,默认使用实例变量名称作为列名,对于不需要映射的字段在示例变量上标注Transient注解。这里要吐槽一下简书的书写代码功能是在太弱,贴个代码好麻烦,这里我就贴出完整代码了。
  • SQLBuilder
    SQLBuilder主要负责根据获取到的表元数据生成SQL,其有四个子类InsertSQLBuilderDeleteSQLBuilderUpdateSQLBuilderUpdateSQLBuilder,这四个子类分别生成增、删、改、查SQL。

3.3 如何使用呢?

我们以对用户的操作为例。

  1. 首先定义一个User实体,如下:
@Table(name = "t_user")
@Getter @Setter
public class User {    
    @Id    
    private Integer id;    
    private String name;    
    private String password;
}

为了追求极简代码,这里使用lombok中定义的注解。

  1. 然后定义对User操作的Mapper,如下:
public interface UserRepository extends BaseMapper<User> {}

这样我们就可以对User进行增删改查操作了。

4. 总结

在我们了解如何使用Provider定义SQL时,编写一个单表自动生成的组件就比较容易了。对于跨表操作我们还是老老实实写SQL吧,或者使用Hibernate。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 210,978评论 6 490
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 89,954评论 2 384
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 156,623评论 0 345
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,324评论 1 282
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,390评论 5 384
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,741评论 1 289
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,892评论 3 405
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,655评论 0 266
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,104评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,451评论 2 325
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,569评论 1 340
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,254评论 4 328
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,834评论 3 312
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,725评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,950评论 1 264
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,260评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,446评论 2 348

推荐阅读更多精彩内容