再见 MybatisPlus,阿里推出新 ORM 框架更牛X

最近看到一个 ORM 框架 Fluent Mybatis 挺有意思的,整个设计理念非常符合工程师思维。

我对官方文档的部分内容进行了简单整理,通过这篇文章带你看看这个新晋 ORM 框架。

官方文档:https://gitee.com/fluent-mybatis/fluent-mybatis/wikis

提前声明一下:对于这类个人维护和开发的框架,如果没有充分的了解,一定一定一定不要用在正式的项目上!不然后续遇到问题会很麻烦的!!!我目前对于 Fluent Mybatis 这个框架也仅仅是感兴趣,想要学习一下它的内部设计。

# Fluent Mybatis 介绍

何为 Fluent Mybatis?

Fluent Mybatis, 是一款 Mybatis 语法增强框架, 综合了 Mybatis Plus, Dynamic SQL, JPA 等框架特性和优点, 利用 annotation processor 生成代码。

Fluent Mybatis 有什么亮点?

使用 Fluent Mybatis 可以不用写具体的 XML 文件,通过 Java API 可以构造出比较复杂的业务 SQL 语句,做到代码逻辑和 SQL 逻辑的合一。不再需要在 Dao 中组装查询或更新操作,在 XML 或 Mapper 中再组装参数。

项目地址:https://gitee.com/fluent-mybatis/fluent-mybatis

图片

 

# Fluent Mybatis 特性一览

图片

# Fluent Mybatis 原理

图片

# Fluent Mybatis vs Mybatis vs Mybatis Plus

对比原生 Mybatis, Mybatis Plus 或者其他框架,FluentMybatis 提供了哪些便利呢?

一、实现需求比较

我们通过一个比较典型的业务需求来具体实现和对比下,假如有学生成绩表结构如下:

create table `student_score`(    id           bigint auto_increment comment '主键ID' primary key,    student_id   bigint            not null comment '学号',    gender_man   tinyint default 0 not null comment '性别, 0:女; 1:男',    school_term  int               null comment '学期',    subject      varchar(30)       null comment '学科',    score        int               null comment '成绩',    gmt_create   datetime          not null comment '记录创建时间',    gmt_modified datetime          not null comment '记录最后修改时间',    is_deleted   tinyint default 0 not null comment '逻辑删除标识') engine = InnoDB default charset=utf8;

现在有需求: 统计 2000 年三门学科('英语', '数学', '语文')及格分数按学期,学科统计最低分,最高分和平均分, 且样本数需要大于 1 条,统计结果按学期和学科排序

我们可以写 SQL 语句如下:

select school_term,       subject,       count(score) as count,       min(score)   as min_score,       max(score)   as max_score,       avg(score)   as max_scorefrom student_scorewhere school_term >= 2000  and subject in ('英语', '数学', '语文')  and score >= 60  and is_deleted = 0group by school_term, subjecthaving count(score) > 1order by school_term, subject;

那上面的需求,分别用 Fluent Mybatis, 原生 Mybatis 和 Mybatis Plus 来实现一番。

使用 Fluent Mybatis 来实现上面的功能 :

图片

我们可以看到 fluent api 的能力,以及 IDE 对代码的渲染效果。

换成 Mybatis 原生实现上面的功能 :

1、定义 Mapper 接口

public interface MyStudentScoreMapper {    List<Map<String, Object>> summaryScore(SummaryQuery paras);}

2、定义接口需要用到的参数实体 SummaryQuery

 
@Data@Accessors(chain = true)public class SummaryQuery {    private Integer schoolTerm;
    private List<String> subjects;
    private Integer score;
    private Integer minCount;}

3、定义实现业务逻辑的 mapper xml 文件

 
<select id="summaryScore" resultType="map" parameterType="cn.org.fluent.Mybatis.springboot.demo.mapper.SummaryQuery">    select school_term,    subject,    count(score) as count,    min(score) as min_score,    max(score) as max_score,    avg(score) as max_score    from student_score    where school_term >= #{schoolTerm}    and subject in    <foreach collection="subjects" item="item" open="(" close=")" separator=",">        #{item}    </foreach>    and score >= #{score}    and is_deleted = 0    group by school_term, subject    having count(score) > #{minCount}    order by school_term, subject</select>

