2013-11-12 23:49:25 +01:00
|
|
|
Separating Concerns using Embeddables
|
|
|
|
-------------------------------------
|
|
|
|
|
|
|
|
Embeddables are classes which are not entities themself, but are embedded
|
|
|
|
in entities and can also be queried in DQL. You'll mostly want to use them
|
|
|
|
to reduce duplication or separating concerns.
|
|
|
|
|
|
|
|
For the purposes of this tutorial, we will assume that you have a ``User``
|
|
|
|
class in your application and you would like to store an address in
|
|
|
|
the ``User`` class. We will model the ``Address`` class as an embeddable
|
|
|
|
instead of simply adding the respective columns to the ``User`` class.
|
|
|
|
|
|
|
|
.. configuration-block::
|
|
|
|
|
|
|
|
.. code-block:: php
|
|
|
|
|
|
|
|
<?php
|
|
|
|
|
|
|
|
/** @Entity */
|
|
|
|
class User
|
|
|
|
{
|
|
|
|
/** @Embedded(class = "Address") */
|
|
|
|
private $address;
|
|
|
|
}
|
|
|
|
|
|
|
|
/** @Embeddable */
|
|
|
|
class Address
|
|
|
|
{
|
|
|
|
/** @Column(type = "string") */
|
|
|
|
private $street;
|
|
|
|
|
|
|
|
/** @Column(type = "string") */
|
|
|
|
private $postalCode;
|
|
|
|
|
|
|
|
/** @Column(type = "string") */
|
|
|
|
private $city;
|
|
|
|
|
|
|
|
/** @Column(type = "string") */
|
|
|
|
private $country;
|
|
|
|
}
|
|
|
|
|
|
|
|
.. code-block:: xml
|
|
|
|
|
|
|
|
<doctrine-mapping>
|
|
|
|
<entity name="User">
|
2013-11-13 00:03:21 +01:00
|
|
|
<embedded name="address" class="Address" />
|
2013-11-12 23:49:25 +01:00
|
|
|
</entity>
|
|
|
|
|
|
|
|
<embeddable name="Address">
|
2013-11-13 00:03:21 +01:00
|
|
|
<field name="street" type="string" />
|
|
|
|
<field name="postalCode" type="string" />
|
|
|
|
<field name="city" type="string" />
|
|
|
|
<field name="country" type="string" />
|
2013-11-12 23:49:25 +01:00
|
|
|
</embeddable>
|
|
|
|
</doctrine-mapping>
|
|
|
|
|
|
|
|
.. code-block:: yaml
|
|
|
|
|
|
|
|
User:
|
|
|
|
type: entity
|
|
|
|
embedded:
|
|
|
|
address:
|
|
|
|
class: Address
|
|
|
|
|
|
|
|
Address:
|
|
|
|
type: embeddable
|
2013-11-13 00:03:21 +01:00
|
|
|
fields:
|
2013-11-12 23:49:25 +01:00
|
|
|
street: { type: string }
|
|
|
|
postalCode: { type: string }
|
|
|
|
city: { type: string }
|
|
|
|
country: { type: string }
|
|
|
|
|
|
|
|
In terms of your database schema, Doctrine will automatically inline all
|
|
|
|
columns from the ``Address`` class into the table of the ``User`` class,
|
|
|
|
just as if you had declared them directly there.
|
|
|
|
|
2014-09-01 12:48:10 +01:00
|
|
|
Column Prefixing
|
|
|
|
----------------
|
|
|
|
|
2014-09-03 11:13:28 +01:00
|
|
|
By default, Doctrine names your columns by prefixing them, using the value
|
|
|
|
object name.
|
2014-09-01 12:48:10 +01:00
|
|
|
|
2014-09-03 11:13:28 +01:00
|
|
|
Following the example above, your columns would be named as ``address_street``,
|
|
|
|
``address_postalCode``...
|
|
|
|
|
|
|
|
You can change this behaviour to meet your needs by changing the
|
|
|
|
``columnPrefix`` attribute in the ``@Embeddable`` notation.
|
|
|
|
|
|
|
|
The following example shows you how to set your prefix to ``myPrefix_``:
|
2014-09-01 12:48:10 +01:00
|
|
|
|
|
|
|
.. configuration-block::
|
|
|
|
|
|
|
|
.. code-block:: php
|
|
|
|
|
|
|
|
<?php
|
|
|
|
|
|
|
|
/** @Entity */
|
|
|
|
class User
|
|
|
|
{
|
2014-09-03 11:13:28 +01:00
|
|
|
/** @Embedded(class = "Address", columnPrefix = "myPrefix_") */
|
2014-09-01 12:48:10 +01:00
|
|
|
private $address;
|
|
|
|
}
|
|
|
|
|
|
|
|
.. code-block:: xml
|
|
|
|
|
|
|
|
<entity name="User">
|
2014-09-03 11:13:28 +01:00
|
|
|
<embedded name="address" class="Address" column-prefix="myPrefix_" />
|
2014-09-01 12:48:10 +01:00
|
|
|
</entity>
|
|
|
|
|
|
|
|
.. code-block:: yaml
|
|
|
|
|
|
|
|
User:
|
|
|
|
type: entity
|
|
|
|
embedded:
|
|
|
|
address:
|
|
|
|
class: Address
|
2014-09-03 11:13:28 +01:00
|
|
|
columnPrefix: myPrefix_
|
2014-09-01 12:48:10 +01:00
|
|
|
|
2014-09-03 11:13:28 +01:00
|
|
|
To have Doctrine drop the prefix and use the value object's property name
|
|
|
|
directly, set ``columnPrefix=false`` (not yet supported with XML configuration):
|
|
|
|
|
|
|
|
.. configuration-block::
|
|
|
|
|
|
|
|
.. code-block:: php
|
|
|
|
|
|
|
|
<?php
|
|
|
|
|
|
|
|
/** @Entity */
|
|
|
|
class User
|
|
|
|
{
|
|
|
|
/** @Embedded(class = "Address", columnPrefix = false) */
|
|
|
|
private $address;
|
|
|
|
}
|
|
|
|
|
|
|
|
.. code-block:: yaml
|
2014-09-01 12:48:10 +01:00
|
|
|
|
|
|
|
User:
|
|
|
|
type: entity
|
|
|
|
embedded:
|
|
|
|
address:
|
|
|
|
class: Address
|
|
|
|
columnPrefix: false
|
|
|
|
|
|
|
|
|
|
|
|
DQL
|
|
|
|
---
|
|
|
|
|
2013-11-12 23:49:25 +01:00
|
|
|
You can also use mapped fields of embedded classes in DQL queries, just
|
|
|
|
as if they were declared in the ``User`` class:
|
|
|
|
|
|
|
|
.. code-block:: sql
|
|
|
|
|
|
|
|
SELECT u FROM User u WHERE u.address.city = :myCity
|
|
|
|
|