php - 基于对缓存对象的引用查询文档
问题描述
我有两种类型的对象User
,它们Company
的数据存储在 MongoDB 集合user
和company
. 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 对象或更改我的数据模型?
解决方案
不幸的是,如果您想使用 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 管理的文档,避免查询数据库。
推荐阅读
- c# - Azure Bot Framework - 无法访问的 QnA Maker 服务
- visual-studio - 如何构建不在 Visual Studio 解决方案中的项目?
- kubernetes - 将 UDP DNS 流量代理到 TCP 后端服务
- reactjs - 扩展运算符将 API 响应分配给接口
- opencv - 如何设置 pytesseract 来解决验证码字母数字和 5 长度
- django - 使用 Zappa 和 Lamda 安排的 Django 管理命令
- feature-selection - Scikit Learn SelectKBest 是否对连续数据本身进行分组?
- python - 如何将 pandas.Series 标签/标题分配给变量?
- node.js - postgres查询说明
- php - 编码以防止标头注入