4、实现业务接口(这里是测试类, 实际应用中应该对应 Dao 类)

 
@RunWith(SpringRunner.class)@SpringBootTest(classes = QuickStartApplication.class)public class MybatisDemo {    @Autowired    private MyStudentScoreMapper mapper;
    @Test    public void Mybatis_demo() {        // 构造查询参数        SummaryQuery paras = new SummaryQuery()            .setSchoolTerm(2000)            .setSubjects(Arrays.asList("英语", "数学", "语文"))            .setScore(60)            .setMinCount(1);
        List<Map<String, Object>> summary = mapper.summaryScore(paras);        System.out.println(summary);    }}

总之,直接使用 Mybatis,实现步骤还是相当的繁琐,效率太低。那换成 Mybatis Plus 的效果怎样呢?

换成 Mybatis Plus 实现上面的功能 :

Mybatis Plus 的实现比 Mybatis 会简单比较多,实现效果如下

图片

如红框圈出的,写 Mybatis Plus 实现用到了比较多字符串的硬编码(可以用 Entity 的 get lambda 方法部分代替字符串编码)。字符串的硬编码,会给开发同学造成不小的使用门槛,个人觉的主要有 2 点:

  1. 字段名称的记忆和敲码困难

  2. Entity 属性跟随数据库字段发生变更后的运行时错误

其他框架,比如 TkMybatis 在封装和易用性上比 Mybatis Plus 要弱,就不再比较了。

二、生成代码编码比较

Fluent Mybatis 生成代码设置 :

public class AppEntityGenerator {    static final String url = "jdbc:mysql://localhost:3306/fluent_Mybatis_demo?useSSL=false&useUnicode=true&characterEncoding=utf-8";
    public static void main(String[] args) {        FileGenerator.build(Abc.class);    }
    @Tables(        /** 数据库连接信息 **/        url = url, username = "root", password = "password",        /** Entity类parent package路径 **/        basePack = "cn.org.fluent.Mybatis.springboot.demo",        /** Entity代码源目录 **/        srcDir = "spring-boot-demo/src/main/java",        /** Dao代码源目录 **/        daoDir = "spring-boot-demo/src/main/java",        /** 如果表定义记录创建,记录修改,逻辑删除字段 **/        gmtCreated = "gmt_create", gmtModified = "gmt_modified", logicDeleted = "is_deleted",        /** 需要生成文件的表 ( 表名称:对应的Entity名称 ) **/        tables = @Table(value = {"student_score"})    )    static class Abc {    }}

Mybatis Plus :

public class CodeGenerator {
    static String dbUrl = "jdbc:mysql://localhost:3306/fluent_Mybatis_demo?useSSL=false&useUnicode=true&characterEncoding=utf-8";
    @Test    public void generateCode() {        GlobalConfig config = new GlobalConfig();        DataSourceConfig dataSourceConfig = new DataSourceConfig();        dataSourceConfig.setDbType(DbType.MYSQL)            .setUrl(dbUrl)            .setUsername("root")            .setPassword("password")            .setDriverName(Driver.class.getName());        StrategyConfig strategyConfig = new StrategyConfig();        strategyConfig            .setCapitalMode(true)            .setEntityLombokModel(false)            .setNaming(NamingStrategy.underline_to_camel)            .setColumnNaming(NamingStrategy.underline_to_camel)            .setEntityTableFieldAnnotationEnable(true)            .setFieldPrefix(new String[]{"test_"})            .setInclude(new String[]{"student_score"})            .setLogicDeleteFieldName("is_deleted")            .setTableFillList(Arrays.asList(                new TableFill("gmt_create", FieldFill.INSERT),                new TableFill("gmt_modified", FieldFill.INSERT_UPDATE)));
        config            .setActiveRecord(false)            .setIdType(IdType.AUTO)            .setOutputDir(System.getProperty("user.dir") + "/src/main/java/")            .setFileOverride(true);
        new AutoGenerator().setGlobalConfig(config)            .setDataSource(dataSourceConfig)            .setStrategy(strategyConfig)            .setPackageInfo(                new PackageConfig()                    .setParent("com.mp.demo")                    .setController("controller")                    .setEntity("entity")            ).execute();    }}

