php - 如何在 Doctrine 映射中描述多列外键

标签 php mysql doctrine foreign-key-relationship doctrine-mapping

我们有一个数据库模式,以简化的(稍微做作的)形式看起来像:

enter image description here

从用户到域的外键设置在列 (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/

相关文章:

php - 如何使 column2 第二行成为第一行,而所有其他列保持不变

php - SimpleSAMLPHP:因为它不是此 SP 的有效 IdP

php - 类型错误 : $(selector) is null

php - 收到错误 fatal error : Call to undefined function mysqli_connect()

php - doctrine odm,查找 id 不是对象 id 的文档

php - Doctrine 未在 Doctrine :schema:update 上创建一对多关系

javascript - 如何使用 Jquery 获取 pickadate 值?

mysql - 跨多个非唯一列的唯一记录

php - 我应该怎么做才能将各种用户输入的字符保存在 MySQL 数据库中?

symfony - 使用学问查询构建器的“不在哪里”查询