Spring Data Jpa 让@Query复杂查询分页支持实体返回

背景

  Spring Data Jpa 虽然可以减少代码中Sql的数量,但其在复杂查询中略显乏力。网上很多文章都采用Java代码的形式去实现复杂查询,但这样一来Sql的效率变得不可控。也有文章采用@Query 注解去执行JPQL原生SQL,本人在使用过程中也倾向于这种方式。
  但有时采用@Query方式,框架无法正常返回我们需要的类型。比如复杂查询后后分页,Repository方法的返回写的是Page<User>,然而真正执行后返回的类型却变成了Page<Object[]>。究竟原因,分析源码后发现PagedExecution并未对执行结果进行处理。

通过对Spring-Data-Jpa源码的分析,发现在RepositoryFactorySupport.java中的getRepository方法(第124行),提供了钩子方法使得我们可以在自己的RepositoryProxyPostProcessor中向目标Repository注入MethodInterceptor(方法拦截器)。

    public <T> T getRepository(Class<T> repositoryInterface, Object customImplementation) {
        ...
        Object target = this.getTargetRepository(information);
        ProxyFactory result = new ProxyFactory();
        result.setTarget(target);
        result.setInterfaces(new Class[]{repositoryInterface, Repository.class});
        ...
        Iterator var8 = this.postProcessors.iterator();
        while(var8.hasNext()) {
            RepositoryProxyPostProcessor processor = (RepositoryProxyPostProcessor)var8.next();
            processor.postProcess(result, information);
        }
        ...
        result.addAdvice(new RepositoryFactorySupport.QueryExecutorMethodInterceptor(information, customImplementation, target));
        return result.getProxy(this.classLoader);
    }

本文将介绍通过自定义RepositoryProxyPostProcessor向Repository(我们应用中声明的Repository)中注入自定义MethodInterceptor以使得@Qurey复杂查询能支持分页POJO的返回。

1.编写业务Repository,并继承JpaRepository

继承JpaRepository使得返回结果支持分页

@Repository
public interface UserRepository extends JpaRepository<User, String> {
    /**
     * 返回指定部门下边的用户
     *
     * @param officeId
     * @param pageable
     * @return
     */
    @Query(value = "SELECT user.id, user.name FROM User user, Office office WHERE user.officeId = office.id AND office.id = ?1 ")
    Page<User> getUserInOffice(String officeId, Pageable pageable);
}

上边的查询还是比较简单的,这只是为了作为例子好理解。如果需要你可以将Sql改得更加复杂。注意,我采用的是JPQL而非Native Sql。启动单元测试调用上述方法,结果的确返回了Page对象,但是当你page.getContent()时,会发现返回的结果为List<Object[]>。
接下来进入主题

2.自定义MethodInterceptor,将content由List<Object[]>转为List<T>

继承MethodInterceptor,重写invoke方法执行其他代理获得Jpql返回结果集后,将List<Object[]>转为List<T>。

public class JpqlBeanMethodInterceptor implements MethodInterceptor {

    /**
    * 用于存放QueryMethod对应的字段和返回类型信息
    */
    private Map<Method, SelectAlias> selectAlias = new HashMap<>();

    public JpqlBeanMethodInterceptor(RepositoryInformation repositoryInformation) {
        Iterator<Method> iterable = repositoryInformation.getQueryMethods().iterator();
        SqlParser sqlParser = new DefaultSqlParser();
        while (iterable.hasNext()) {
            Method method = iterable.next();
            Query query = method.getAnnotation(Query.class);
            if (query == null || query.nativeQuery()) {
                continue;
            }
            //获取返回类型
            Class clazz = getGenericReturnClass(method);
            if (clazz == null) {
                continue;
            }
            SelectAlias alias = sqlParser.getAlias(query.value(), clazz);
            if (alias == null) {
                continue;
            }
            selectAlias.put(method, alias);
        }
    }

