把Spring Data JPA改造成mybatis那样,读取配置文件中的动态SQL

该博客介绍了一种动态拼接SQL的方法,类似MyBatis,通过Freemarker实现。它能返回任何对象,在Spring Data JPA之上扩展且不影响原有功能。还详细说明了使用步骤,如在pom.xml导入包、配置sql模板位置等,同时提到别名下划线与驼峰可自动转换。

主要解决的问题:

  1. 动态的拼接 SQL, 类似 mybatis, 通过 freemarker 来实现的

  2. 能够返回任何对象,List,Page....

  3. 在 spring data jpa 之上扩展,不影响 jpa 原有功能

How to use ?

1. pom.xml 中导入如下包:

<dependency>    
  <groupId>com.rabbit</groupId>    
  <artifactId>spring-data-jpa-template</artifactId>    
  <version>${version}</version>
</dependency>

2. 项目中所有的 Repository 继承 GenericJpaRepository,不要使用原来的 JpaRepository

public interface UserRepository extends GenericJpaRepository<User, Long> {}

3. 初始化 FreemarkerSqlTemplates 配置 sql 模板的位置,默认是在 classpath:/sql/


@Bean
public FreemarkerSqlTemplates freemarkerSqlTemplates() {
    FreemarkerSqlTemplates templates = new FreemarkerSqlTemplates();
    templates.setSuffix(".sftl");   //指定模板的后缀
    templates.setTemplateLocation("classpath:/sqltemplates");   //模板文件的位置
    templates.setTemplateBasePackage("com.**.**.test");  //可选
    templates.setAutoCheck(sqlTemplateAutoCheck);   //是否需要开启自动检测 sql 文件的改变,建议在开发环境中开启,避免重启
    return templates;
}

4. 指定 repositoryFactoryBeanClass

@EnableJpaRepositories(
        basePackages = {"com.**.repository"},
        repositoryFactoryBeanClass = GenericJpaRepositoryFactoryBean.class)

5. 创建模板文件 user.sftl, 例如:通过 id 查询用户

--findById
SELECT * FROM t_user WHERE id=:id

模板文件使用的是freemarker, 所有都是freemarker的语法

6. 通过 @TemplateQuery("findById") 来指定使用 sql 来查询

public interface UserRepository extends GenericJpaRepository<User, Long> {
    @TemplateQuery("findById")
    public User findById(@Params("id") Long id);
}

@TemplateQuery 可以指定 sql 模板文件中对应的名字,如果没有指定,那么默认使用方法的名字


查询返回可以是任何类型

public interface SampleRepository extends GenericJpaRepository<Sample, Long> {
 
    @TemplateQuery
    Page<Sample> findByContent(String content, Pageable pageable);  //分页实体对象
 
    @TemplateQuery
    List<Sample> findByTemplateQueryObject(SampleQuery sampleQuery, Pageable pageable);  //List 实体对象
 
    @TemplateQuery
    Long countContent(String content);
 
    @TemplateQuery
    List<SampleDTO> findDtos(); List的 DTO 对象
 
}

Dto 中的对应可以是 Java 中的基本类型,Map 对象, 自定义的类, 都是通过别人注入。eg:

public class User {
    private String name;
    private Integer age;
    private Address address;
    private String personLike;
    private Map<String, Object> attrs;
 
    //.....
}

对应的 SQL 模板,

SELECT
user.name AS 'name',
user.age,
address.name AS 'address.name',
address.code AS 'address.code',
user.persion_like,
user.attr AS 'attrs.attr',
user.attr2 AS 'attrs.attr2'
FROM
t_user user
LEFT JOIN t_address address on user.address_id = address.id
...

别名下划线与驼峰可以自动转换

freemarker参考链接:http://freemarker.foofun.cn/dgui_quickstart_basics.html


作者:samHuangLiang

来源链接:

https://blog.youkuaiyun.com/samHuangLiang/article/details/79549984

### Spring Data JPAMyBatis 的特点及适用场景 #### 特点对比 ##### Spring Data JPA Spring Data JPA 是基于 Java Persistence API (JPA) 构建的一个模块,旨在简化数据访问层的开发工作。它提供了丰富的功能来操作关系型数据库。 - **声明式查询**:开发者可以通过接口方法名称定义查询逻辑,而无需编写具体的 SQL 语句[^1]。 - **集支持**:与 Spring 生态系统的其他组件紧密协作,特别是 Spring MVC 和事务管理等方面有良好的集效果。 - **自动分页和排序**:内置对分页(Pagination)以及排序(Sorting)的支持,使得处理大量记录变得简单高效。 - **事件监听机制**:允许注册实体生命周期中的各种回调函数,在特定时间触发自定义行为。 - **缓存配置灵活**:可以方便地启用二级缓存,并且能够指定不同的策略以优化性能表现。 ##### MyBatis MyBatis 则是一个持久化框架,专注于映射 XML 或注解形式描述的对象模型到数据库表结构之间转换的过程。 - **SQL 映射文件**:采用 XML 文件或者注解方式书写原生 SQL 查询语句,给予使用者极大的灵活性去控制执行细节[^2]。 - **动态 SQL 支持**:提供了一套强大的模板引擎用于构建复杂的条件组合查询,从而减少冗余代码量并提高可维护性。 - **延迟加载特性**:当关联对象未被显式请求时不会立即读取其全部内容,有助于降低内存占用率和提升响应速度。 - **多库适配性强**:由于不依赖于任何 ORM 工具内部实现,因此更容易适应多种不同类型的 RDBMS 平台。 #### 使用案例分析 对于那些希望快速搭建起一套标准的企业级应用架构来说,尤其是倾向于利用约定优于配置原则的应用程序,可以选择 **Spring Data JPA** 。这种情况下,业务需求相对固定,变更频率较低;同时团队员熟悉面向对象编程范式,则更有利于发挥该技术栈的优势。 相反,如果项目涉及到大量的复杂查询操作或是需要频繁调整底层存储方案的话,那么 **MyBatis** 可能更加合适一些。因为这里强调的是细粒度级别的 SQL 控制权,以及跨多个异构数据库环境下的兼容能力。 ```java // Example of using Spring Data JPA repository to perform CRUD operations. public interface UserRepository extends JpaRepository<User, Long> { List<User> findByLastName(String lastName); } // Example of defining mapper interfaces with MyBatis annotations. @Mapper public interface UserMapper { @Select("SELECT * FROM users WHERE last_name = #{lastName}") List<User> selectByLastName(@Param("lastName") String lastName); } ```
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值