1
0
mirror of synced 2024-12-15 15:46:02 +03:00
doctrine2/manual/en/architecture.txt

82 lines
5.2 KiB
Plaintext
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

This chapter gives an overview of the overall architecture, terminology and constraints of
Doctrine 2. It is recommended to read this chapter carefully.
## Entities
An entity is a lightweight, persistent domain object. An entity can be any regular
PHP class observing the following restrictions:
* An entity class must not be final or contain final methods.
* All persistent properties/field of any entity class should always be private or protected, otherwise lazy-loading might not work as expected.
* An entity class must not implement `__clone` or [do so safely](http://www.doctrine-project.org/documentation/cookbook/2_0/en/implementing-wakeup-or-clone).
* An entity class must not implement `__wakeup` or [do so safely](http://www.doctrine-project.org/documentation/cookbook/2_0/en/implementing-wakeup-or-clone).
Also consider implementing [Serializable](http://de3.php.net/manual/en/class.serializable.php]) instead.
* Any two entity classes in a class hierarchy that inherit directly or indirectly from one another must not have a mapped property with the same name.
That is, if B inherits from A then B must not have a mapped field with the same name as an already mapped field that is inherited from A.
Entities support inheritance, polymorphic associations, and polymorphic queries.
Both abstract and concrete classes can be entities. Entities may extend non-entity
classes as well as entity classes, and non-entity classes may extend entity classes.
> **TIP**
> The constructor of an entity is only ever invoked when *you* construct a new instance
> with the *new* keyword. Doctrine never calls entity constructors, thus you are free to use
> them as you wish and even have it require arguments of any type.
### Entity states
An entity instance can be characterized as being NEW, MANAGED, DETACHED or REMOVED.
* A NEW entity instance has no persistent identity, and is not yet associated with an EntityManager and a UnitOfWork (i.e. those just created with the "new" operator).
* A MANAGED entity instance is an instance with a persistent identity that is associated with an EntityManager and whose persistence is thus managed.
* A DETACHED entity instance is an instance with a persistent identity that is not (or no longer) associated with an EntityManager and a UnitOfWork.
* A REMOVED entity instance is an instance with a persistent identity, associated with an EntityManager, that will be removed from the database upon transaction commit.
### Persistent fields
The persistent state of an entity is represented by instance variables. An
instance variable must be directly accessed only from within the methods of the
entity by the entity instance itself. Instance variables must not be accessed by
clients of the entity. The state of the entity is available to clients only through
the entitys methods, i.e. accessor methods (getter/setter methods) or other
business methods.
Collection-valued persistent fields and properties must be defined in terms of
the `Doctrine\Common\Collections\Collection` interface. The collection
implementation type may be used by the application to initialize fields or
properties before the entity is made persistent. Once the entity becomes
managed (or detached), subsequent access must be through the interface type.
### Serializing entities
Serializing entities can be problematic and is not really recommended, at least not as long as an
entity instance still holds references to proxy objects or is still managed by an EntityManager.
If you intend to serialize (and unserialize) entity instances that still hold references to proxy objects
you may run into problems with private properties because of technical limitations.
Proxy objects implement `__sleep` and it is not possible for `__sleep` to return names of
private properties in parent classes. On the other hand it is not a solution for proxy objects
to implement `Serializable` because Serializable does not work well with any potential cyclic
object references (at least we did not find a way yet, if you did, please contact us).
## The EntityManager
The `EntityManager` class is a central access point to the ORM functionality
provided by Doctrine 2. The `EntityManager` API is used to manage the persistence
of your objects and to query for persistent objects.
### Transactional write-behind
An `EntityManager` and the underlying `UnitOfWork` employ a strategy called
"transactional write-behind" that delays the execution of SQL statements in
order to execute them in the most efficient way and to execute them at the end
of a transaction so that all write locks are quickly released. You should see
Doctrine as a tool to synchronize your in-memory objects with the database in
well defined units of work. Work with your objects and modify them as usual and
when you're done call `EntityManager#flush()` to make your changes persistent.
### The Unit of Work
Internally an `EntityManager` uses a `UnitOfWork`, which is a typical
implementation of the [Unit of Work pattern](http://martinfowler.com/eaaCatalog/unitOfWork.html), to keep track of all the things that need to be done
the next time `flush` is invoked. You usually do not directly interact with
a `UnitOfWork` but with the `EntityManager` instead.