三、三者对比总结

看完 3 个框架对同一个功能点的实现, 各位看官肯定会有自己的判断,笔者这里也总结了一份比较。

图片

# Fluent Mybatis 实战

接下来,我们来看看如何使用 Fluent Mybatis 来实现增删改查。

引入依赖

新建 Maven 工程,设置项目编译级别为 Java8 及以上,引入 Fluent Mybatis 依赖包。

<dependencies>    <!-- 引入fluent-mybatis 运行依赖包, scope为compile -->    <dependency>        <groupId>com.github.atool</groupId>        <artifactId>fluent-mybatis</artifactId>        <version>1.9.3</version>    </dependency>    <!-- 引入fluent-mybatis-processor, scope设置为provider 编译需要,运行时不需要 -->    <dependency>        <groupId>com.github.atool</groupId>        <artifactId>fluent-mybatis-processor</artifactId>        <version>1.9.3</version>    </dependency></dependencies>

创建表

create schema fluent_mybatis;
create table hello_world(    id           bigint unsigned auto_increment primary key,    say_hello    varchar(100) null,    your_name    varchar(100) null,    gmt_created   datetime   DEFAULT NULL COMMENT '创建时间',    gmt_modified datetime   DEFAULT NULL COMMENT '更新时间',    is_deleted   tinyint(2) DEFAULT 0 COMMENT '是否逻辑删除') ENGINE = InnoDB  CHARACTER SET = utf8 comment '简单演示表';

# 创建数据库表对应的 Entity 类

创建数据库表对应的 Entity 类: HelloWorldEntity, 你只需要简单的做 3 个动作:

  1. 根据驼峰命名规则命名 Entity 类和字段

  2. HelloWorldEntity 继承 IEntity 接口类

  3. 在 HelloWorldEntity 类上加注解 @FluentMybatis

@FluentMybatispublic class HelloWorldEntity extends RichEntity {    private Long id;
    private String sayHello;
    private String yourName;
    private Date gmtCreated;
    private Date gmtModified;
    private Boolean isDeleted;
    // get, set, toString 方法
   @Override   public Class<? extends IEntity> entityClass() {      return HelloWorldEntity.class;   }}

执行编译。

IDE 编译:

图片

Maven 编译:mvn clean compile

Gradle 编译:gradle clean compile

# 配置数据源

1.数据源 DataSource 配置

2.Mybatis 的 mapper 扫描路径

3.Mybatis 的 SqlSessionFactoryBean

@ComponentScan(basePackages = "cn.org.atool.fluent.mybatis.demo1")@MapperScan("cn.org.atool.fluent.mybatis.demo1.entity.mapper")@Configurationpublic class HelloWorldConfig {    /**     * 设置dataSource属性     *     * @return     */    @Bean    public DataSource dataSource() {        BasicDataSource dataSource = new BasicDataSource();        dataSource.setDriverClassName("com.mysql.jdbc.Driver");        dataSource.setUrl("jdbc:mysql://localhost:3306/fluent_mybatis?useUnicode=true&characterEncoding=utf8");        dataSource.setUsername("root");        dataSource.setPassword("password");        return dataSource;    }
    /**     * 定义mybatis的SqlSessionFactoryBean     *     * @param dataSource     * @return     */    @Bean    public SqlSessionFactoryBean sqlSessionFactoryBean(DataSource dataSource) {        SqlSessionFactoryBean bean = new SqlSessionFactoryBean();        bean.setDataSource(dataSource);        return bean;    }
   @Bean   public MapperFactory mapperFactory() {      return new MapperFactory();   }}

很简单吧,在这里,你即不需要配置任何 Mybatis xml 文件, 也不需要写任何 · 接口, 但你已经拥有了强大的增删改查的功能,并且是 Fluent API,让我们写一个测试来见证一下 Fluent Mybatis 的魔法力量!

# 测试

