简体   繁体   English

处理实体继承Spring Boot

[英]Handling entities inheritance spring boot

I'm working with this tutorial to handle entities inheritance. 我正在使用本教程来处理实体继承。 I have person and company entities that extends the User entity. 我有扩展User实体的个人和公司实体。

@Entity
@Inheritance
public abstract class User { 

@Id
private long id;

@NotNull
private String email;

// getters and settres
}

@Entity
public class Person extends User { 
private int age;
// getters and settres and other attributs
}

@Entity
public class Company extends User { 
private String companyName;
// getters and settres and other attribut
}

then UserRpository ,PersonRepository and Company Repository that extends the UserBaseRepository. 然后扩展UserBaseRepository的UserRpository,PersonRepository和Company Repository。

@NoRepositoryBean
public interface UserBaseRepository<T extends User> 
extends CrudRepository<T, Long> {

public T findByEmail(String email);

}

@Transactional
public interface UserRepository extends UserBaseRepository<User> { }

@Transactional
public interface PersonRepository extends UserBaseRepository<Person> { }

@Transactional
public interface CompanyRepository extends UserBaseRepository<Company> { }

the probleme is when calling personRepository.findAll() to get all persons , in result i got also companies. 问题是当调用personRepository.findAll()来获取所有人时,结果我也得到了公司。

Your issue is with the "Discriminator" column that JPA requires. 您的问题出在JPA要求的“ Discriminator”列中。 You are using the @Inheritance annotation and by default that will use the InheritanceType.SINGLE_TABLE strategy. 您正在使用@Inheritance批注,默认情况下将使用InheritanceType.SINGLE_TABLE策略。 That means the following: 这意味着:

  1. Your inherited entities Person and Company will go into a single table. 您继承的实体PersonCompany将进入一个表。
  2. JPA will require a Discriminator to differentiate between entity types. JPA将需要一个鉴别器来区分实体类型。

I did the following to make it work for your use case: 为了使它适用于您的用例,我做了以下工作:

Entities: 实体:

@Inheritance
@Entity
@Table(name = "user_table")
public abstract class User {

    @Id
    private long id;

    @NotNull
    @Column
    private String email;

    public long getId() {
        return id;
    }

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

    public String getEmail() {
        return email;
    }

    public void setEmail(String email) {
        this.email = email;
    }
}

@Entity
public class Company  extends User {

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

    public String getCompanyName() {
        return companyName;
    }

    public void setCompanyName(String companyName) {
        this.companyName = companyName;
    }
}

@Entity
public class Person extends User {

    @Column
    private int age;

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }
}

DB Schema: 数据库架构:

-- user table
create table user_table (
  id BIGINT         NOT NULL PRIMARY KEY,
    email             VARCHAR(50) NOT NULL,
    age               INT,
    company_name      VARCHAR(50),
    dtype             VARCHAR(80) -- Discriminator
);

Some test data: 一些测试数据:

insert into user_table(id, dtype, age, email) values
(1,'Person', 25, 'john.doe@email.com'),
(2,'Person',22, 'jane.doe@email.com');

insert into user_table(id, dtype, company_name, email) values
(3,'Company','Acme Consultants', 'acme@company.com'),
(4,'Company', 'Foo Consultants', 'foo@company.com');

Repositories: 仓库:

@NoRepositoryBean
public interface UserBaseRepository<T extends User> extends CrudRepository<T, Long> {

    T findByEmail(String email);
}

@Transactional
public interface PersonRepository extends UserBaseRepository<Person> {

}

@Transactional
public interface CompanyRepository extends UserBaseRepository<Company> {

}

JUnit Tests: JUnit测试:

public class MultiRepositoryTest extends BaseWebAppContextTest {

    @Autowired
    private PersonRepository personRepository;

    @Autowired
    private CompanyRepository companyRepository;

    @Test
    public void testGetPersons() {
        List<Person> target = new ArrayList<>();
        personRepository.findAll().forEach(target::add);
        Assert.assertEquals(2, target.size());
    }
    @Test
    public void testGetCompanies() {
        List<Company> target = new ArrayList<>();
        companyRepository.findAll().forEach(target::add);
        Assert.assertEquals(2, target.size());
    }

}

The above tests pass. 以上测试通过。 That indicates JPA now utilizes the discriminator correctly to retrieve the required records. 这表明JPA现在正确使用了鉴别器来检索所需的记录。

For JPA related theory for your question, see this link . 有关您的问题的JPA相关理论,请参见此链接

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM