2010-08-01 18:37:53 +04:00
|
|
|
A partial object is an object whose state is not fully initialized after being
|
|
|
|
reconstituted from the database and that is disconnected from the rest of its data.
|
|
|
|
The following section will describe why partial objects are problematic and what the approach of Doctrine2 to this problem is.
|
|
|
|
|
|
|
|
> **NOTE**
|
|
|
|
> The partial object problem in general does not apply to methods or
|
|
|
|
> queries where you do not retrieve the query result as objects. Examples are:
|
|
|
|
> `Query#getArrayResult()`, `Query#getScalarResult()`, `Query#getSingleScalarResult()`,
|
|
|
|
> etc.
|
|
|
|
|
2010-11-01 23:16:12 +03:00
|
|
|
## What is the problem?
|
2010-08-01 18:37:53 +04:00
|
|
|
|
|
|
|
In short, partial objects are problematic because they are usually objects with
|
|
|
|
broken invariants. As such, code that uses these partial objects tends to be
|
|
|
|
very fragile and either needs to "know" which fields or methods can be safely
|
|
|
|
accessed or add checks around every field access or method invocation. The same
|
|
|
|
holds true for the internals, i.e. the method implementations, of such objects.
|
|
|
|
You usually simply assume the state you need in the method is available, after
|
|
|
|
all you properly constructed this object before you pushed it into the database,
|
|
|
|
right? These blind assumptions can quickly lead to null reference errors when
|
|
|
|
working with such partial objects.
|
|
|
|
|
|
|
|
It gets worse with the scenario of an optional association (0..1 to 1). When
|
2010-10-08 01:39:52 +04:00
|
|
|
the associated field is NULL, you don't know whether this object does not have
|
2010-08-01 18:37:53 +04:00
|
|
|
an associated object or whether it was simply not loaded when the owning object
|
|
|
|
was loaded from the database.
|
|
|
|
|
|
|
|
These are reasons why many ORMs do not allow partial objects at all and instead
|
|
|
|
you always have to load an object with all its fields (associations being proxied).
|
|
|
|
One secure way to allow partial objects is if the programming language/platform
|
|
|
|
allows the ORM tool to hook deeply into the object and instrument it in such a
|
|
|
|
way that individual fields (not only associations) can be loaded lazily on first
|
|
|
|
access. This is possible in Java, for example, through bytecode instrumentation.
|
|
|
|
In PHP though this is not possible, so there is no way to have "secure" partial
|
|
|
|
objects in an ORM with transparent persistence.
|
|
|
|
|
|
|
|
Doctrine, by default, does not allow partial objects. That means, any query that
|
|
|
|
only selects partial object data and wants to retrieve the result as objects
|
|
|
|
(i.e. `Query#getResult()`) will raise an exception telling you that
|
|
|
|
partial objects are dangerous. If you want to force a query to return you partial
|
|
|
|
objects, possibly as a performance tweak, you can use the `partial` keyword as follows:
|
|
|
|
|
2010-11-01 23:16:12 +03:00
|
|
|
<?php
|
2010-08-01 18:37:53 +04:00
|
|
|
$q = $em->createQuery("select partial u.{id,name} from MyApp\Domain\User u");
|
|
|
|
|
2010-11-01 23:16:12 +03:00
|
|
|
## When should I force partial objects?
|
2010-08-01 18:37:53 +04:00
|
|
|
|
|
|
|
Mainly for optimization purposes, but be careful of premature optimization as partial objects
|
|
|
|
lead to potentially more fragile code.
|