我想在 2 个实体之间有一个 JoinTable,这两个实体派生自同一个所有者实体。因此,当使用@ManyToMany关系加入此实体时,我得到了@JoinTable(如下面的DDL):
CREATE TABLE IF NOT EXISTS `local_services`.`service_provided_on` (
`provider_id` INT UNSIGNED NOT NULL,
`service_id` INT UNSIGNED NOT NULL,
`service_point_no` INT UNSIGNED NOT NULL,
`work_station_no` INT UNSIGNED NOT NULL,
PRIMARY KEY (`provider_id`, `service_id`, `service_point_no`, `work_station_no`),
INDEX `fk_provider_service_has_work_station_work_station1_idx` (`service_point_no` ASC, `work_station_no` ASC, `provider_id` ASC),
INDEX `fk_provider_service_has_work_station_provider_service1_idx` (`provider_id` ASC, `service_id` ASC),
CONSTRAINT `fk_service_provided_on_provider_service`
FOREIGN KEY (`provider_id` , `service_id`)
REFERENCES `local_services`.`provider_service` (`provider_id` , `service_id`)
ON DELETE CASCADE
ON UPDATE CASCADE,
CONSTRAINT `fk_service_provided_work_station`
FOREIGN KEY (`service_point_no` , `work_station_no` , `provider_id`)
REFERENCES `local_services`.`work_station` (`service_point_no` , `work_station_no` , `provider_id`)
ON DELETE CASCADE
ON UPDATE CASCADE)
ENGINE = InnoDB
如您所见,有 2 个外键,每个外键都使用相同的provider_id列。我想使用此@JoinTable定义给定提供商在属于它的工作场所(工作站)上提供的服务。很明显,ID 为 ex. 5 的提供商提供的服务只能在属于 ID 为 5 的提供商的工作场所提供。因此,最好在每个外键之间共享此@JoinColumn。例如,当尝试插入与提供商ID不匹配的工作场所/服务以引发一些异常时!
我尝试做这样的事情:
@ManyToMany(fetch = FetchType.LAZY)
@JoinTable(name = "service_provided_on",
joinColumns = {
@JoinColumn(name = "provider_id", referencedColumnName = "provider_id", nullable = false, columnDefinition = "BIGINT UNSIGNED"),
@JoinColumn(name = "service_id", referencedColumnName = "service_id", nullable = false, columnDefinition = "INT UNSIGNED")
},
inverseJoinColumns = {
@JoinColumn(name = "provider_id", referencedColumnName = "provider_id", insertable = false, updatable = false),
@JoinColumn(name = "service_point_no", referencedColumnName = "service_point_no", nullable = false, columnDefinition = "INT UNSIGNED"),
@JoinColumn(name = "work_station_no", referencedColumnName = "work_station_no", nullable = false, columnDefinition = "INT UNSIGNED")
}
)
但它显然不起作用并引发这样的异常:
Caused by: org.hibernate.MappingException: Repeated column in mapping for collection: pl.salonea.entities.WorkStation.providedServices column: provider_id"}}
我考虑重命名一个外键,这provider_id例如work_station_provider_id但随后我将被允许插入不匹配的provider_ids,也许我可以定义一些约束来阻止这种行为(如何在 JPA 中定义它?它可以工作,但我会有具有相同provider_id的冗余列
在某些方面,以这种方式映射模型不太有意义。在你的多对多关系中,没有什么可以保证关系中的Service
s和Location
s必须具有相同的Provider
。我会更改您的对象模型以明确该要求。在Provider
和新对象ServiceProvidedOn
之间添加一对多关系(不是最好的名称;也许类似于ServiceOccurrence
?)。所以Provider
会有一个集合属性serviceProvidedOns
;新的类ServiceProvidedOn
将有三个属性provider
、service
和workStation
(正确映射)。ServiceProvidedOn
的主键将从其他三个对象(Provider
、Service
和WorkStation
)派生。
[我对你的模型做了一些假设;所以我希望这个建议是有意义的。