注入 HelloWorldEntity 对应的 Mapper 类: HelloWorldMapper, 这个类是 Fluent Mybatis 编译时生成的。

使用 HelloWorldMapper 进行删除、插入、查询、修改操作。

@RunWith(SpringJUnit4ClassRunner.class)@ContextConfiguration(classes = HelloWorldConfig.class)public class HelloWorldTest {    /**     * Fluent Mybatis编译时生成的Mapper类     */    @Autowired    HelloWorldMapper mapper;
    @Test    public void testHelloWorld() {        /**         * 为了演示方便,先删除数据         */        mapper.delete(mapper.query()            .where.id().eq(1L).end());        /**         * 插入数据         */        HelloWorldEntity entity = new HelloWorldEntity();        entity.setId(1L);        entity.setSayHello("hello world");        entity.setYourName("Fluent Mybatis");        entity.setIsDeleted(false);        mapper.insert(entity);        /**         * 查询 id = 1 的数据         */        HelloWorldEntity result1 = mapper.findOne(mapper.query()            .where.id().eq(1L).end());        /**         * 控制台直接打印出查询结果         */        System.out.println("1. HelloWorldEntity:" + result1.toString());        /**         * 更新id = 1的记录         */        mapper.updateBy(mapper.updater()            .set.sayHello().is("say hello, say hello!")            .set.yourName().is("Fluent Mybatis is powerful!").end()            .where.id().eq(1L).end()        );        /**         * 查询 id = 1 的数据         */        HelloWorldEntity result2 = mapper.findOne(mapper.query()            .where.sayHello().like("hello")            .and.isDeleted().eq(false).end()            .limit(1)        );        /**         * 控制台直接打印出查询结果         */        System.out.println("2. HelloWorldEntity:" + result2.toString());    }}

输出:

 
 HelloWorldEntity:HelloWorldEntity{id=1, sayHello='hello world', yourName='Fluent Mybatis', gmtCreate=null, gmtModified=null, isDeleted=false} HelloWorldEntity:HelloWorldEntity{id=1, sayHello='say hello, say hello!', yourName='Fluent Mybatis is powerful!', gmtCreate=null, gmtModified=null, isDeleted=false}

神奇吧!我们再到数据库中查看一下结果

图片

现在,我们已经通过一个简单例子演示了 Fluent Mybatis 的强大功能, 在进一步介绍 Fluent Mybatis 更强大功能前,我们揭示一下为啥我们只写了一个数据表对应的 Entity 类, 却拥有了一系列增删改查的数据库操作方法。

Fluent Mybatis 根据 Entity 类上 @FluentMybatis 注解在编译时, 会在 target 目录 class 目录下自动编译生成一系列文件:

图片

