Added more information about flushing semantics.
This commit is contained in:
parent
400be2a3e6
commit
0428774ec8
@ -1,4 +1,3 @@
|
|||||||
++ Understanding
|
|
||||||
|
|
||||||
In this chapter we will help you understand the `EntityManager` and the `UnitOfWork`.
|
In this chapter we will help you understand the `EntityManager` and the `UnitOfWork`.
|
||||||
A Unit of Work is similar to an object-level transaction. A new Unit of Work is
|
A Unit of Work is similar to an object-level transaction. A new Unit of Work is
|
||||||
@ -8,49 +7,7 @@ implicity started when an EntityManager is initially created or after
|
|||||||
|
|
||||||
A Unit of Work can be manually closed by calling EntityManager#close(). Any
|
A Unit of Work can be manually closed by calling EntityManager#close(). Any
|
||||||
changes to objects within this Unit of Work that have not yet been persisted
|
changes to objects within this Unit of Work that have not yet been persisted
|
||||||
are lost.
|
are lost.
|
||||||
|
|
||||||
+++ The size of a Unit of Work
|
|
||||||
|
|
||||||
The size of a Unit of Work mainly refers to the number of managed entities at
|
|
||||||
a particular point in time.
|
|
||||||
|
|
||||||
+++ The cost of flush()
|
|
||||||
|
|
||||||
How costly a flush operation is in terms of performance mainly depends on 2 factors:
|
|
||||||
|
|
||||||
* The size of your current Unit of Work
|
|
||||||
* The configured change tracking policies
|
|
||||||
|
|
||||||
You can get the size of your Unit of Work as follows:
|
|
||||||
|
|
||||||
[php]
|
|
||||||
$uowSize = $em->getUnitOfWork()->size();
|
|
||||||
|
|
||||||
The size represents the number of managed entities in the Unit of Work. This
|
|
||||||
size affects the performance of flush() operations due to change tracking
|
|
||||||
(see "Change Tracking Policies") and, of course, memory consumption, so you
|
|
||||||
may want to check it from time to time during development.
|
|
||||||
|
|
||||||
> **CAUTION**
|
|
||||||
> Do not invoke `flush` after every change to an entity or every single invocation of
|
|
||||||
> persist/remove/merge/... This is an anti-pattern and unnecessarily reduces the
|
|
||||||
> performance of your application. Instead, form units of work that operate on your objects
|
|
||||||
> and call `flush` when you are done. While serving a single HTTP request there should
|
|
||||||
> be usually no need for invoking `flush` more than 0-2 times.
|
|
||||||
|
|
||||||
+++ Direct access to a Unit of Work
|
|
||||||
|
|
||||||
You can get direct access to the Unit of Work by calling `EntityManager#getUnitOfWork()`.
|
|
||||||
This will return the UnitOfWork instance the EntityManager is currently using.
|
|
||||||
|
|
||||||
[php]
|
|
||||||
$uow = $em->getUnitOfWork();
|
|
||||||
|
|
||||||
> **NOTE**
|
|
||||||
> Directly manipulating a UnitOfWork is not recommended. When working directly with the
|
|
||||||
> UnitOfWork API, respect methods marked as INTERNAL by not using them and carefully read
|
|
||||||
> the API documentation.
|
|
||||||
|
|
||||||
++ Persisting entities
|
++ Persisting entities
|
||||||
|
|
||||||
@ -79,7 +36,7 @@ Example:
|
|||||||
> **CAUTION**
|
> **CAUTION**
|
||||||
> Generated entity identifiers / primary keys are guaranteed to be available after the
|
> Generated entity identifiers / primary keys are guaranteed to be available after the
|
||||||
> next successful flush operation that involves the entity in question.
|
> next successful flush operation that involves the entity in question.
|
||||||
> You can not reply on a generated identifier to be available directly after invoking `persist`.
|
> You can not rely on a generated identifier to be available directly after invoking `persist`.
|
||||||
> The inverse is also true. You can not rely on a generated identifier being not available
|
> The inverse is also true. You can not rely on a generated identifier being not available
|
||||||
> after a failed flush operation.
|
> after a failed flush operation.
|
||||||
|
|
||||||
@ -88,15 +45,12 @@ The semantics of the persist operation, applied on an entity X, are as follows:
|
|||||||
* If X is a new entity, it becomes managed. The entity X will be entered into the database as a result of the flush operation.
|
* If X is a new entity, it becomes managed. The entity X will be entered into the database as a result of the flush operation.
|
||||||
* If X is a preexisting managed entity, it is ignored by the persist operation. However, the persist operation is cascaded to entities referenced by X, if the relationships from X to these other entities are mapped with cascade=PERSIST or cascade=ALL (see "Transitive Persistence").
|
* If X is a preexisting managed entity, it is ignored by the persist operation. However, the persist operation is cascaded to entities referenced by X, if the relationships from X to these other entities are mapped with cascade=PERSIST or cascade=ALL (see "Transitive Persistence").
|
||||||
* If X is a removed entity, it becomes managed.
|
* If X is a removed entity, it becomes managed.
|
||||||
* If X is a detached entity, the behavior is undefined.
|
* If X is a detached entity, an exception will be thrown on flush.
|
||||||
|
|
||||||
> **CAUTION**
|
|
||||||
> Do not pass detached entities to the persist operation.
|
|
||||||
|
|
||||||
|
|
||||||
++ Removing entities
|
++ Removing entities
|
||||||
|
|
||||||
An entity can be removed from persistent storage by passing it to the `EntityManager#remove($entity)` method. By applying the `remove` operation on some entity, that entity becomes REMOVED, which means that its persistent state will be deleted once `EntityManager#flush()` is invoked. The in-memory state of an entity is unaffected by the `remove` operation.
|
An entity can be removed from persistent storage by passing it to the `EntityManager#remove($entity)` method. By applying the `remove` operation on some entity, that entity becomes REMOVED, which means that its persistent state will be deleted once `EntityManager#flush()` is invoked.
|
||||||
|
|
||||||
> **CAUTION**
|
> **CAUTION**
|
||||||
> Just like `persist`, invoking `remove` on an entity does NOT cause an immediate SQL
|
> Just like `persist`, invoking `remove` on an entity does NOT cause an immediate SQL
|
||||||
@ -115,7 +69,9 @@ The semantics of the remove operation, applied to an entity X are as follows:
|
|||||||
* If X is a managed entity, the remove operation causes it to become removed. The remove operation is cascaded to entities referenced by X, if the relationships from X to these other entities is mapped with cascade=REMOVE or cascade=ALL (see "Transitive Persistence").
|
* If X is a managed entity, the remove operation causes it to become removed. The remove operation is cascaded to entities referenced by X, if the relationships from X to these other entities is mapped with cascade=REMOVE or cascade=ALL (see "Transitive Persistence").
|
||||||
* If X is a detached entity, an InvalidArgumentException will be thrown.
|
* If X is a detached entity, an InvalidArgumentException will be thrown.
|
||||||
* If X is a removed entity, it is ignored by the remove operation.
|
* If X is a removed entity, it is ignored by the remove operation.
|
||||||
* A removed entity X will be removed from the database as a result of the flush operation.
|
* A removed entity X will be removed from the database as a result of the flush operation.
|
||||||
|
|
||||||
|
After an entity has been removed its in-memory state is the same as before the removal, except for generated identifiers.
|
||||||
|
|
||||||
|
|
||||||
++ Detaching entities
|
++ Detaching entities
|
||||||
@ -340,6 +296,107 @@ Do not blindly apply cascade=all to all associations as it will unnecessarily
|
|||||||
degrade the performance of your application.
|
degrade the performance of your application.
|
||||||
|
|
||||||
|
|
||||||
|
++ Synchronization with the Database
|
||||||
|
|
||||||
|
The state of persistent entities is synchronized with the database on flush of an `EntityManager`
|
||||||
|
which commits the underlying `UnitOfWork`. The synchronization involves writing any updates to
|
||||||
|
persistent entities and their relationships to the database. Thereby bidirectional relationships
|
||||||
|
are persisted based on the references held by the owning side of the relationship as explained
|
||||||
|
in the Association Mapping chapter.
|
||||||
|
|
||||||
|
The flush operation applies to a managed entity with the following semantics:
|
||||||
|
|
||||||
|
* The entity itself is synchronized to the database, if it has changed.
|
||||||
|
|
||||||
|
For all (initialized) relationships of the entity the following semantics apply to each
|
||||||
|
associated entity X:
|
||||||
|
|
||||||
|
* If X is new and persist operations are configured to cascade on the relationship,
|
||||||
|
X will be persisted.
|
||||||
|
* If X is new and no persist operations are configured to cascade on the relationship,
|
||||||
|
an exception will be thrown as this indicates a programming error.
|
||||||
|
* If X is removed and persist operations are configured to cascade on the relationship,
|
||||||
|
an exception will be thrown as this indicates a programming error (X would be re-persisted by the cascade).
|
||||||
|
* If X is detached and persist operations are configured to cascade on the relationship,
|
||||||
|
an exception will be thrown (This is semantically the same as passing X to persist()).
|
||||||
|
|
||||||
|
The flush operation applies to a removed entity by deleting its persistent state from the database.
|
||||||
|
No cascade options are relevant for removed entities on flush.
|
||||||
|
|
||||||
|
+++ The size of a Unit of Work
|
||||||
|
|
||||||
|
The size of a Unit of Work mainly refers to the number of managed entities at
|
||||||
|
a particular point in time.
|
||||||
|
|
||||||
|
+++ The cost of flushing
|
||||||
|
|
||||||
|
How costly a flush operation is, mainly depends on two factors:
|
||||||
|
|
||||||
|
* The size of the EntityManager's current UnitOfWork.
|
||||||
|
* The configured change tracking policies
|
||||||
|
|
||||||
|
You can get the size of a UnitOfWork as follows:
|
||||||
|
|
||||||
|
[php]
|
||||||
|
$uowSize = $em->getUnitOfWork()->size();
|
||||||
|
|
||||||
|
The size represents the number of managed entities in the Unit of Work. This
|
||||||
|
size affects the performance of flush() operations due to change tracking
|
||||||
|
(see "Change Tracking Policies") and, of course, memory consumption, so you
|
||||||
|
may want to check it from time to time during development.
|
||||||
|
|
||||||
|
> **CAUTION**
|
||||||
|
> Do not invoke `flush` after every change to an entity or every single invocation of
|
||||||
|
> persist/remove/merge/... This is an anti-pattern and unnecessarily reduces the
|
||||||
|
> performance of your application. Instead, form units of work that operate on your objects
|
||||||
|
> and call `flush` when you are done. While serving a single HTTP request there should
|
||||||
|
> be usually no need for invoking `flush` more than 0-2 times.
|
||||||
|
|
||||||
|
+++ Direct access to a Unit of Work
|
||||||
|
|
||||||
|
You can get direct access to the Unit of Work by calling `EntityManager#getUnitOfWork()`.
|
||||||
|
This will return the UnitOfWork instance the EntityManager is currently using.
|
||||||
|
|
||||||
|
[php]
|
||||||
|
$uow = $em->getUnitOfWork();
|
||||||
|
|
||||||
|
> **NOTE**
|
||||||
|
> Directly manipulating a UnitOfWork is not recommended. When working directly with the
|
||||||
|
> UnitOfWork API, respect methods marked as INTERNAL by not using them and carefully read
|
||||||
|
> the API documentation.
|
||||||
|
|
||||||
|
+++ Entity State
|
||||||
|
|
||||||
|
As outlined in the architecture overview an entity can be in one of four possible states:
|
||||||
|
NEW, MANAGED, REMOVED, DETACHED. If you explicitly need to find out what the current state
|
||||||
|
of an entity is in the context of a certain `EntityManager` you can ask the underlying
|
||||||
|
`UnitOfWork`:
|
||||||
|
|
||||||
|
[php]
|
||||||
|
switch ($em->getUnitOfWork()->getEntityState($entity)) {
|
||||||
|
case UnitOfWork::MANAGED:
|
||||||
|
...
|
||||||
|
case UnitOfWork::REMOVED:
|
||||||
|
...
|
||||||
|
case UnitOfWork::DETACHED:
|
||||||
|
...
|
||||||
|
case UnitOfWork::NEW:
|
||||||
|
...
|
||||||
|
}
|
||||||
|
|
||||||
|
An entity is in MANAGED state if it is associated with an `EntityManager` and it is not REMOVED.
|
||||||
|
|
||||||
|
An entity is in REMOVED state after it has been passed to `EntityManager#remove()` until the
|
||||||
|
next flush operation of the same EntityManager. A REMOVED entity is still associated with an
|
||||||
|
`EntityManager` until the next flush operation.
|
||||||
|
|
||||||
|
An entity is in DETACHED state if it has persistent state and identity but is currently not
|
||||||
|
associated with an `EntityManager`.
|
||||||
|
|
||||||
|
An entity is in NEW state if has no persistent state and identity and is not associated with an
|
||||||
|
`EntityManager` (for example those just created via the "new" operator).
|
||||||
|
|
||||||
|
|
||||||
++ Querying
|
++ Querying
|
||||||
|
|
||||||
Doctrine 2 provides the following ways, in increasing level of power and flexibility, to query for persistent objects. You should always start with the simplest one that suits your needs.
|
Doctrine 2 provides the following ways, in increasing level of power and flexibility, to query for persistent objects. You should always start with the simplest one that suits your needs.
|
||||||
|
Loading…
Reference in New Issue
Block a user