我们有一个数据库模式,以简化的(稍微做作的)形式看起来像:
从用户到域的外键设置在列 (domainId, groupId) 上,以保证引用完整性。此结构可以很好地满足预期目的。
然而,对于一个与同一个数据库对话的新应用程序,我现在需要为 Doctrine 创建一个映射来映射上述结构,包括 两个 列上的外键关系。
我试过以下方法:
<entity name="User" table="users">
<!-- other fields -->
<many-to-one field="domain" target-entity="Domain" fetch="LAZY">
<join-columns>
<join-column name="domainId" referenced-column-name="domainId"/>
<join-column name="groupId" referenced-column-name="groupId"/>
</join-columns>
</many-to-one>
</entity>
但这给了我一个错误:
UnitOfWork.php 第 2649 行:未定义索引:groupId
所以,我的问题是:
在 Doctrine 中描述多列多对一外键关系的正确方法是什么?
为了完整起见,数据库按照上述 ERD 中的描述为模式创建代码:
CREATE TABLE `users` (
`userId` INT(10) UNSIGNED NOT NULL AUTO_INCREMENT,
`groupId` INT(10) UNSIGNED NOT NULL,
`domainId` INT(10) UNSIGNED NOT NULL,
`someData` VARCHAR(32),
PRIMARY KEY (`userId`),
KEY `key_users_groupId_domainId` (`groupId`, `domainId`)
) ENGINE=InnoDB;
CREATE TABLE `domains` (
`domainId` INT(10) UNSIGNED NOT NULL AUTO_INCREMENT,
`groupId` INT(10) UNSIGNED NOT NULL,
`someOtherData` VARCHAR(32),
PRIMARY KEY (`domainId`),
KEY `key_domains_groupId` (`groupId`)
) ENGINE=InnoDB;
CREATE TABLE `groups` (
`groupId` INT(10) UNSIGNED NOT NULL AUTO_INCREMENT,
`someMoreData` VARCHAR(32),
PRIMARY KEY (`groupId`)
) ENGINE=InnoDB;
ALTER TABLE `users`
ADD CONSTRAINT `fk_users_domains` FOREIGN KEY (`groupId`, `domainId`) REFERENCES `domains` (`groupId`, `domainId`),
ADD CONSTRAINT `fk_users_groups` FOREIGN KEY (`groupId`) REFERENCES `groups` (`groupId`);
ALTER TABLE `domains`
ADD CONSTRAINT `fk_domains_groups` FOREIGN KEY (`groupId`) REFERENCES `groups` (`groupId`);
最佳答案
对于您的问题,这不是一个很好的答案。另外,我从未使用过 Doctrine 或 Doctrine2。但是我花了一些时间环顾四周,几乎找到了前三个引用资料:
Doctrine multiple composite foreign key ,一个问题,虽然它没有显示 XML 映射,并且可能是偏离基础的,但至少它似乎与 FK 中的多列有关。并回答根据答案忽略的 Doctrine2 的某些方面。
Doctrine2 Map entities with composite foreign keys in ...一个没有收集到太多值(value)但至少可以作为欺骗候选人折叠到你的问题中的问题。
XML Mapping Doctrine2 XML 映射文档。它对搜索文本 multi
没有任何值(value),但搜索 composite
是这样说的:
For composite keys you can specify more than one id-element, however surrogate keys are recommended for use with Doctrine 2.
这让我想到这个Wikipedia Surrogate 的定义说明:
The surrogate is internally generated by the system and is invisible to the user or application.
和Natural vs Surrogate .关于在两者之间进行选择的讨论。
回到您的模型,按独立性降序排列:
CREATE TABLE `groups` (
`groupId` INT(10) UNSIGNED NOT NULL AUTO_INCREMENT,
`someMoreData` VARCHAR(32),
PRIMARY KEY (`groupId`)
) ENGINE=InnoDB;
CREATE TABLE `domains` (
`domainId` int(10) unsigned NOT NULL AUTO_INCREMENT,
`groupId` int(10) unsigned NOT NULL,
`someOtherData` varchar(32) DEFAULT NULL,
PRIMARY KEY (`domainId`),
KEY `key_domains_groupId` (`groupId`),
CONSTRAINT `fk_domains_groups` FOREIGN KEY (`groupId`) REFERENCES `groups` (`groupId`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
CREATE TABLE `users` (
`userId` int(10) unsigned NOT NULL AUTO_INCREMENT,
`groupId` int(10) unsigned NOT NULL,
`domainId` int(10) unsigned NOT NULL,
`someData` varchar(32) DEFAULT NULL,
PRIMARY KEY (`userId`),
KEY `key_users_groupId_domainId` (`groupId`,`domainId`),
CONSTRAINT `fk_users_domains` FOREIGN KEY (`groupId`, `domainId`) REFERENCES `domains` (`groupId`, `domainId`),
CONSTRAINT `fk_users_groups` FOREIGN KEY (`groupId`) REFERENCES `groups` (`groupId`)
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
一些草稿:
truncate table groups; -- disallowed
delete from groups;
alter table groups auto_increment 1; -- reset, after running delete from.
insert groups(someMoreData) values ('group0001'),('group0002');
select * from groups;
insert domains(groupId,someOtherData) values
(1,'sod'),(1,'sod'),(1,'sod'),
(2,'sod'),(2,'sod');
select * from domains; -- AI 1 to 5 above
insert users(groupId,domainId,someData) values (1,1,'sd'); -- success
insert users(groupId,domainId,someData) values (1,3,'sd'); -- success
insert users(groupId,domainId,someData) values (1,4,'sd'); -- Error 1452 fk failure
很明显,users
并不真正需要复合 FK 到 domains
。相反,它只需要一个列 FK 到 domains
的代理 AI PK 中。这足以实现与您正在做的相同的效果。
为了与此保持一致,users.domainId
就足够了,users.groupId
引入了反规范化,后者应该被丢弃。
无论如何,希望这对您有所帮助。
关于php - 如何在 Doctrine 映射中描述多列外键,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38234061/