    @Override
    public Object invoke(MethodInvocation invocation) throws Throwable {
        //执行方法获得结果
        Object obj = invocation.proceed();
        if (!checkCanConvert(obj)) {
            return obj;
        }
        SelectAlias alias = selectAlias.get(invocation.getMethod());
        if (alias == null) {
            return obj;
        }
        List content = getPageContent((PageImpl) obj);
        convert(content, alias);
        return obj;
    }
    //由于篇幅原因只贴出部分代码
    ...
}

上述代码由于篇幅原因只贴出部分,后续整理完代码后将共享出来。代码中sqlParser采用jsqlparser从Sql中取出返回的字段解析后转换为SelectAlia。注意这部分代码是在构造方法中执行的,后续代码决定了这部分代码只会在Spring Boot启动的时候每个Repository执行一次,以提高执行效率。

3.自定义RepositoryProxyPostProcessor

在postProcess时向Repository注入JpqlBeanMethodInterceptor

public class JpqlBeanPostProcessor implements RepositoryProxyPostProcessor {

    @Override
    public void postProcess(ProxyFactory factory, RepositoryInformation repositoryInformation) {
        factory.addAdvice(new JpqlBeanMethodInterceptor(repositoryInformation));
    }

}

4.自定义JpaRepositoryFactoryBean

创建RepositoryFactory时,向其加入我们自定义的RepositoryProxyPostProcessor

public class GmRepositoryFactoryBean<T extends Repository<S, ID>, S, ID extends Serializable>
        extends JpaRepositoryFactoryBean<T, S, ID> {

    public GmRepositoryFactoryBean(Class<? extends T> repositoryInterface) {
        super(repositoryInterface);
    }

    @Override
    protected RepositoryFactorySupport createRepositoryFactory(EntityManager entityManager) {
        JpaRepositoryFactory jpaRepositoryFactory = new JpaRepositoryFactory(entityManager);
        jpaRepositoryFactory.addRepositoryProxyPostProcessor(new JpqlBeanPostProcessor());
        return jpaRepositoryFactory;
    }
}

这一步大家应该很熟悉了,写过公用Repository的朋友应该都知道其作用。不过注意一下,我们在返回JpaRepositoryFactory前,要将我们的RepositoryProxyPostProcessor 加进postProcessors中,不然前边就白做了。

5.让我们的JpaRepositoryFactoryBean起作用

这一步大家应该也比较熟悉了,直接上代码。

@Configuration
@EnableJpaRepositories(repositoryFactoryBeanClass = GmRepositoryFactoryBean.class)
@EnableSpringDataWebSupport
public class JpaDataConfig {

}

注意要放到项目目录下或者在Application把这个文件的包加到BasePackages中,反正就是要让Spring Boot启动的时候扫的到就对啦。

最后

再次启动单元测试,重新执行getUserInOffice方法,可以看到返回的Page中的content已经正常返回User的List了,大功告成!经测试,其执行速度与原先的返回Object[]基本保持一致,有时甚至更快。这个我也不知道为什么会更快,可能与网络因素有关吧,毕竟我在测试的时候数据库不在本地。。。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,319评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,801评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,567评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,156评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,019评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,090评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,500评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,192评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,474评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,566评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,338评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,212评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,572评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,890评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,169评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,478评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,661评论 2 335

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,495评论 18 139
  • Spring Boot 参考指南 介绍 转载自:https://www.gitbook.com/book/qbgb...
    毛宇鹏阅读 46,678评论 6 342
  • 这部分主要是开源Java EE框架方面的内容,包括Hibernate、MyBatis、Spring、Spring ...
    杂货铺老板阅读 1,316评论 0 2
  • 3.1. 核心概念 CrudRepository包含增删查改基础功能 PagingAndSortingReposi...
    titvax阅读 1,729评论 0 2
  • 飞落的流星,像那次回眸 而幸福 除了无限的追寻 没什么可以控制那种诱惑 当深夜掩盖了所有光芒 俺必须学会修补...
    南山野客阅读 150评论 0 1