现在公司使用mybatis作为DAL层的框架.

使用起来比较简单,使用xml进行SQL的书写,java代码使用接口执行.

但在写一些简单SQL的时候会显得非常繁琐:

  1. xml和java分离(设计上为了解耦),一些字段是否设置等需要反复查看(虽然可以通过插件直达);
  2. 原生无法热加载,修改xml后需要重启(可以使用三方实现);
  3. xml的动态SQL没有java灵活.

上述这些"缺点"仅仅针对写简单的sql,特别是第一条.

我对一张表的需求是简单的CRUD,那使用xml就会非常麻烦.

最基本的需求:

  • 简单安全的SQL拼接方案;
  • 方便的类型映射.

探索了几个方案,接下来一一介绍.

JOOQ

这是考虑的第一个方案,有个没好的愿景,类型安全的SQL,不用担心出现SQL的语法错误.

但实际使用遇到了比较麻烦的事情,他的codegen.

        <generate>
<pojos>true</pojos>
<daos>true</daos>
<keys>false</keys>
<indexes>false</indexes>
<globalObjectReferences>false</globalObjectReferences>
</generate>

自己加了pojodao,一定会生成Table,Record,洋洋洒洒一堆文件摆在眼前.

因为是互联网企业,业务也在发现会相当不稳定,管理这坨生成的代码会很麻烦,而且不简洁.

当然也有不是用codegen的方式,但最后感觉还是过于庞大,也比之前直接用mybatis更麻烦了.

ActiveJDBC

active record的实践.

总体感觉非常简洁清晰,想起了当初用JFinal的感觉.

当时看了文档就感觉是他了,不需要自己手写SQL,同时兼容多种方言,单测容易用H2替换运行.

但实际使用发现了一个严峻的问题,严峻到不得不放弃他的问题... ...

他需要Instrument的支持.(子类使用父类的静态方法但是要在字节码中改为子类调用(复制了那些方法到子类... ...))

且无论是动态方案还是静态方案都需要修改IDE配置.

这意味着我只要在现有项目里使用就会有一群开发过来揍我

Mybatis SqlBuilder&@SelectProvider

绕了一圈,回到了原点... ...

但这个方案的确是满足了解决了上面3个问题.

org.apache.ibatis.jdbc.SQL本身提供了几个方法方便拼写简单的SQL.

例子如下:

new SQL()
.SELECT(COLUMNS)
.FROM("user AS u")
.INNER_JOIN("user_info AS ui ON u.user_id = ui.user_id")
.WHERE("u.user_type = #{userType}")
.WHERE("u.deleted = 0")
.ORDER("u.user_id")
.toString()
+ " LIMIT #{start},#{num}";

LIMIT要自己拼,因为他不是标准SQL.

接下来接口一样定义,用上对应注解(@UpdateProvider,@InsertProvider,@DeleteProvider)

    @SelectProvider(type = UserProvider.class, method = "getUserByType")
UserInfo getUserByType(@Param("userType") int userType);

JdbcTemplate

使用NamedParameterJdbcTemplate会感觉好一点,占位符可以用:name的形式.

我们spring也一起用,对应的依赖也有.

这个也可行,而且写起来还可以,不过他本身缺少SQL拼接方便的工具,适合写简单没有使用条件分支生成SQL的情况.

这个网上资料很多,这里稍微列一些常用的需求.

根据class自动映射

可以使用BeanPropertyRowMapper,会按照属性进行SQL字段的关联,java的驼峰会变成小写和下划线的形式进行匹配.

例子:

private static final BeanPropertyRowMapper<User> USER_MAPPER = BeanPropertyRowMapper.newInstance(User.class);  

... ...

MapSqlParameterSource params = new MapSqlParameterSource();
params.addValue("userId", userId);
List<ServiceApply> applies = mTemplate.query(
"SELECT user_id,user_name,created FROM user WHERE user_id=:userId",
params, USER_MAPPER);

获取插入ID

KeyHolder key = new GeneratedKeyHolder();
mTemplate.update(
"INSERT INTO user(user_name,age) VALUES (:name,:age)",
params, key);
Number keyValue = key.getKey();

批量插入

List<MapSqlParameterSource> batchParams = names
.stream()
.distinct()
.map(e -> new MapSqlParameterSource("name", name))
.collect(Collectors.toList());
mTemplate.batchUpdate(
"INSERT IGNORE INTO user(user_name) VALUES (:name)",
batchParams.toArray(new SqlParameterSource[0]));

参考链接:

https://github.com/shakarelic/mybatis-reload

https://www.jooq.org/

http://javalite.io/activejdbc

https://zh.wikipedia.org/wiki/Active_Record

http://www.mybatis.org/mybatis-3/statement-builders.html

https://docs.spring.io/spring/docs/current/javadoc-api/org/springframework/jdbc/core/namedparam/NamedParameterJdbcTemplate.html

