1

是否有任何已知的方法来使用面向方面的注释(或其他任何工作)进行休眠分页,并使用切入点来识别任何给定方法中的 a并将Criteria其应用到API 方法?setFirstResultssetMaxResults

例如,我想在我的休眠GenericDao实现中应用如下分页:

@Paginate
public List<T> filter(Set<Criterion> filter, Map<String, String> alias){
    Criteria criteria = session.createCriteria(type.class);
    fillCriteria(criteria, filter, alias);
    return criteria.list();
}

并且@Paginate注释将适用setFirstResultssetMaxResults通话criteria.list()。这可能吗?

注意:@Paginate是我想使用的一个例子。

4

1 回答 1

1

是的,您可以通过AspectJ做到这一点。

这是一个自洽的独立代码示例。您可以复制并粘贴类和方面并运行它。我没有使用原始的 Hibernate 类,而是使用一些虚拟替代品,因为我不是 Hibernate 用户,而且我也想将此作为通用答案:

虚拟的、类似休眠的接口和类:

package de.scrum_master.app;

public interface Criterion {}
package de.scrum_master.app;

public class MyCriterion implements Criterion {
    private String criterion;

    public MyCriterion(String criterion) {
        this.criterion = criterion;
    }

    @Override
    public String toString() {
        return "MyCriterion [criterion=" + criterion + "]";
    }
}
package de.scrum_master.app;

import java.util.List;

public interface Criteria {
    Criteria add(Criterion criterion);
    List list();
    Criteria setFirstResult(int firstResult);
    Criteria setMaxResults(int maxResults);
}
package de.scrum_master.app;

import java.util.LinkedList;
import java.util.List;

public class MyCriteria implements Criteria {
    private List<Criterion> criteria = new LinkedList<>();

    @Override
    public Criteria add(Criterion criterion) {
        criteria.add(criterion);
        return this;
    }

    @Override
    public List list() {
        return new LinkedList();
    }

    @Override
    public Criteria setFirstResult(int firstResult) {
        add(new MyCriterion("first result " + firstResult));
        return this;
    }

    @Override
    public Criteria setMaxResults(int maxResults) {
        add(new MyCriterion("max results " + maxResults));
        return this;
    }

    @Override
    public String toString() {
        return "MyCriteria [criteria=" + criteria + "]";
    }
}

如您所见,方法setFirstResult(..)setMaxResults(..)通过添加特殊标准add(..)到内部列表。我这样做是为了稍后说明方面的效果。

分页注释:

package de.scrum_master.app;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface Paginate {}

驱动应用:

package de.scrum_master.app;

import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;

public class Application {
    @Paginate
    public List filter(Set<Criterion> filter, Map<String, String> alias) {
        Criteria criteria = new MyCriteria();
        for (Criterion criterion : filter)
            criteria.add(criterion);
        return criteria.list();
    }

    public static void main(String[] args) {
        Set<Criterion> filterCriteria = new HashSet<>();
        filterCriteria.add(new MyCriterion("where"));
        filterCriteria.add(new MyCriterion("group by"));
        new Application().filter(filterCriteria, null);
    }
}

该应用程序包含一个由 注释的方法@Paginate,与您的示例代码执行类似的操作。它虽然不调用任何分页方法。这将通过以下方面来完成。

分页方面:

package de.scrum_master.aspect;

import java.util.List;
import de.scrum_master.app.Paginate;
import de.scrum_master.app.Criteria;

public aspect PaginationAspect {
    Object around(Criteria criteria) : 
        call(public List Criteria+.list())  &&
        cflow(execution(@Paginate * *(..))) &&
        target(criteria)
    {
        System.out.println(thisJoinPoint);
        System.out.println("Original criteria: " + criteria);
        criteria.setFirstResult(5);
        criteria.setMaxResults(10);
        System.out.println("Modified criteria: " + criteria);
        return proceed(criteria);
    }
}

Criteria+.list()切入点拦截对(包含子类)的调用,但仅在由 注释的任何执行方法+的控制流 ( ) 中发出时。调用目标绑定到一个参数,以便在绑定到该切入点的通知方法中使用它。我们需要一个对调用目标的引用,因为我们想在它上面调用分页方法。这就是建议的作用,控制台输出清楚地说明了这一点。cflow()@PaginateCriteria criteria

控制台输出:

call(List de.scrum_master.app.Criteria.list())
Original criteria: MyCriteria [criteria=[MyCriterion [criterion=group by], MyCriterion [criterion=where]]]
Modified criteria: MyCriteria [criteria=[MyCriterion [criterion=group by], MyCriterion [criterion=where], MyCriterion [criterion=first result 5], MyCriterion [criterion=max results 10]]]

享受!

于 2014-10-21T14:26:09.703 回答