- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
我在 Doctrine Symfony2 中有实体:用户、 channel 、视频和评论;用户可以报告其中之一。我用这些字段设计了 Report 实体:
我如何引用报告的实体 ?? 因为所有报告的字段对于所有实体都是相似的我只想为报告使用一个表并将这些字段添加到报告实体:
这是最佳实践还是我应该为每种报告创建单独的表格??
编辑:基于@Alex 的回答,我改进了 Report 类并添加了这些方法:
setEntity($entity){
if ($obj instanceof Video){
$this->referenceEntityName = 'Video';
$this->setVideo();
}
elseif($obj instanceof Comment){
$this->referenceEntityName == 'Comment'
$this->setComment();
}
//...
}
getEntity(){
if($this->referenceEntityName == 'Video'){
$this->getVideo()
}// ifelse statements for other entities ...
}
我现在有 4 个关系,每个实例只使用其中一个,是不是有点乱!?再说一次,这是最佳实践还是我应该做些其他事情?如果我想使用FormBuilder类,有什么问题吗??
最佳答案
在一个简单的解决方案中,例如您只有用户(而不是视频、评论和 channel ),解决方案会很简单;每个用户可以有多个报告,每个报告必须只属于一个用户。这是一对多的关系——一个用户有很多报告。在 Symfony 2 和 Doctrine 中,这将被建模为:
// src/Acme/DemoBundle/Entity/User.php
// ...
use Doctrine\Common\Collections\ArrayCollection;
class User
{
// ...
/**
* @ORM\OneToMany(targetEntity="Report", mappedBy="user")
*/
protected $reports;
public function __construct()
{
$this->reports = new ArrayCollection();
}
// ...
}
和
// src/Acme/DemoBundle/Entity/Report.php
// ...
class Report
{
// ...
/**
* @ORM\ManyToOne(targetEntity="User", inversedBy="reports")
* @ORM\JoinColumn(name="user_id", referencedColumnName="id")
*/
protected $user;
// ...
}
在这种情况下,要创建报告并将其与用户相关联,我们将:
// get the User the Report will belong to
$user = $em->getRepository('AcmeDemoBundle:User')->find(1);
// create the Report
$report = new Report();
// add the User to the Report
$report->setUser($user);
// then persist it, etc ...
请注意,setUser() 方法可用是因为运行了控制台命令以自动生成它们。强烈推荐这样做,因为它为您创建了必要的类型提示。对于 Symfony 2.5 之前的安装,命令是:
php app/console doctrine:generate:entities Acme
>= 2.5安装,命令为:
php bin/console doctrine:generate:entities Acme
您的要求使这个简单的示例有些复杂,因为报告也可以属于评论和视频等。为了便于示例,我们将这些东西称为实体。一个糟糕的方法是简单地向报表添加 3 个新属性,每个属性对应一个新实体,然后为实体添加 3 个新的 setter 方法。这很糟糕,原因有两个:一个报表将只属于一个实体,因此 3 个属性和设置方法将永远不会用于每个报表实体。其次,如果您向业务模型添加新实体或删除实体,则需要编辑报告实体以及数据库架构。
一种更好的方法是在您的报表中简单地拥有一个属性和设置方法,它可以应用于您的所有实体。因此,我们可以调用 setEntity
,而不是调用 setUser
,并让它接受 4 种方法中的任何一种。考虑到这种方法,让我们回顾一下第一个示例,并且注意为 setUser
方法生成的函数签名中的类型提示:
public function setUser(Acme\DemoBundle\Entity\User $user)
看到它需要是 Acme\DemoBundle\Entity\User
类型。我们如何克服这个问题,让它接受 4 个实体中的任何一个?解决方案是让所有实体都从父类派生。然后在基类做函数类型提示:
public function setUser(Acme\DemoBundle\Entity\Base $entity)
基类将包含所有公共(public)元素,特别是“名称”,以及作为报告的数组集合:
// src/Acme/DemoBundle/Entity/Base.php
// ...
use Doctrine\Common\Collections\ArrayCollection;
class Base
{
// ...
/**
* @ORM\Column(name="name", type="text")
*/
protected $name
/**
* @ORM\OneToMany(targetEntity="Report", mappedBy="baseEntity")
*/
protected $reports;
public function __construct()
{
$this->reports = new ArrayCollection();
}
// ...
}
然后对于每个 child ,例如用户和视频:
// src/Acme/DemoBundle/Entity/User.php
// ...
use AcmeDemoBundle\Entity\Base;
class User extends Base
{
/**
* @ORM\Column(name="firstname", type="text")
*/
protected $firstName;
// ...
}
和视频
// src/Acme/DemoBundle/Entity/Video.php
// ...
use AcmeDemoBundle\Entity\Base;
class Video extends Base
{
/**
* @ORM\Column(name="title", type="text")
*/
protected $title;
// ...
并更改我们的报告实体:
// src/Acme/DemoBundle/Entity/Report.php
// ...
class Report
{
// ...
/**
* @ORM\ManyToOne(targetEntity="Base", inversedBy="reports")
* @ORM\JoinColumn(name="base_id", referencedColumnName="id")
*/
protected $baseEntity;
// ...
}
记得运行 doctrine 命令来生成 setBaseEntity 方法。当您这样做时,请注意它现在将接受任何派生自 Base
的类然后,以对视频进行报告为例,我们获取视频、创建报告并将视频添加到报告中:
$video = // get the video you want
$report = new Report();
$report->setBaseEntity($video);
要检索属于评论的所有报告,我们获取评论,并获取报告:
$video = // get the video you want
$reports = $video->getReports();
foreach($reports as $report){
$reportText = $report->getText(); // assuming the Report has a `text` field
}
更新:
这些Entities之间的继承关系可以在数据库中用Doctrine using Single Table Inheritance建模:
/**
* @ORM\Entity
* @ORM\Table(name="base_entities")
* @ORM\InheritanceType("SINGLE_TYPE")
* @ORM\Discriminator(name="entity_type", type="string")
* @ORM\DiscriminatorMap({"user" = "User", "comment" = "Comment", "video" = "Video", "channel" = "Channel"})
*/
关于php - symfony2 中可选的多个 oneToMany 关系,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/24839160/
我开始从事一个用 Symfony 2.8 编写的大型项目。将整个项目升级到 SF 3 需要数百小时,现在还不可能。我想到了一个想法,将 symfony/symfony 包解压到它替换的单个包中(com
我在提交表单后使用 FOSUserEvents,但订阅者调用了两次。 这样我的验证码第一次有效第二次无效 这是我的代码 router = $router; $this->request
我有以下路线: blog_show: path: /test/123 defaults: { _controller: TotalcanBravofillBundle:Te
我是测试新手。我想测试我的功能。我已经成功安装了 phpUnit。我在互联网上查看了许多教程。但我无法获得有关测试的正确信息。这是我的功能代码: public function loginAction
我正在尝试重现 facebook batch requests 的行为在他们的图形 api 上运行。 所以我认为最简单的解决方案是在 Controller 上向我的应用程序发出几个请求,例如: pub
在 Symfony Progress Bar documentation有一个超酷酒吧的示例图像。不幸的是,看起来文档的其余部分没有解释如何获得这样的结果。 这是图片,以防您错过: 我怎么才能得到它?
我使用Finder发送假脱机电子邮件,但是自动名称生成器将点放在文件名中,有时它们出现在文件的开头。 查找程序似乎无法获取具有该名称的文件-那些文件被隐藏了……有人经历过这种行为吗?有什么建议如何使用
我正在尝试进行 LDAP 身份验证,我目前遇到此类错误: ServiceNotFoundException: The service "security.firewall.map.context.ma
有没有办法验证和检查集合数组是否为空。我已经尝试过: /** * @Assert\NotBlank() * @Assert\Length( min = 1) */ protected $work
使用Smyfony2和Doctrin2,可以使用以下示例创建数据固定装置:http://symfony.com/doc/current/bundles/DoctrineFixturesBundle/i
我看到在大多数Symfony 2示例中,例如,不存在记录时,Symfony 2会引发异常。我认为这种方法对最终用户不友好。为什么有人更喜欢引发异常而不在Flashbag上添加一些错误消息? 最佳答案
我对项目中的以下服务有疑问: app.security.guardAuthenticatorLoginPassword: class: AppBundle\Security\LoginPa
symfony缓存和登录Docker容器存在问题。 Web服务器从www-data用户和组执行,当我使用docker上安装的php从docker容器中清除symfony缓存时,它从root执行。 因此
我想了解 symfony 中的服务 我已阅读http://symfony.com/doc/2.3/book/service_container.html#creating-configuring-se
因为我对 Symfony 和 Doctrine 还很陌生,所以我有一个可能很愚蠢的问题;-) 有人可以用简单的词语向我解释集合(尤其是实体中的ArrayCollections)吗?它是什么以及何时以及
我收到了这个表格: {{ form_start(form) }} {{ form_end(form) }} 我想检查用户是否登录,我这样做了: {% if is_g
我的网站已准备好部署,我正在尝试将其设置为在线。 一些信息: 主持人是 OVH。 它不允许 SSH,我必须使用 FTP 发送文件。也没有命令行。 我现在希望能够在子目录中设置网站:/www/test(
过去几个月以来,我一直在尝试与symfony合作。昨晚我自动删除了不需要的存储库。之后,我无法使用symfony命令创建新的symfony项目。当我在终端中运行Symfony new Security
In the environnement variable, then in system variable, I edited the path and added在环境变量中,然后在系统变量
我们有一个 Symfony 1.4 应用程序,想升级到 Symfony 4。是否有可能或者我们必须重新编程该应用程序? 我们询问了我们附近的一家软件公司,他们告诉我们必须重新编写应用程序。 最佳答案
我是一名优秀的程序员,十分优秀!