如何更简单方便地执行SQL操作?的更多相关文章

  1. Mybatis-plus 实体类新增属性,使用实体类执行sql操作时忽略该属性 注解

    @TableField(exist = false) 注解加载bean属性上,表示当前属性不是数据库的字段,但在项目中必须使用,这样在新增等使用bean的时候,mybatis-plus就会忽略这个,不 ...

  2. EF Core中执行Sql语句查询操作之FromSql,ExecuteSqlCommand,SqlQuery

    一.目前EF Core的版本为V2.1 相比较EF Core v1.0 目前已经增加了不少功能. EF Core除了常用的增删改模型操作,Sql语句在不少项目中是不能避免的. 在EF Core中上下文 ...

  3. 三种执行SQL语句的的JAVA代码

    问题描述: 连接数据库,执行SQL语句是必不可少的,下面给出了三种执行不通SQL语句的方法. 1.简单的Statement执行SQL语句.有SQL注入,一般不使用. public static voi ...

  4. MySQL执行sql查询并上传至远程服务器

    最近项目中有需要做一个shell脚本,可以对一个数据库执行sql操作,并将结果转为txt,筛选结果用tab隔开,保存至一个远程服务器上,以供其他人用Excel读取用txt中的内容. MySQL中将结果 ...

  5. 使用Hive或Impala执行SQL语句,对存储在HBase中的数据操作

    CSSDesk body { background-color: #2574b0; } /*! zybuluo */ article,aside,details,figcaption,figure,f ...

  6. 使用Hive或Impala执行SQL语句,对存储在Elasticsearch中的数据操作(二)

    CSSDesk body { background-color: #2574b0; } /*! zybuluo */ article,aside,details,figcaption,figure,f ...

  7. 使用Hive或Impala执行SQL语句,对存储在Elasticsearch中的数据操作

    http://www.cnblogs.com/wgp13x/p/4934521.html 内容一样,样式好的版本. 使用Hive或Impala执行SQL语句,对存储在Elasticsearch中的数据 ...

  8. 追踪SQL Server执行delete操作时候不同锁申请与释放的过程

    一直以为很了解sqlserver的加锁过程,在分析一些特殊情况下的死锁之后,尤其是并发单表操作发生的死锁,对于加解锁的过程,有了一些重新的认识,之前的知识还是有一些盲区在里面的.delete加锁与解锁 ...

  9. C#参数化执行SQL语句,防止漏洞攻击本文以MySql为例【20151108非查询操作】

    为什么要参数化执行SQL语句呢? 一个作用就是可以防止用户注入漏洞. 简单举个列子吧. 比如账号密码登入,如果不用参数, 写的简单点吧,就写从数据库查找到id和pw与用户输入一样的数据吧 sql:se ...

随机推荐

  1. 2019 蓝桥杯省赛 A 组模拟赛(一)-忽明忽暗

    走廊里有 nn 盏灯,编号依次为 1,2,3,...,n,由学校电路控制中心管理.初始时,所有灯都是关闭的.某黑客入侵了学校电路控制中心,黑客想让灯忽明忽暗,进行了 n 轮操作.第 i 轮操作,会让所 ...

  2. tp5 mkdir(): Permission denied 问题

    今天使用tp5 线上上传图片的时候遇到了一个问题 mkdir(): Permission denied 如图 百度了一下 发现大家都说 chmod -R 777 runtime 能解决问题 尝试了一下 ...

  3. jQuery 动态绑定插件livequery的用法

  4. JS获取键盘事件

    <script type="text/javascript" language=JavaScript charset="UTF-8"> docume ...

  5. hashlib模块,shutil,模块 ,,xml 文件解析,configparser,模块,类,什么是类

    1 什么是hash hash是一种算法,该算法接受传入的内容,经过运算得到一串hash值 如果把hash算法比喻为一座工厂 那传给hash算法的内容就是原材料 生成的hash值就是生产出的产品 2.为 ...

  6. mysql8.0.13修改密码

    在安装完数据库后,由于自己不小心直接关闭了安装窗口,或者长时间没有使用root用户登录系统,导致忘记了root密码,这时就需要重置MySQL的root密码.当然,最简单方式自然是删除数据库的data目 ...

  7. Could not resolve placeholder 'IMAGE_SERVER_URL' in string value "${IMAGE_SERVER_URL}"

    这种问题 在网上查的是说使用了重复的property-placeholder   可能是在别的xml 也用了property-placeholder 解决方法 加上  ignore-unresolva ...

  8. day11函数(形参实参)

    形参与实参 def fn(形参们): pass # 形参:定义函数,在括号内声明的变量名,用来结束外界传来的值# 实参:调用函数,在括号内传入的实际值,值可以为常量.变量.表达式或三者的组合 # 注: ...

  9. angular-ui-bootstrap typeahead 智能提示 自动补全 获取焦点不触发问题的解决

    项目中有一处使用了angular-ui-bootstrap中的typeahead来实现输入框智能提示语自动化补全的功能,存在一个bug, 即输入文字后,当再次点击文本框,其获取焦点后并不会触发智能提示 ...

  10. es5中的for in 与es6中的for of的用法与区别

    for in 用与循环遍历对象中的属性键值 for of用于循环遍历出数组中的属性值 for in 也可以遍历数组,但是局限是他会把数组的其他属性键值也会遍历出,例如给数组添加一个属性arr.name ...