web-dev-qa-db-fra.com

Comment joindre des tables sur une clé non primaire en utilisant JPA et Hibernate

J'ai 3 modèles User, House, UserHouseMap. Et je dois accéder à la maison d'un utilisateur via la carte. Le seul problème est que c'est une ancienne base de données et je ne peux pas changer le fait que j'ai besoin de mapper User à UserHouseMap en utilisant user.name, Qui est une clé non primaire.

Hibernate continue de me donner des erreurs disant que je dois l'avoir comme clé primaire ou j'obtiens des erreurs disant A JPA error occurred (Unable to build EntityManagerFactory): Unable to find column with logical name: name in org.hibernate.mapping.Table(users) and its related supertables and secondary tables

J'ai essayé @Formula Comme solution de contournement, mais cela n'a pas fonctionné. J'ai également essayé @JoinColumnOrFormula Mais cela n'a pas fonctionné non plus. Voici ma solution avec @Formula

@Expose
@ManyToOne(targetEntity = House.class)
@Formula("(select * from houses inner join user_house_map on houses.house_name = user_house_map.house_name where user_house_map.user_name=name)")
public House house;

Voici ma tentative d'une solution @JoinColumnOrFormula.

@Expose
@ManyToOne(targetEntity = House.class)
@JoinColumnsOrFormulas({
        @JoinColumnOrFormula(formula=@JoinFormula(value="select name from users where users.id= id", referencedColumnName="name")),
        @JoinColumnOrFormula(column = @JoinColumn(name= "house_name", referencedColumnName="house_name"))
})
public House house;

Voici ma cartographie

@Id
@GeneratedValue
@Expose
public Long id;

@Expose
@Required
@ManyToOne
@JoinTable(
        name="user_house_map",
        joinColumns=
        @JoinColumn(unique=true,name="user_name", referencedColumnName="name"),
        inverseJoinColumns=
        @JoinColumn(name="house_name", referencedColumnName="house_name"))
private House house;

Voici les schémas DB

tilisateurs

                               Table "public.users"
        Column         |            Type             |          Modifiers          
-----------------------+-----------------------------+-----------------------------
 name                  | character varying(255)      |
 id                    | integer                     | not null 
Indexes:
    "user_pkey" PRIMARY KEY, btree (id)
Foreign-key constraints:
    "housing_fkey" FOREIGN KEY (name) REFERENCES user_house_map(user_name) DEFERRABLE INITIALLY DEFERRED

Maisons

                Table "public.houses"
    Column     |          Type          | Modifiers 
---------------+------------------------+-----------
 house_name    | character varying(255) | not null
 address       | text                   | 
 city          | text                   | 
 state         | text                   | 
 Zip           | integer                | 
 Zip_ext       | integer                | 
 phone         | text                   | 
Indexes:
    "house_pkey" PRIMARY KEY, btree (house_name)
Referenced by:
    TABLE "user_house_map" CONSTRAINT "house_map_fkey" FOREIGN KEY (house_name) REFERENCES house(house_name) DEFERRABLE INITIALLY DEFERRED

serHouseMap

         Table "public.user_house_map"
   Column    |          Type          | Modifiers 
-------------+------------------------+-----------
 user_name   | character varying(255) | not null
 house_name  | character varying(255) | not null
Indexes:
    "user_house_map_pkey" PRIMARY KEY, btree (user_name)
    "user_house_map_house_key" btree (house_name)
Foreign-key constraints:
    "user_house_map_house_fkey" FOREIGN KEY (house_name) REFERENCES houses(house_name) DEFERRABLE INITIALLY DEFERRED
Referenced by:
    TABLE "users" CONSTRAINT "housing_fkey" FOREIGN KEY (name) REFERENCES user_house_map(user_name) DEFERRABLE INITIALLY DEFERRED
27
user2158382

Voici à quoi devrait ressembler votre mappage:

@Entity
public class User {

    @Id
    private Long id;

    private String name;

    @OneToMany(mappedBy = "user")
    private List<UserHouseMap> houses = new ArrayList<>();
}

@Entity
public class House {

    @Id
    @Column(name = "house_name", nullable = false, unique = true) 
    private String house_name;

    private String address;

    @OneToMany(mappedBy = "house")
    private List<UserHouseMap> users = new ArrayList<>();
}

@Entity
public class UserHouseMap implements Serializable {

    @Id @ManyToOne
    @JoinColumn(name = "user_name", referencedColumnName = "name")
    private User user;

    @Id @ManyToOne
    @JoinColumn(name = "house_name", referencedColumnName = "house_name")
    private House house;
}

User et House ont accès à leurs entités UserHouseMap associées, correspondant au schéma de la base de données.

Utiliser deux associations un-à-plusieurs est toujours mieux que de s'appuyer sur des relations plusieurs-à-plusieurs.

25
Vlad Mihalcea