这么优雅的Java ORM没见过吧!
Java的ORM框架有很多,但由于Java语言的限制大部分都不够优雅也不够简单,所以作者只能另辟蹊径造轮子了。照旧先看示例代码了解个大概,然后再解释实现原理。
一、ORM示例
1. Insert
public CompletableFuture<Void> insert() {
var obj = new sys.entities.Demo("MyName"); //构造参数为主键
obj.Age = 100; //设置实体属性的值
return obj.saveAsync();
}
2. Update
- 更新单个实体(必须具备主键)
public CompletableFuture<Void> update(sys.entities.Demo obj) {
obj.Age = 200;
return obj.saveAsync();
}
- 根据条件更新(必须指定条件以防误操作)
public CompletableFuture<?> update() {
var cmd = new SqlUpdateCommand<sys.entities.Demo>();
cmd.update(e -> e.City = "Wuxi"); //更新字段
cmd.update(e -> e.Age = e.Age + 1); //更新累加字段
cmd.where(e -> e.Name == "Johne"); //更新的条件
var outs = cmd.output(e -> e.Age); //更新的同时返回指定字段
return cmd.execAsync().thenApply(rows -> {
System.out.println("更新记录数: " + rows);
System.out.println("返回的值: " + outs.get(0));
return "Done.";
});
}
3. Delete
- 删除单个实体(必须具备主键)
public CompletableFuture<Void> update(sys.entities.Demo obj) {
obj.markDeleted(); //先标记为删除状态
return obj.saveAsync(); //再调用保存方法
}
- 根据条件删除(必须指定条件以防误操作)
public CompletableFuture<?> delete() {
var cmd = new SqlDeleteCommand<sys.entities.Demo>();
cmd.where(e -> e.Age < 0 || e.Age > 200);
return cmd.execAsync();
}
4. Transaction
由于作者讨厌隐式事务,所以事务命令必须显式指定。
public CompletableFuture<?> transaction() {
var obj1 = new sys.entities.Demo("Demo1");
obj1.Age = 11;
var obj2 = new sys.entities.Demo("Demo2");
obj2.Age = 22;
return DataStore.DemoDB.beginTransaction().thenCompose(txn -> { //开始事务
return obj1.saveAsync(txn) //事务保存obj1
.thenCompose(r -> obj2.saveAsync(txn)) //事务保存obj2
.thenCompose(r -> txn.commitAsync()); //递交事务
}).thenApply(r -> "Done");
}
5. Sql查询
- Where条件
public CompletableFuture<?> query(String key) {
var q = new SqlQuery<sys.entities.Demo>();
q.where(e -> e.Age > 10 && e.Age < 80);
if (key != null)
q.andWhere(e -> e.Name.contains(key)); //拼接条件
return q.toListAsync(); //返回List<sys.entities.Demo>
}
- 分页查询
public CompletableFuture<?> query(int pageSize, int pageIndex) {
var q = new SqlQuery<sys.entities.Demo>();
return q.skip(pageSize * pageIndex)
.take(pageSize)
.toListAsync();
}
- 结果映射至匿名类
public CompletableFuture<?> query() {
var q = new SqlQuery<sys.entities.Demo>();
return q.toListAsync(e -> new Object() { //返回List<匿名类>
public final String Name = e.Name; //匿名类属性 = 实体属性表达式
public final int Age = e.Age + 10;
public final String Father = e.Parent.Name;
}).thenApply(appbox.data.JsonResult::new);
}
- 结果映射至继承的匿名类
public CompletableFuture<?> query() {
var q = new SqlQuery<sys.entities.Demo>();
q.where(e -> e.Parent.Name == "Rick");
return q.toListAsync(e -> new sys.entities.Demo() { //返回List<? extens Demo>
public final String Father = e.Parent.Name;
});
}
- 结果映射至树状结构列表
public CompletableFuture<?> tree() {
var q = new SqlQuery<sys.entities.Demo>();
q.where(t -> t.Name == "Rick");
return q.toTreeAsync(t -> t.Childs); //参数指向EntitySet(一对多成员)
}
- EntityRef(一对一引用的实体成员)自动Join
public CompletableFuture<?> query() {
var q = new SqlQuery<sys.entities.Customer>();
q.where(cus -> cus.City.Name == "Wuxi");
return q.toListAsync();
}
生成的Sql:
Select t.* From "Customer" t Left Join "City" j1 On j1."Code"=t."CityCode"
- 手工指定Join
public CompletableFuture<?> join() {
var q = new SqlQuery<sys.entities.Customer>();
var j = new SqlQueryJoin<sys.entities.City>();
q.leftJoin(j, (cus, city) -> cus.CityCode == city.Code);
q.where(j, (cus, city) -> city.Name == "Wuxi");
return q.toListAsync();
}
- 子查询
public CompletableFuture<?> subQuery() {
var sq = new SqlQuery<sys.entities.Demo>();
sq.where(s -> s.ParentName == "Rick");
var q = new SqlQuery<sys.entities.Demo>();
q.where(t -> DbFunc.in(t.Name, sq.toSubQuery(s -> s.Name)));
return q.toListAsync();
}
- GroupBy
public CompletableFuture<?> groupBy() {
var q = new SqlQuery<sys.entities.Demo>();
q.groupBy(t -> t.ParentName) //多个可重复
.having(t -> DbFunc.sum(t.Age) > 10);
return q.toListAsync(t -> new Object() {
public final String group = t.ParentName == null ? "可怜的孩子" : t.ParentName;
public final int totals = DbFunc.sum(t.Age);
}).thenApply(appbox.data.JsonResult::new);
}
二、实现原理
其实以上的示例代码并非最终运行的代码,作者利用Eclipse jdt将上述代码在编译发布服务模型时分析转换为最终的运行代码,具体过程如下:
1. jdt分析服务虚拟代码生成AST抽象语法树;
2. 遍历AST树,将实体对象的读写属性改写为getXXX(), setXXX();
var name = obj.Name; //读实体属性
obj.Name = "Rick"; //写实体属性
改写为:
var name = obj.getName();
obj.setName("Rick");
3. 遍历AST树,将查询相关方法的参数转换为运行时表达式;
public CompletableFuture<?> query(String key) {
var q = new SqlQuery<sys.entities.Employee>();
q.where(e -> e.Manager.Name + "a" == key + "b");
return q.toListAsync();
}
转换为:
public CompletableFuture<?> query(String key) {
var q = new appbox.store.query.SqlQuery<>(-7018111290459553788L, SYS_Employee.class);
q.where(e -> e.m("Name").plus(obj.getName()).plus("c").eq("Rick"));
return q.toListAsync();
}
4. 根据服务模型使用到的实体模型生成相应实体的运行时代码;
5. 最后编译打包服务模型的字节码。
以上请参考源码的ServiceCodeGenerator及EntityCodeGenerator类。
三、性能与小结
作者写了个简单查询的服务,测试配置为MacBook主机(wrk压测 + 数据库)->4核I7虚拟机(服务端),测试结果如下所示qps可达1万,已包括实体映射转换及序列化传输等所有开销。这里顺便提一下,由于框架是全异步的,所以没有使用传统的JDBC驱动,而是使用了jasync-sql(底层为Netty)来驱动数据库。
wrk -c200 -t2 -d20s -s post_bin.lua http://10.211.55.8:8000/api
Running 20s test @ http://10.211.55.8:8000/api
2 threads and 200 connections
Thread Stats Avg Stdev Max +/- Stdev
Latency 18.97ms 5.84ms 89.15ms 81.55%
Req/Sec 5.32k 581.92 6.48k 65.00%
211812 requests in 20.02s, 36.76MB read
Requests/sec: 10578.90
Transfer/sec: 1.84MB
边码代码边码文实属不易,作者需要您的支持请您多多点赞推荐!另欢迎感兴趣的小伙伴加入我们!
这么优雅的Java ORM没见过吧!的更多相关文章
- 这是你没见过的不一样的redis
转: 这是你没见过的不一样的redis 提到Redis,大家一定会想到的几个点是什么呢? 高并发,KV存储,内存数据库,丰富的数据结构,单线程(6版本之前) 那么,接下来,上面提到的这些,都会一一给大 ...
- 超赞!12套你没见过的社交媒体 & 社交网站图标
如今,社交网络成为我们信息获取和传播的重要途径,很多网站都有把内容分享到社交媒体的功能.社交媒体图标作为向用户传递信息的重要媒介,不管是在网页还是 Web 应用程序中都非常需要.今天这篇文章和大家分享 ...
- 【原创】怎样才能写出优雅的 Java 代码?这篇文章告诉你答案!
本文已经收录自 JavaGuide (59k+ Star):[Java学习+面试指南] 一份涵盖大部分Java程序员所需要掌握的核心知识. 本文比较简短,基本就是推荐一些对于写好代码非常有用的文章或者 ...
- 这份java多线程笔记,你真得好好看看,我还没见过总结的这么全面的
1.线程,进程和多线程 1.程序:指指令和数据的有序集合,其本身没有任何意义,是一个静态的概念 2.进程:指执行程序的一次执行过程,是一个动态的概念.是系统资源分配的单位(注意:很多多线程是模拟出来的 ...
- StackOverflow: 你没见过的七个最好的Java答案
StackOverflow发展到目前,已经成为了全球开发者的金矿.它能够帮助我们找到在各个领域遇到的问题的最有用的解决方案,同时我们也会从中学习到很多新的东西.这篇文章是在我们审阅了StackOver ...
- java中三种常见内存溢出错误的处理方法
更多 10 相信有一定java开发经验的人或多或少都会遇到OutOfMemoryError的问题,这个问题曾困扰了我很长时间,随着解决各类问题经验的积累以及对问题根源的探索,终于有了一个比较深入的 ...
- Qt国际化相关类(以前没见过codec->toUnicode,QTextCodec,QLocale.toString和QLocale::setDefault,QInputMethod::locale())
QTextCodec QTextCodec为文本编码之间提供转换. Qt用Unicode 来存储,绘制和操作字符串.在很多情况下你可能希望操作不同编码的数据.例如,大部分日本文档是以Shift-JIS ...
- java中三种常见内存溢出错误的处理方法(good)
相信有一定java开发经验的人或多或少都会遇到OutOfMemoryError的问题,这个问题曾困扰了我很长时间,随着解决各类问题经验的积累以及对问题根源的探索,终于有了一个比较深入的认识. 在解决j ...
- Java ORM Hibernate 入门笔记
一.下载 官网地址:http://hibernate.org/ Hibernate下有ORM(关系型数据库).OGM(NoSQL数据库).Search(对象全文检索).Validator的工具. OR ...
随机推荐
- 小程序view的显示与隐藏
需要在全局数据块中,设定一个控制键. data: { ......//省略其他代码 showView: true }, 然后是在wxml中,view的class中设置2个class,并用三目表达式来进 ...
- 世界上最快的排序算法——Timsort
前言 经过60多年的发展,科学家和工程师们发明了很多排序算法,有基本的插入算法,也有相对高效的归并排序算法等,他们各有各的特点,比如归并排序性能稳定.堆排序空间消耗小等等.但是这些算法也有自己的局限性 ...
- Tensorflow学习笔记No.11
图像定位 图像定位是指在图像中将我们需要识别的部分使用定位框进行定位标记,本次主要讲述如何使用tensorflow2.0实现简单的图像定位任务. 我所使用的定位方法是训练神经网络使它输出定位框的四个顶 ...
- 「 洛谷 」P2768 珍珠项链
珍珠项链 题目限制 内存限制:125.00MB 时间限制:1.00s 标准输入输出 题目知识点 动态规划 \(dp\) 矩阵 矩阵乘法 矩阵加速 矩阵快速幂 题目来源 「 洛谷 」P2768 珍珠项链 ...
- JQuery统一复写美化项目中所有radio单选按钮样式
老项目要升级改版,对于分散在各页面的样式不好处理,怕有遗漏,尤其是优化input表单,修改其默认样式,接下来,我将给大家分享一下,我在项目中的总结. 效果 上代码: 1.简单搞一搞 CSS,此处代码有 ...
- sqli-labs less32-37(宽字节注入)
less-32 Bypass addslashes() less-33 Bypass addslashes() less-34 Bypass Add SLASHES less-35 addslashe ...
- Day8 python高级特性-- 迭代 Iteration
通过for循环来遍历 list.tuple.dict.甚至是字符串,这种遍历被称为迭代. 相比于C的for循环, Python的for循环抽象成都更好,不仅可以用在list或tuple上,还可以用在其 ...
- Hive通过Jdbc获取表的字段信息
参考代码如下: /** * 按顺序返回字段 * desc table的返回结果形式如下: hive> describe ind01acoM; OK acq_ins_id_cd string cu ...
- html 05-HTML标签图文详解(二)
05-HTML标签图文详解(二) #本文主要内容 列表标签:<ul>.<ol>.<dl> 表格标签:<table> 框架标签及内嵌框架<ifram ...
- [日常摸鱼]bzoj2823 [AHOI2012]信号塔
题意:$n$个点,求最小圆覆盖,$n \leq 5e5$ 这题数据是随机的hhh 我们可以先求出凸包然后对凸包上的点求最小圆覆盖-(不过直接求应该也行?) 反正随便写好像都能过- #include&l ...