【问题标题】:Spring not loading data even with FetchType.EAGER set即使设置了 FetchType.EAGER,Spring 也不会加载数据
【发布时间】:2017-02-26 22:10:46
【问题描述】:

我有两个模型正在尝试从 REST API(宠物和媒体)取回数据。我试图通过 FetchType.EAGER 注释急切地加载宠物和媒体之间的 oneToMany 关系,但是当我编写 MediaRepository 时数据没有出现。如果我不实现该文件,媒体关系和数据会在响应中返回。

随着 MediaRepository.java 的实现,GET /pets 返回:

{
  "id": 72,
  "name": "Spot",
  "description": "Annoying as hell",
  "media": [], <-- why is this here only if I don't implement MediaRepository?
  ...
}

没有 MediaRepository.java 实现,GET /pets 返回:

{
  "id": 72,
  "name": "Spot",
  "description": "Annoying as hell",
  ... (No media array in response)
}

Pet.java

@Entity
public class Pet implements Serializable {

    private static final long serialVersionUID = 1L;

    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    private Long id;

    @Column(name="name")
    private String name;

    @Column(name="description")
    private String description;

    @OneToMany(cascade=CascadeType.ALL, mappedBy="pet", FetchType.EAGER, orphanRemoval=true)
    private List<Media> media;

    @ManyToOne
    private Category category;

    @Enumerated(EnumType.STRING)
    private Status status;
}

Media.java

@Entity
public class Media implements Serializable {
    private static final long serialVersionUID = 1L;

    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    private Long id;

    @Column(name="url")
    private String url;

    @Column(name="title")
    private String title;

    @ManyToOne
    private Pet pet;
}

PetRepository.java

import org.springframework.data.jpa.repository.JpaRepository;

public interface PetRepository extends JpaRepository<Pet, Long> {
}

MediaRepository.java

import org.springframework.data.jpa.repository.JpaRepository;

public interface MediaRepository extends JpaRepository<Media, Long> {
}

【问题讨论】:

  • 你在使用 Spring Data Rest 吗?
  • @AlanHay 是的,我是。
  • 能否也加db表结构,pet如何映射到media?

标签: java spring hibernate spring-data-jpa spring-data-rest


【解决方案1】:

这与 Hibernate fetch 策略无关。

您看到的行为是 Spring Data Rest 的设计方式。当您为媒体定义了存储库后,您将看到响应中提供了一个链接,供客户端检索关联的媒体项目。如果没有存储库,则必须在响应中内联关联,因为当然无法独立检索集合。

如果您希望在响应中选择性地嵌入集合,那么您可以通过定义投影来实现。

@Projection(name = "inlineData", types=Pet.class)
public interface PetProjection{

    Long getId();
    String getName();
    String getDescription();
    List<Media> getMedia();
}

您可以将此投影自动应用于集合资源:

@RepositoryRestResource(excerptProjection = PetProjection.class)
public interface PetRepository extends JpaRepository<Pet, Long> {}

对于项目资源,客户通常会指定他们希望这些数据内联:

例如

http://example.com/api/pets/1?projection=inlineData

http://docs.spring.io/spring-data/rest/docs/current/reference/html/#projections-excerpts

【讨论】:

  • 谢谢!这正是我所需要的。
  • 我遇到了投影无法正常工作的问题,在我尝试了这三件事后,它开始工作了——谁能解释一下为什么? * 将投影命名为与类相同(使用 spring boot 2),* 创建第二个投影,* 运行 maven clean。
猜你喜欢
  • 1970-01-01
  • 2021-05-18
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2019-06-16
  • 2012-05-18
  • 1970-01-01
  • 2011-12-06
相关资源
最近更新 更多