这些文件的具体作用如下:

  • mapper/*Mapper : Mybatis 的 Mapper 定义接口, 定义了一系列通用的数据操作接口方法。

  • dao/*BaseDao : Dao 实现基类, 所有的 DaoImpl 都继承各自基类 根据分层编码的原则,我们不会在 Service 类中直接使用 Mapper 类,而是引用 Dao 类。我们在 Dao 实现类中根据条件实现具体的数据操作方法。

  • wrapper/*Query : Fluent Mybatis 核心类, 用来进行动态 sql 的构造, 进行条件查询。

  • wrapper/*Updater : Fluent Mybatis 核心类, 用来动态构造 update 语句。

  • helper/*Mapping : Entity 表字段和 Entity 属性映射定义类

  • helper/*Segment: Query 和 Updater 具体功能实现, 包含几个实现: select, where, group by, having by, order by, limit

  • IEntityRelation : 处理 Entity 关联(一对一, 一对多, 多对多)关系的接口

  • Ref : 引用 Fluent Mybatis 生成的对象的快捷入口工具类

# 总结

上面只是 Fluent Mybatis 常规实现增删改查的方式,Fluent Mybatis 现在又推出了专门面向表单级的增删改查,声明即实现。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/804089.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Nginx反向代理与Tomcat实现ssm项目前后端分离部署

Nginx nginx是一款http和支持反向代理的web服务器&#xff0c;以其优越的性能被广泛使用。以下是百度百科的介绍。 Nginx (engine x) 是一个高性能的HTTP和反向代理web服务器&#xff0c;同时也提供了IMAP/POP3/SMTP服务。Nginx是由伊戈尔赛索耶夫为俄罗斯访问量第二的Rambler.…

智慧园区水电能源监控管理系统

随着智慧城市的快速发展&#xff0c;智慧园区作为城市智能化的重要组成部分&#xff0c;其能源监控管理系统显得尤为关键。智慧园区水电能源监控管理系统&#xff0c;是利用先进的信息技术和自动控制技术&#xff0c;对园区内的水电能源使用进行实时监控、管理和优化的综合性智…

美国34401A安捷伦数字万用表

181/2461/8938产品概述&#xff1a; 附加功能: 6 1/2位数分辨率10种测量功能:DC/交流电压、DC/交流电流、2线和4线电阻、二极管、连续性、频率、周期基本精度:0.0035% DC&#xff0c;0.06%交流1000 V最大电压输入&#xff0c;3 A最大电流输入每秒1000次读数512读取记忆 安捷…

Linux C++ 027-STL之deque容器

Linux C 027-STL之deque容器 本节关键字&#xff1a;Linux、C、deque 相关库函数&#xff1a;pubsh_back、begin、front、sort deque基本概念 功能&#xff1a;双端数组&#xff0c;可以对头端进行插入删除操作。 deque 与 vector 的区别&#xff1a; &#xff08;1&#x…

vue将html生成pdf并分页

jspdf html2canvas 此方案有很多的css兼容问题&#xff0c;比如虚线边框、svg、页数多了内容显示不全、部分浏览器兼容问题&#xff0c;光是解决这些问题就耗费了我不少岁月和精力 后面了解到新的技术方案&#xff1a; jspdf html-to-image npm install --save html-to-i…

关于pandas 无法读取 csv 文件数据的解决方式

你好&#xff0c;我是 shengjk1&#xff0c;多年大厂经验&#xff0c;努力构建 通俗易懂的、好玩的编程语言教程。 欢迎关注&#xff01;你会有如下收益&#xff1a; 了解大厂经验拥有和大厂相匹配的技术等 希望看什么&#xff0c;评论或者私信告诉我&#xff01; 文章目录 …

LabVIEW和2D激光扫描的受电弓滑板磨耗精确测量

LabVIEW和2D激光扫描的受电弓滑板磨耗精确测量 在电气化铁路运输中&#xff0c;受电弓滑板的健康状况对于保障列车安全行驶至关重要。受电弓滑板作为连接电网与列车的直接介质&#xff0c;其磨损情况直接影响到电能的有效传输及列车的稳定运行。精确、快速测量受电弓滑板磨损情…

IntelliJ IDEA 2024.1安装与激活[破解]

一&#xff1a;IDEA官方下载 ①如题&#xff0c;先到IDEA官方下载&#xff0c;简简单单 ②IDEA官方&#xff1a;IntelliJ IDEA – the Leading Java and Kotlin IDE 二&#xff1a;获取脚本 &#x1f31f;网盘下载&#xff1a;jetbra (密码&#xff1a;lzh7) &#x1f31f;获取…

CLI的使用与IOS基本命令

1、实验目的 通过本实验可以掌握&#xff1a; CLI的各种工作模式个CLI各种编辑命令“?” 和【Tab】键使用方法IOS基本命令网络设备访问限制查看设备的相关信息 2、实验拓扑 CLI的使用与IOS基本命令使用拓扑如下图所示。 3、实验步骤 &#xff08;1&#xff09;CLI模式的切…

Visual Studio Code 终端为管理员权限

第一部 1、 Visual Studio Code 快捷方式启动选项加上管理员启动 第二步 管理员方式运行 powershell Windows 10的任务栏自带了搜索。或者开始菜单选搜索只需在搜索框中输入powershell。 在出来的搜索结果中右击Windows PowerShell&#xff0c;然后选择以管理员方式运行。 执…

使用Docker Registry-v2搭建镜像仓库详细教程

我们使用docker来部署私有化镜像仓库… 1、下载 registry:v2 镜像 docker pull registry:22、在私有仓库所在的主机目录新建一个文件夹&#xff0c;用于持久化保存仓库中的镜像 mkdir -p /opt/registry3、启动registry镜像 使用docker镜像启动私有仓库容器服务&#xff0c;…

ArcGIS Desktop使用入门(四)工具箱——属性域

系列文章目录 ArcGIS Desktop使用入门&#xff08;一&#xff09;软件初认识 ArcGIS Desktop使用入门&#xff08;二&#xff09;常用工具条——标准工具 ArcGIS Desktop使用入门&#xff08;二&#xff09;常用工具条——编辑器 ArcGIS Desktop使用入门&#xff08;二&#x…

nacos服务治理

nacos 服务演变之路 单体架构 集群级垂直化 SOA 微服务 微服务优缺点 SOA与微服务区别 springcloud技术栈 服务发现概念 服务发现两种方式–客户端服务发现 服务发现两种方式–服务端发现 服务发现技术对比 nacos架构图 nacos实战 服务发现 源码解析 nacos实现了springcloud…

网络安全---RSA公钥加密与签名

实验项目&#xff1a;RSA公钥加密与签名实验 1.实验目的 本实验的学习目标是让学生获得 RSA 算法的动手经验。 通过课堂学习&#xff0c;学生应该已经了解 RSA 算法的理论部分&#xff0c; 知道在数学上如何生成公钥、私钥以及如何执行加密、解密和签名生成、验证。 通过使用…

Docker容器嵌入式开发:Docker Ubuntu18.04配置mysql数据库

在 Ubuntu 18.04 操作系统中安装 MySQL 数据库的过程。下面是安装过程的详细描述: 首先,使用以下命令安装 MySQL 服务器: sudo apt install mysql-server系统会提示是否继续安装,按下 Y 键确认。 安装过程中,系统会下载并安装 MySQL 相关的软件包,包括 libaio1、mysql…

STM32+ESP8266水墨屏天气时钟:文字取模和图片取模教程

项目背景 本次的水墨屏幕项目需要显示一些图片和文字&#xff0c;所以需要对图片和文字进行取模。 取模步骤 1.打开取模软件 2.选择图形模式 3.设置字模选项 注意&#xff1a;本次项目采用的是水墨屏&#xff0c;并且是局部刷新的代码&#xff0c;所以设置字模选项可能有点…

实验3 交换机基本配置

实验3 交换机基本配置 一、 原理描述二、 实验目的三、 实验内容四、实验步骤1.建立实验拓扑2.设备编址3.检测链路连通性4.交换机双工模式配置 一、 原理描述 交换机&#xff08;Switch&#xff09;也称为交换式集线器&#xff0c;其工作在OSI 第二层&#xff08;数据链路层)上…

Windows摄像头推流-RTSP

0.背景&#xff1a; 调试rtsp视频流时&#xff0c;没有网络摄像头怎么办&#xff0c;只需要在同一个局域网下&#xff0c;用windows推送rtsp流&#xff0c;就可以在linux进行接收。 1.下载资源包 资源包链接&#xff1a;https://pan.baidu.com/s/1008I7TKazE4JgFiozhtekg?pw…

【算法深度探索】动态规划之旅(1):挑战OJ题海,解锁15道经典难题,让你成为DP大师!

&#x1f4c3;博客主页&#xff1a; 小镇敲码人 &#x1f680; 欢迎关注&#xff1a;&#x1f44d;点赞 &#x1f442;&#x1f3fd;留言 &#x1f60d;收藏 &#x1f30f; 任尔江湖满血骨&#xff0c;我自踏雪寻梅香。 万千浮云遮碧月&#xff0c;独傲天下百坚强。 男儿应有龙…

智慧公厕:提升城市管理效率,改善居民生活体验

智慧公厕作为城市基础设施的重要组成部分&#xff0c;正逐渐成为改善城市品质和提升居民生活体验的一项关键措施。通过智能化管理、数字化使用和信息化运行&#xff0c;智慧公厕不仅可以为城市居民带来更舒适便利的使用体验&#xff0c;而且对于城市的高质量发展、宜居性和包容…