27

我现在试图弄清楚这段代码大约两个小时,例如在下面的课程中,这些字段代表什么?

import java.util.Date;
import javax.persistence.metamodel.ListAttribute;
import javax.persistence.metamodel.SingularAttribute;
import javax.persistence.metamodel.StaticMetamodel;
import java.util.UUID;

@StaticMetamodel(Address.class)
public class Address_ extends  {
    public static volatile SingularAttribute<Address, Long> id;
    public static volatile SingularAttribute<Address, UUID> personId;
    public static volatile SingularAttribute<Address, Person> person;
}

Address.class是一个具有以下定义的java类:

@Entity
@Table(name = "address", schema = "public")
public class Address{
    private Long id;
    private Person person;
    private UUID personId;
    //....
}

请您解释一下@StaticMetamodel@SingularAttribute注释的用途是什么?这可能很简单,但我无法理解。

4

2 回答 2

27

根据文档

静态元模型是一系列类,它们“镜像”域模型中的实体和可嵌入项,并提供对镜像类属性的元数据的静态访问。

静态元模型具有以下属性:

  • 对于 package 中的每个托管类,都会创建Xpackagep中的元模型类。X_p
  • y对于由 class 声明的每个持久的非集合值属性X,其中的类型yY,元模型类必须包含如下声明:

SingularAttribute例子:

public static volatile SingularAttribute<X, Y> y;

静态元模型对于使用 JPA 的 Criteria API创建类型安全的查询很有用。

例如,让我们有以下两个实体,Order并且Item

@Entity
public class Order {
    @Id 
    @GeneratedValue
    Integer id;

    @ManyToOne 
    Customer customer;

    @OneToMany 
    Set<Item> items;

    BigDecimal totalCost;

    // accessors
}

Item实体:

@Entity  
public class Item { 
    @Id
    @GeneratedValue
    Integer id;

    int quantity;

    @ManyToOne
    Order order;

    // accessors
}

这是一个类型安全的条件查询,使用 Criteria API 构建:

CriteriaBuilder cb = entityManager.getCriteriaBuilder();
CriteriaQuery<Order> cq = cb.createQuery(Order.class);
SetJoin<Order, Item> itemNode = cq.from(Order.class).join(Order_.items);
cq.where(cb.equal(itemNode.get(Item_.id), 5)).distinct(true);

Item_.id注意和的用法Order_.item。这些静态访问静态元模型属性(反映实体属性),这样可以确保正确构建查询。

于 2013-10-31T10:55:35.387 回答
4

最近我一直在思考这个问题,因为我一直在努力学习和理解 JPA。我相信我已经回答了你的问题:为什么我们需要元模型,为什么我们不能只使用实体模型?

看看这个实体:

@Entity  
public class Item { 
    @Id
    @GeneratedValue
    Integer id;

    int quantity;

    @ManyToOne
    Order order;

    // accessors
}

请注意,实体上的所有属性都没有关键字 static。这意味着为了使用它们,我们需要创建一个新对象。

当我们使用 CriteriaBuilder 构建查询时,我们不需要创建对象......我们只想使用实体上的属性来生成我们的查询。这就是我们拥有元模型的原因!它们创建我们无需创建对象即可访问的静态属性。所以我们可以做康斯坦丁提到的事情:

CriteriaBuilder cb = entityManager.getCriteriaBuilder();
CriteriaQuery<Order> cq = cb.createQuery(Order.class);
SetJoin<Order, Item> itemNode = cq.from(Order.class).join(Order_.items);
cq.where(cb.equal(itemNode.get(Item_.id), 5)).distinct(true);

在这里,我们不是在制作“Item”对象……我们只需要知道它的属性。MetaModel 上的静态属性使我们能够这样做!

于 2020-02-06T19:45:51.617 回答