JAVA 下唯一一款搞定 OLTP+OLAP 的强类型查询这就是最好用的 ORM 相见恨晚
JAVA下唯一一款搞定OLTP+OLAP的强类型查询这就是最好用的ORM相见恨晚
介绍
首先非常感谢 FreeSQL 提供的部分源码,让我借鉴了不少功能点,整体设计并没有参考FreeSQL
(因为java压根没有expression所以没办法参考)只是在数据库方言上FreeSQL
提供的SQL让我少走了很多弯路,所以才让easy-query
可以走的这么迅速
丑话说在前头,这是java下面唯一一款可以完全替代SQL
的强类型ORM,完美支持OLTP和OLAP语法筛选记住是唯一一款
想体验完整版请查看文档博客篇幅有限见谅本次仅展示OLTP的对象关联查询
easy-query
文档地址 https://xuejmnet.github.io/easy-query-doc/ (为什么没有gitee的文档因为gitee pages挂掉了目前没办法更新)
GITHUB地址 https://github.com/xuejmnet/easy-query
GITEE地址 https://gitee.com/xuejm/easy-query
java下面唯一一款支持强类型OLTP和OLAP语法并且支持分表分库的最好用的ORM,为什么是最好用的OLTP那么我们先来看一个简单的例子
- 用户、角色、菜单典型的多对多关联关系(隐式子查询)
- 其中用户和用户所在地址为一对一关系(隐式join)
@Table("t_user") @Data @EntityProxy public class SysUser implements ProxyEntityAvailable<SysUser , SysUserProxy> { @Column(primaryKey = true) private String id; private String name; private LocalDateTime createTime; @Navigate(value = RelationTypeEnum.ManyToMany, mappingClass = UserRole.class, selfMappingProperty = "userId", targetMappingProperty = "roleId") private List<SysRole> roles; @Navigate(value = RelationTypeEnum.OneToOne,targetProperty = "userId") private SysUserAddress address; @Override public Class<SysUserProxy> proxyTableClass() { return SysUserProxy.class; } } @Table("t_role") @Data @EntityProxy public class SysRole implements ProxyEntityAvailable<SysRole, SysRoleProxy> { @Column(primaryKey = true) private String id; private String name; private LocalDateTime createTime; @Navigate(value = RelationTypeEnum.ManyToMany, mappingClass = UserRole.class, selfMappingProperty = "roleId", targetMappingProperty = "userId") private List<SysUser> users; @Navigate(value = RelationTypeEnum.ManyToMany, mappingClass = RoleMenu.class, selfMappingProperty = "roleId", targetMappingProperty = "menuId") private List<SysMenu> menus; @Override public Class<SysRoleProxy> proxyTableClass() { return SysRoleProxy.class; } } @Table("t_user_role") @Data @EntityProxy public class UserRole implements ProxyEntityAvailable<UserRole , UserRoleProxy> { @Column(primaryKey = true) private String id; private String userId; private String roleId; @Override public Class<UserRoleProxy> proxyTableClass() { return UserRoleProxy.class; } } @Table("t_menu") @Data @EntityProxy public class SysMenu implements ProxyEntityAvailable<SysMenu , SysMenuProxy> { @Column(primaryKey = true) private String id; private String name; private String route; private String icon; @Navigate(value = RelationTypeEnum.ManyToMany, mappingClass = RoleMenu.class, selfMappingProperty = "menuId", targetMappingProperty = "roleId") private List<SysRole> roles; @Override public Class<SysMenuProxy> proxyTableClass() { return SysMenuProxy.class; } } @Table("t_role_menu") @Data @EntityProxy public class RoleMenu implements ProxyEntityAvailable<RoleMenu , RoleMenuProxy> { @Column(primaryKey = true) private String id; private String roleId; private String menuId; @Override public Class<RoleMenuProxy> proxyTableClass() { return RoleMenuProxy.class; } } @Table("t_user_address") @Data @EntityProxy public class SysUserAddress implements ProxyEntityAvailable<SysUserAddress , SysUserAddressProxy> { @Column(primaryKey = true) private String id; private String userId; private String province; private String city; private String area; private String addr; @Override public Class<SysUserAddressProxy> proxyTableClass() { return SysUserAddressProxy.class; } }
对应关系为用户和角色是多对多,角色和菜单也是多对多
案例1
查询杭州或绍兴的用户
List<SysUser> userInHz = easyEntityQuery.queryable(SysUser.class) .where(s -> { //隐式子查询会自动join用户表和地址表 s.or(()->{ s.address().city().eq("杭州市"); s.address().city().eq("绍兴市"); }); }).toList(); SELECT t.`id`, t.`name`, t.`create_time` FROM `t_user` t LEFT JOIN `t_user_address` t1 ON t1.`user_id` = t.`id` WHERE ( t1.`city` = '杭州市' OR t1.`city` = '绍兴市' )
查询用户叫做小明的返回小明的姓名和小明所在地址
List<Draft2<String, String>> userNameAndAddr = easyEntityQuery.queryable(SysUser.class) .where(s -> { s.name().eq("小明"); }).select(s -> Select.DRAFT.of( s.name(), s.address().addr()//隐式join因为用户返回了地址标的地址信息 )).toList(); SELECT t.`name` AS `value1`, t1.`addr` AS `value2` FROM `t_user` t LEFT JOIN `t_user_address` t1 ON t1.`user_id` = t.`id` WHERE t.`name` = '小明'
案例2
查询用户下面存在角色是收货员
的用户
List<SysUser> 收货员 = easyEntityQuery.queryable(SysUser.class) .where(s -> { s.roles().where(role -> { role.name().eq("收货员"); }).any(); }).toList(); SELECT t.`id`, t.`name`, t.`create_time` FROM `t_user` t WHERE EXISTS ( SELECT 1 FROM `t_role` t1 WHERE EXISTS ( SELECT 1 FROM `t_user_role` t2 WHERE t2.`role_id` = t1.`id` AND t2.`user_id` = t.`id` LIMIT 1 ) AND t1.`name` = '收货员' LIMIT 1 )
案例3
查询用户下面存在角色是XX员
,并且存在个数大于5个的用户,就是说需要满足用户下面的角色是xx员
的起码有5个及以上的
List<SysUser> 收货员 = easyEntityQuery.queryable(SysUser.class) .where(s -> { //筛选条件为角色集合里面有角色名称叫做xx员的 s.roles().where(role -> { role.name().likeMatchRight("员"); }).count().gt(5L);//count数量大于5个 }).toList(); -- 第1条sql数据 SELECT t.`id`, t.`name`, t.`create_time` FROM `t_user` t WHERE ( SELECT COUNT(*) FROM `t_role` t1 WHERE EXISTS ( SELECT 1 FROM `t_user_role` t2 WHERE t2.`role_id` = t1.`id` AND t2.`user_id` = t.`id` LIMIT 1 ) AND t1.`name` LIKE '%员' ) > 5
案例4
查询用户下面存在的任意角色不大于2022年创建的
LocalDateTime localDateTime = LocalDateTime.of(2022, 1, 1, 0, 0); List<SysUser> 收货员 = easyEntityQuery.queryable(SysUser.class) .where(s -> { //筛选条件为角色集合里面有角色最大时间不能大于2022年的 s.roles().max(role -> role.createTime()).lt(localDateTime); }).toList(); SELECT t.`id`, t.`name`, t.`create_time` FROM `t_user` t WHERE ( SELECT MAX(t1.`create_time`) FROM `t_role` t1 WHERE EXISTS ( SELECT 1 FROM `t_user_role` t2 WHERE t2.`role_id` = t1.`id` AND t2.`user_id` = t.`id` LIMIT 1 ) ) < '2022-01-01 00:00'
案例5
查询每个用户和前3个最早创建的角色(支持分页)适用于评论和评论子表前N个
List<SysUser> 收货员 = easyEntityQuery.queryable(SysUser.class) //前面的表达式表示要返回roles后面的表示如何返回返回按时间正序的3个 .includes(s -> s.roles(),x->{ x.orderBy(r->r.createTime().asc()).limit(3); }) .toList();
案例6
查询用户小明下面的菜单
//方式1多次查询 List<SysMenu> menus = easyEntityQuery.queryable(SysUser.class) .where(s -> { s.name().eq("小明"); }) .toList(x -> x.roles().flatElement().menus().flatElement()); //方式2一次次查询 List<SysMenu> menus = easyEntityQuery.queryable(SysMenu.class) .where(s -> { //判断菜单下的角色存在角色的用户叫做小明的 s.roles().any(role -> { role.users().any(user -> { user.name().eq("小明"); }); }); }).toList(); -- 第1条sql数据 SELECT t.`id`, t.`name`, t.`route`, t.`icon` FROM `t_menu` t WHERE EXISTS ( SELECT 1 FROM `t_role` t1 WHERE EXISTS ( SELECT 1 FROM `t_role_menu` t2 WHERE t2.`role_id` = t1.`id` AND t2.`menu_id` = t.`id` LIMIT 1 ) AND EXISTS ( SELECT 1 FROM `t_user` t3 WHERE EXISTS ( SELECT 1 FROM `t_user_role` t4 WHERE t4.`user_id` = t3.`id` AND t4.`role_id` = t1.`id` LIMIT 1 ) AND t3.`name` = '小明' LIMIT 1 ) LIMIT 1 )
最后
这边展示了非常强大的OLTP查询模式,OLAP也是非常强大可以group+join,实现from (匿名sql) 也可以join (匿名sql)
一款具有强类型OLTP+OLAP的完美解决方案,并且完美支持mybatis系列的任意架构逐步构建迁移,不会产生任何冲突,因为easy-query本身就是零依赖,并且完全免费,完全开源(包括文档!!!包括文档!!!包括文档!!!)
我相信easy-query
是一款可以完完全全打动您的ORM作品,也是全java唯一一款全sql替代性产品

低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。
持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。
- 上一篇
JS 网页全自动翻译 v3.3 发布,源码级翻译速度大幅提升
两行 js 实现 html 全自动翻译。 无需改动页面、无语言配置文件、无 API Key、对 SEO 友好! 升级说明 translate.js 增加 select语种切换自定义 translate.selectLanguageTag.customUI 可以根据自己喜好自定义语种切换的ui translate.api 增加线程池统一调控,html翻译速度提升一倍。 translate.admin 的翻译控制台自定义翻译指定的sitemap.xml,增加顺序读取,执行翻译的url顺序跟sitemap.xml中url出现的顺序保持一致,以便针对十万级、百万级网址数据批量翻译时,如果中间出现意外中断,还可以手动通过翻译日志找到翻译终端的位置,人为指定开始位置。 translate.admin 的翻译控制台-翻译任务排队情况,如果当前任务中,有我网站的任务,那么就会出现一个结束任务的按钮,会从我自己的翻译任务从总的任务池中移除。如果我的任务正在执行中,则会立即终止执行,并从任务池中移除。 translate.api 优化线程池,如果上次执行异常了,那么直接设置当前线程池中线程进行标记,结束当...
- 下一篇
【小白课程】openKylin 系统监视器新功能介绍
openKylin系统监视器是一款面向openKylin操作系统用户的桌面应用,它满足了用户对于各种系统资源的监控和管理。在前面章节中,我们为大家讲解介绍了openKylin系统监视器的功能作用,包括处理器、交换空间、网络、磁盘、进程和服务等。今天我们就来讲一讲它即将在openKylin 2.0版本上新的那些好用功能。 一、最小化到托盘显示 以前大家点击右上角的“X”按钮,系统监视器将直接退出;现在UKUI SIG增加了最小化到托盘显示的功能,点击右上角的“X”按钮会在任务栏显示一个系统监视器图标,将鼠标悬浮上去可显示CPU占用率、网络上传与下载速率、内存占用率、磁盘占用率。 鼠标右键菜单可退出系统监视器或激活系统监视器窗口,鼠标左键单击激活系统监视器窗口。 二、处理器、内存、网络历史等详情页显示 openKylin系统监视器新增处理器、内存、网络历史三个模块的详情显示界面,可以通过点击主界面左侧的处理器、内存、网络历史模块,在主界面右边进行详情显示,并可以通过点击右上角的“隐藏详情”按钮返回进程页面。 新增处理器详情页展示处理器总利用率、当前频率、频率、插槽、逻辑处理器、虚拟化、L1...
相关文章
文章评论
共有0条评论来说两句吧...
文章二维码
点击排行
推荐阅读
最新文章
- Docker使用Oracle官方镜像安装(12C,18C,19C)
- Docker安装Oracle12C,快速搭建Oracle学习环境
- CentOS7安装Docker,走上虚拟化容器引擎之路
- Linux系统CentOS6、CentOS7手动修改IP地址
- Docker快速安装Oracle11G,搭建oracle11g学习环境
- CentOS7编译安装Cmake3.16.3,解决mysql等软件编译问题
- CentOS关闭SELinux安全模块
- CentOS7编译安装Gcc9.2.0,解决mysql等软件编译问题
- Hadoop3单机部署,实现最简伪集群
- CentOS6,7,8上安装Nginx,支持https2.0的开启