ElementCollection 和“无法延迟初始化角色集合”异常

LSO*_*LSO 7 spring jpa spring-data

我对 Spring 很陌生,我正在尝试弄清楚如何使用 @ElementCollection。

我有以下课程:

@Embeddable
public class Phone {
      private String type;
      private String areaCode;
      @Column(name="P_NUMBER")
      private String number;

    public Phone() {
    }

    public Phone(String type, String areaCode, String number) {
        super();
        this.type = type;
        this.areaCode = areaCode;
        this.number = number;
    }

    public String getNumber() {
        return number;
    }

    public void setNumber(String number) {
        this.number = number;
    }


    public String getAreaCode() {
        return areaCode;
    }

    public void setAreaCode(String areaCode) {
        this.areaCode = areaCode;
    }

    public String getType() {
        return type;
    }

    public void setType(String type) {
        this.type = type;
    }
}

@Entity
public class Employee {

    @Id
    @GeneratedValue(strategy = GenerationType.AUTO)
    @Column(name = "EMP_ID")
    private long id;

    @ElementCollection//(fetch=FetchType.EAGER)
    @CollectionTable(name = "PHONE", joinColumns = @JoinColumn(name = "OWNER_ID"))
    private List<Phone> phones  = new ArrayList<Phone>();;

    public Employee() {
    }

    public long getId() {
        return id;
    }

    public void setId(long id) {
        this.id = id;
    }

    public List<Phone> getPhones() {
        return phones;
    }

    public void setPhones(List<Phone> phones) {
        this.phones = phones;
    }
}
Run Code Online (Sandbox Code Playgroud)

存储库:

@Repository
public interface EmployeeRepository extends CrudRepository<Employee, Long>{

    public Employee findById(long id);

}
Run Code Online (Sandbox Code Playgroud)

然后我在 main 方法中使用它:

public static void main(String[] args) {
        ConfigurableApplicationContext context = SpringApplication.run(Application.class, args);
        EmployeeRepository repository = context.getBean(EmployeeRepository.class);

        Phone phone = new Phone("work", "613", "494-1234");
        Employee emp = new Employee();
        emp.getPhones().add(phone);
        repository.save(emp);

        emp = repository.findById(1);
        for (Phone p : emp.getPhones()) {
            System.out.println(p.getNumber());
        }

        context.close();

    }
Run Code Online (Sandbox Code Playgroud)

它抛出异常(当 emp.getPhones() 被调用时):线程“main”org.hibernate.LazyInitializationException 中的异常:未能延迟初始化角色集合:elcol.repository.Employee.phones,无法初始化代理 - 没有会话

如果我将 (fetch=FetchType.EAGER) 添加到 @ElementCollection 注释(在上面的 Employee 类中的代码中注释) - 一切正常。

如果没有 FetchType.EAGER,我该如何解决这个问题?

Pre*_*ric 7

findById(long id)实现中,添加这个Hibernate.initialize(emp.getPhones()).

您的存储库服务应返回您需要的所有已初始化数据,以便调用该服务的客户端保持独立。简而言之,如果您不需要客户端的员工电话,请不要初始化它。如果您确实需要它 - 初始化它。

编辑

使用 spring 数据,您显然没有实现,因此您可以指定将使用的查询,并获取查询中的数据(问题带有标记,jpa所以我想您可以使用JpaRepository

@Repository
public interface EmployeeRepository extends JpaRepository<Employee, Long>{

    @Query("SELECT e FROM Employee e JOIN FETCH e.phones WHERE e.id = (:id)")
    public Employee findById(long id);
}
Run Code Online (Sandbox Code Playgroud)