有关 ResultSetMapping 的更新问题,请参阅最后的编辑
我定义了两个实体(Item 和 ItemType),其中一个与另一个具有 ManyToOne 关联。由于生成查找正确项目的一些复杂性,我有很多 native 查询。这些查询总是返回第一个实体的所有列 (SELECT items.* ...)。
我发现我的关联在第一项上始终为空,我不确定自己做错了什么。任何帮助将不胜感激。
实体:
namespace AppBundle\Entity;
use Psr\Log\LoggerInterface;
use Doctrine\ORM\Mapping as ORM;
/**
* @ORM\Table(name="items")
* @ORM\Entity(repositoryClass="AppBundle\Entity\ItemRepository")
*/
class Item {
/**
* @ORM\Column(type="integer")
* @ORM\Id
* @ORM\GeneratedValue(strategy="AUTO")
*/
private $id;
/**
* @ORM\Column(name="account_id", type="integer")
*/
private $accountId;
/**
* @ORM\ManyToOne(targetEntity="ItemType")
* @ORM\JoinColumn(name="item_type_id", referencedColumnName="id")
*/
private $itemType;
// ..snip.. //
}
元素类型
namespace AppBundle\Entity;
use Doctrine\ORM\Mapping as ORM;
/**
* @ORM\Table(name="item_types")
* @ORM\Entity(repositoryClass="AppBundle\Entity\ItemTypeRepository")
*/
class ItemType {
/**
* @ORM\Column(type="integer")
* @ORM\Id
* @ORM\GeneratedValue(strategy="AUTO")
*/
private $id;
/**
* @ORM\Column(name="account_id", type="integer")
*/
private $accountId;
/**
* @ORM\Column(type="string", length=128)
*/
private $name;
// ..snip.. //
}
查询是由我的 ItemRepository 类的 getItem 方法生成的。这有点长,但归结为 SELECT items.* FROM items ... 通过 getEntityManager()->createNativeQuery($sql, $rsm); 运行的查询
namespace AppBundle\Entity;
use Psr\Log\LoggerInterface;
use Doctrine\ORM\Query\ResultSetMapping;
/**
* ItemRepository
*
* This class was generated by the Doctrine ORM. Add your own custom
* repository methods below.
*/
class ItemRepository extends \Doctrine\ORM\EntityRepository
{
/**
* @var \Psr\Log\LoggerInterface $logger
*/
protected $logger;
/**
* @var ItemTypeRepository
*/
protected $itemTypes;
/**
* @var ItemValueRepository
*/
protected $itemValues;
/**
* @var FieldRepository
*/
protected $fields;
/**
* Called by service bindings in services.yml instead of __construct, which is needed by
* Doctrine.
*/
public function initService(LoggerInterface $logger,
ItemTypeRepository $itemTypes,
ItemValueRepository $itemValues,
FieldRepository $fields)
{
$this->logger = $logger;
$this->itemTypes = $itemTypes;
$this->itemValues = $itemValues;
$this->fields = $fields;
}
/**
* Get items for an account via itemId
*
* @param integer $accountId a user's account id
* @param $itemId unique ID for an Item
* @return Item_model
*/
public function getItem($accountId, $itemId, $restrictedUserOwnerItemType, $restrictedUserOwnerItemId)
{
$this->logger->debug(__METHOD__.'::params::'.json_encode(['accountId' => $accountId, 'itemId' => $itemId,
'restrictedUserOwnerItemType' => $restrictedUserOwnerItemType, 'restrictedUserOwnerItemId' => $restrictedUserOwnerItemId]));
if(!$accountId || !$itemId || !is_numeric($restrictedUserOwnerItemType) || !is_numeric($restrictedUserOwnerItemId))
throw new \InvalidArgumentException('getItem requires accountId, itemId, restrictedUserOwnerItemType and restrictedUserOwnerItemId');
/*
$query = $this->itemsModel->builder();
$result = $query->where('account_id', '=', $accountId)
->where('id', '=', $itemId)
->first();
*/
$sql = "SELECT items.*, ".
"item_types.id AS item_type_id, ".
"item_types.account_id AS item_type_account_id, ".
"item_types.name AS item_type_name, ".
"item_types.plural_name AS item_type_name, ".
"item_types.label AS item_type_label, ".
"item_types.plural_label AS item_type_plural_label, ".
"item_types.are_users AS item_type_are_users, ".
"item_types.own_users AS item_type_own_users ".
"FROM items ".
"JOIN item_types ON item_types.id = items.item_type_id ";
$isRestrictedUser = $restrictedUserOwnerItemType != 0 || $restrictedUserOwnerItemId != 0;
if($isRestrictedUser)
{
// Limit to items that are visible to restricted users
$sql .= <<<SQL
WHERE item_types.visible_to_restricted_users = 1
SQL;
// Limit to items that contain a relationship field pointed at the same owner item type,
// with the same item ID. For instance, limit items to those that have a Clients relationship
// field with "Acme Co." client selected as the client.
$sql .= <<<SQL
AND items.id IN ( /* Where Item Belongs to Same Owner */
SELECT item_id
FROM item_values
JOIN fields ON fields.id = item_values.field_id
JOIN items ON items.id = item_values.item_id AND item_values.ver = items.ver
JOIN item_types ON item_types.id = items.item_type_id
WHERE item_values.value = ?
AND fields.field_type = "Relationship"
AND fields.field_item_type_id = ?)
SQL;
$params[] = $restrictedUserOwnerItemId; // Example: 3 -- CLIENT ID
$params[] = $restrictedUserOwnerItemType; // Example: 10 -- CLIENTS
$sql .= "AND ";
} else {
$sql .= "WHERE ";
}
$sql .= "items.account_id = ? AND items.id = ? ";
$params[] = $accountId;
$params[] = $itemId;
// Get raw records
$rsm = $this->standardResultSetMapping();
// $this->logger->debug($sql);
// $this->logger->debug(print_r($params, true));
echo $sql;
$query = $this->getEntityManager()->createNativeQuery($sql, $rsm);
$query->setParameters($params);
// Wake up the entities
$result = array();
foreach($query->getResult() as $row) {
$row->initServiceEntity($this->logger, $this, $this->itemValues, $this->fields);
$result[] = $row;
}
if(!$result || count($result) == 0)
throw new \InvalidArgumentException("Item could not be located for Item #".$itemId.". You may not have permission to view this item or it may not exist.");
else
{
return $result[0];
}
}
private function standardResultSetMapping()
{
$rsm = new ResultSetMapping();
// Class, Table
$rsm->addEntityResult('\AppBundle\Entity\Item', 'items');
$rsm->addEntityResult('\AppBundle\Entity\ItemType', 'item_types');
// Table, Column, Property
$rsm->addFieldResult('items', 'id', 'id');
$rsm->addFieldResult('items', 'account_id', 'accountId');
//$rsm->addFieldResult('items', 'item_type_id', 'itemTypeId');
$rsm->addFieldResult('items', 'field_count', 'fieldCount');
$rsm->addFieldResult('items', 'ver', 'ver');
$rsm->addFieldResult('items', 'title', 'title');
$rsm->addMetaResult('items', 'item_type_id', 'item_type_id', true);
$rsm->addFieldResult('item_types', 'item_type_id', 'id');
$rsm->addFieldResult('item_types', 'item_type_name', 'name');
$rsm->addFieldResult('item_types', 'item_type_plural_name', 'pluralName');
$rsm->addFieldResult('item_types', 'item_type_label', 'label');
$rsm->addFieldResult('item_types', 'item_type_plural_label','pluralLabel');
$rsm->addFieldResult('item_types', 'item_type_are_users', 'areUsers');
$rsm->addFieldResult('item_types', 'item_type_own_users', 'ownUsers');
return $rsm;
}
}
Item 实体被返回但总是有一个空的 itemType:
Item {#548 ▼
-id: 23
-accountId: 1
-itemType: null
-fieldCount: 4
-ver: 1451940837
-title: "New Item"
#fields: []
#itemValues: []
#cacheValues: []
#logger: Logger {#268 ▶}
#itemsRepository: ItemRepository {#349 ▶}
#itemValuesRepository: ItemValueRepository {#416 ▶}
#fieldsRepository: FieldRepository {#338 ▶}
#loaded: true
#changeCount: 0
}
item_types 数据
id account_id name plural_name label plural_label are_users own_users
31 1 task tasks Task Tasks 1 0
项目数据
id account_id item_type_id field_count ver title
23 1 31 4 1451940837 New Item
编辑 我认为我已将其缩小到 ResultSetMapping 配置。更新了上面的代码。结果现在返回两个不同的对象,但没有将它们连接起来(Item 的 itemType 仍然是 null):
object(AppBundle\Entity\Item)[560]
private 'id' => int 23
private 'accountId' => int 1
private 'itemType' => null
private 'fieldCount' => int 4
private 'ver' => int 1451940837
private 'title' => string 'New Item' (length=8)
protected 'fields' =>
array (size=0)
empty
protected 'itemValues' =>
array (size=0)
empty
protected 'cacheValues' =>
array (size=0)
empty
protected 'logger' => null
protected 'itemsRepository' => null
protected 'itemValuesRepository' => null
protected 'fieldsRepository' => null
protected 'loaded' => boolean false
protected 'changeCount' => int 0
object(AppBundle\Entity\ItemType)[507]
private 'id' => int 31
private 'accountId' => int 1
private 'name' => string 'task' (length=4)
private 'pluralName' => string 'tasks' (length=5)
private 'label' => string 'Task' (length=4)
private 'pluralLabel' => string 'Tasks' (length=5)
private 'areUsers' => boolean true
private 'ownUsers' => boolean false
所以现在的问题基本上是:
我如何设置 ResultSetMapping 以便它返回一个实体,所有加入的关联都完好无损?
最佳答案
Doctrine 的 documentation on Native SQL有一些很好的见解,它很清楚你的错误是什么。对您现有帖子的简短回答是,您应该为您的 ItemType 实体使用 addJoinedEntityResult() 而不是 addEntityResult()。
documentation for Entity Results状态:
An entity result describes an entity type that appears as a root element in the transformed result.
这意味着如果您在同一个映射中添加两个实体结果,您将获得当前看到的结果 - Item 和 ItemType 作为两个不同的对象返回.但是,您知道这两者是相关的,所以 Joined Entity Result更有意义:
A joined entity result describes an entity type that appears as a joined relationship element in the transformed result, attached to a (root) entity result.
要按原样直接修复代码,您需要更改
$rsm->addEntityResult('\AppBundle\Entity\ItemType', 'item_types');
为此:
$rsm->addJoinedEntityResult(
'\AppBundle\Entity\ItemType',
'item_types',
'items',
'itemType'
);
格式为addJoinedEntityResult($class, $alias, $parentAlias, $relation),因此您可以看到添加的第三个和第四个参数指向父别名和 <指向>指向>ItemType 的 strong>Item 类。
综上所述,我认为你把它弄得太复杂了,使用 ResultSetMappingBuilder 可以大大简化你的代码.这可以自动将字段映射到它们的等效 SQL 列,然后如果您更改了字段的名称或数据库中的列的名称,您将不必挖掘所有代码来更新映射.
因此,无需调用复杂的 standardResultSetMapping() 函数,您可以简单地执行以下操作:
$rsm = new ResultSetMappingBuilder($this->_em);
$rsm->addRootEntityFromClassMetadata('AppBundle\Entity\Item', 'items');
$rsm->addJoinedEntityFromClassMetadata('AppBundle\Entity\ItemType', 'item_types', 'items', 'itemType',
['id' => 'item_type_id',
'account_id' => 'item_type_id',
'name' => 'item_type_name',
'plural_name' => 'item_type_plural_name',
'label' => 'item_type_label',
'plural_label' => 'item_type_plural_label',
'are_users' => 'item_type_are_users',
'own_users' => 'item_type_own_users']
);
有了它,您就可以消除冗余代码,使其不易出错、更易于测试并自动处理对您的实体和数据库的更新。第二次调用显示您仍然可以传递一组重命名的列。
关于php - 如何为关联/外键设置 ResultSetMapping [是为什么 native 查询返回的关联始终为空],我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/34617942/
类classAprivatedeffooputs:fooendpublicdefbarputs:barendprivatedefzimputs:zimendprotecteddefdibputs:dibendendA的实例a=A.new测试a.foorescueputs:faila.barrescueputs:faila.zimrescueputs:faila.dibrescueputs:faila.gazrescueputs:fail测试输出failbarfailfailfail.发送测试[:foo,:bar,:zim,:dib,:gaz].each{|m|a.send(m)resc
我有一个模型:classItem项目有一个属性“商店”基于存储的值,我希望Item对象对特定方法具有不同的行为。Rails中是否有针对此的通用设计模式?如果方法中没有大的if-else语句,这是如何干净利落地完成的? 最佳答案 通常通过Single-TableInheritance. 关于ruby-on-rails-Rails-子类化模型的设计模式是什么?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.co
我正在使用的第三方API的文档状态:"[O]urAPIonlyacceptspaddedBase64encodedstrings."什么是“填充的Base64编码字符串”以及如何在Ruby中生成它们。下面的代码是我第一次尝试创建转换为Base64的JSON格式数据。xa=Base64.encode64(a.to_json) 最佳答案 他们说的padding其实就是Base64本身的一部分。它是末尾的“=”和“==”。Base64将3个字节的数据包编码为4个编码字符。所以如果你的输入数据有长度n和n%3=1=>"=="末尾用于填充n%
我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i
为什么4.1%2返回0.0999999999999996?但是4.2%2==0.2。 最佳答案 参见此处:WhatEveryProgrammerShouldKnowAboutFloating-PointArithmetic实数是无限的。计算机使用的位数有限(今天是32位、64位)。因此计算机进行的浮点运算不能代表所有的实数。0.1是这些数字之一。请注意,这不是与Ruby相关的问题,而是与所有编程语言相关的问题,因为它来自计算机表示实数的方式。 关于ruby-为什么4.1%2使用Ruby返
我希望我的UserPrice模型的属性在它们为空或不验证数值时默认为0。这些属性是tax_rate、shipping_cost和price。classCreateUserPrices8,:scale=>2t.decimal:tax_rate,:precision=>8,:scale=>2t.decimal:shipping_cost,:precision=>8,:scale=>2endendend起初,我将所有3列的:default=>0放在表格中,但我不想要这样,因为它已经填充了字段,我想使用占位符。这是我的UserPrice模型:classUserPrice回答before_val
我刚刚为fedora安装了emacs。我想用emacs编写ruby。为ruby提供代码提示、代码完成类型功能所需的工具、扩展是什么? 最佳答案 ruby-mode已经包含在Emacs23之后的版本中。不过,它也可以通过ELPA获得。您可能感兴趣的其他一些事情是集成RVM、feature-mode(Cucumber)、rspec-mode、ruby-electric、inf-ruby、rinari(用于Rails)等。这是我当前用于Ruby开发的Emacs配置:https://github.com/citizen428/emacs
我正在使用Sequel构建一个愿望list系统。我有一个wishlists和itemstable和一个items_wishlists连接表(该名称是续集选择的名称)。items_wishlists表还有一个用于facebookid的额外列(因此我可以存储opengraph操作),这是一个NOTNULL列。我还有Wishlist和Item具有续集many_to_many关联的模型已建立。Wishlist类也有:selectmany_to_many关联的选项设置为select:[:items.*,:items_wishlists__facebook_action_id].有没有一种方法可以
它不等于主线程的binding,这个toplevel作用域是什么?此作用域与主线程中的binding有何不同?>ruby-e'putsTOPLEVEL_BINDING===binding'false 最佳答案 事实是,TOPLEVEL_BINDING始终引用Binding的预定义全局实例,而Kernel#binding创建的新实例>Binding每次封装当前执行上下文。在顶层,它们都包含相同的绑定(bind),但它们不是同一个对象,您无法使用==或===测试它们的绑定(bind)相等性。putsTOPLEVEL_BINDINGput
我可以得到Infinity和NaNn=9.0/0#=>Infinityn.class#=>Floatm=0/0.0#=>NaNm.class#=>Float但是当我想直接访问Infinity或NaN时:Infinity#=>uninitializedconstantInfinity(NameError)NaN#=>uninitializedconstantNaN(NameError)什么是Infinity和NaN?它们是对象、关键字还是其他东西? 最佳答案 您看到打印为Infinity和NaN的只是Float类的两个特殊实例的字符串