发布于 2015-08-27 16:46:37 | 131 次阅读 | 评论: 0 | 来源: 网络整理
One of the goals of bundles is to create discreet bundles of functionality that do not have many (if any) dependencies, allowing you to use that functionality in other applications without including unnecessary items.
Doctrine 2.2 includes a new utility called the ResolveTargetEntityListener
,
that functions by intercepting certain calls inside Doctrine and rewriting
targetEntity
parameters in your metadata mapping at runtime. It means that
in your bundle you are able to use an interface or abstract class in your
mappings and expect correct mapping to a concrete entity at runtime.
This functionality allows you to define relationships between different entities without making them hard dependencies.
Suppose you have an InvoiceBundle which provides invoicing functionality and a CustomerBundle that contains customer management tools. You want to keep these separated, because they can be used in other systems without each other, but for your application you want to use them together.
In this case, you have an Invoice
entity with a relationship to a
non-existent object, an InvoiceSubjectInterface
. The goal is to get
the ResolveTargetEntityListener
to replace any mention of the interface
with a real object that implements that interface.
This article uses the following two basic entities (which are incomplete for
brevity) to explain how to set up and use the ResolveTargetEntityListener
.
A Customer entity:
// src/Acme/AppBundle/Entity/Customer.php
namespace AcmeAppBundleEntity;
use DoctrineORMMapping as ORM;
use AcmeCustomerBundleEntityCustomer as BaseCustomer;
use AcmeInvoiceBundleModelInvoiceSubjectInterface;
/**
* @ORMEntity
* @ORMTable(name="customer")
*/
class Customer extends BaseCustomer implements InvoiceSubjectInterface
{
// In this example, any methods defined in the InvoiceSubjectInterface
// are already implemented in the BaseCustomer
}
An Invoice entity:
// src/Acme/InvoiceBundle/Entity/Invoice.php
namespace AcmeInvoiceBundleEntity;
use DoctrineORMMapping AS ORM;
use AcmeInvoiceBundleModelInvoiceSubjectInterface;
/**
* Represents an Invoice.
*
* @ORMEntity
* @ORMTable(name="invoice")
*/
class Invoice
{
/**
* @ORMManyToOne(targetEntity="AcmeInvoiceBundleModelInvoiceSubjectInterface")
* @var InvoiceSubjectInterface
*/
protected $subject;
}
An InvoiceSubjectInterface:
// src/Acme/InvoiceBundle/Model/InvoiceSubjectInterface.php
namespace AcmeInvoiceBundleModel;
/**
* An interface that the invoice Subject object should implement.
* In most circumstances, only a single object should implement
* this interface as the ResolveTargetEntityListener can only
* change the target to a single object.
*/
interface InvoiceSubjectInterface
{
// List any additional methods that your InvoiceBundle
// will need to access on the subject so that you can
// be sure that you have access to those methods.
/**
* @return string
*/
public function getName();
}
Next, you need to configure the listener, which tells the DoctrineBundle about the replacement:
# app/config/config.yml
doctrine:
# ...
orm:
# ...
resolve_target_entities:
AcmeInvoiceBundleModelInvoiceSubjectInterface: AcmeAppBundleEntityCustomer
<!-- app/config/config.xml -->
<container xmlns="http://symfony.com/schema/dic/services"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:doctrine="http://symfony.com/schema/dic/doctrine"
xsi:schemaLocation="http://symfony.com/schema/dic/services http://symfony.com/schema/dic/services/services-1.0.xsd
http://symfony.com/schema/dic/doctrine http://symfony.com/schema/dic/doctrine/doctrine-1.0.xsd">
<doctrine:config>
<doctrine:orm>
<!-- ... -->
<doctrine:resolve-target-entity interface="AcmeInvoiceBundleModelInvoiceSubjectInterface">AcmeAppBundleEntityCustomer</doctrine:resolve-target-entity>
</doctrine:orm>
</doctrine:config>
</container>
// app/config/config.php
$container->loadFromExtension('doctrine', array(
'orm' => array(
// ...
'resolve_target_entities' => array(
'AcmeInvoiceBundleModelInvoiceSubjectInterface' => 'AcmeAppBundleEntityCustomer',
),
),
));
With the ResolveTargetEntityListener
, you are able to decouple your
bundles, keeping them usable by themselves, but still being able to
define relationships between different objects. By using this method,
your bundles will end up being easier to maintain independently.