首页 > 解决方案 > 基于对缓存对象的引用查询文档

问题描述

我有两种类型的对象User,它们Company的数据存储在 MongoDB 集合usercompany. User包含对 的引用Company。我可以在 UserRepository 中使用以下代码查询用户:

$this
  ->createQueryBuilder()
  ->field('employer')->references($company);

Company然而,这是一个使用 Redis 缓存的重度使用对象。当没有缓存存在时,一切正常。但是当从缓存中获取公司实例时。Doctrine 工作单元不知道该实例。因此,执行上面的代码将导致以下错误:

Cannot create a DBRef for class App\Document\Company without an identifier. Have you forgotten to persist/merge the document first?

我发现在从 Redis 获取公司后,我可以使用 hack 将公司注册到工作单元:

$company = $this->fetchFromCache($params);
$documentManager->getUnitOfWork()->registerManaged($company, $company->getId(), []);

然而,这似乎很难看。有没有一种方法可以查询用户而无需让 Doctrine 知道 Company 对象或更改我的数据模型?

标签: phpdoctrinedoctrine-odm

解决方案


不幸的是,如果您想使用 ODM 进行查询,您需要以某种方式让 ODM 知​​道您的对象。虽然您registerManaged将在 UnitOfWork 中注册文档时工作,但它有一个缺点,即您可能拥有的任何关系都会被破坏和/或 Doctrine 会在持久/刷新期间假定它们是新的。你可以试试这个:

$company = $this->fetchFromCache($params);
if ($dm->getUnitOfWork()->containsId($company->getId(), Company::class)) {
    return $dm->find(Company::class, $company->getId());
}
return $dm->merge($company);

如前所述,merge它将递归地合并您的文档及其嵌入的文档/关系,并将它们标记为由 ODM 管理,就像它们刚刚被获取一样。请注意,合并将符合您的级联设置以供参考。有关合并的更多信息可以在 docs中找到。另请注意,合并文档可能会影响已获取的文档并使用正在合并的内容更新其字段。

Merge操作只会在我们检查 ODM 是否已经管理由从缓存中获取的 id 标识的公司之前执行一次。如果是这样,我们将调用$dm->find()which 不会命中数据库,因为它首先会查找对象的内存映射并返回已管理的实例。这样,您将始终获得由 ODM 管理的文档,避免查询数据库。


推荐阅读