Custom Collections
This feature was introduced in version 1.1 |
By default, Doctrine uses ArrayCollection
implementation of its Collection
interface to hold both embedded and referenced documents. That collection may then
be wrapped by a PersistentCollection
to allow for change tracking and other
persistence-related features.
1 <?php
use Doctrine\Common\Collections\ArrayCollection;
/** @Document */
class Application
{
// ...
/**
* @EmbedMany(targetDocument=Section::class)
*/
private $sections;
public function __construct()
{
$this->sections = new ArrayCollection();
}
// ...
}
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
For most cases this solution is sufficient but more sophisticated domains could use their own collections (e.g. a collection that ensures its contained objects are sorted) or to simply add common filtering methods that otherwise would otherwise be added to owning document's class.
Custom Collection Classes
You may want to check malarzm/collections
which provides alternative implementations of Doctrine's |
Using your own Collection
implementation is as simple as specifying the
collectionClass
parameter in the @EmbedMany
or @ReferenceMany
mapping
and ensuring that your custom class is initialized in the owning class' constructor:
1 <?php
use Doctrine\Common\Collections\ArrayCollection;
/** @Document */
class Application
{
// ...
/**
* @EmbedMany(
* collectionClass=SectionCollection::class
* targetDocument=Section::class
* )
*/
private $sections;
public function __construct()
{
$this->sections = new SectionCollection();
}
// ...
}
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
If you are satisfied with ArrayCollection
and only want
to sprinkle it with some filtering methods, you may just extend it:
Alternatively, you may want to implement the whole class from scratch:
Taking Control of the Collection's Constructor
By default, Doctrine assumes that it can instantiate your collections in same
manner as an ArrayCollection
(i.e. the only parameter is an optional PHP
array); however, you may want to inject additional dependencies into your
custom collection class(es). This will require you to create a
PersistentCollectionFactory implementation,
which Doctrine will then use to construct its persistent collections.
You may decide to implement this class from scratch or extend our
AbstractPersistentCollectionFactory
:
1 <?php
use Doctrine\ODM\MongoDB\PersistentCollection\AbstractPersistentCollectionFactory;
use Symfony\Component\EventDispatcher\EventDispatcherInterface;
final class YourPersistentCollectionFactory extends AbstractPersistentCollectionFactory
{
private $eventDispatcher;
public function __construct(EventDispatcherInterface $eventDispatcher)
{
$this->eventDispatcher = $eventDispatcher;
}
protected function createCollectionClass(string $collectionClass)
{
switch ($collectionClass) {
case SectionCollection::class:
return new $collectionClass([], $this->eventDispatcher);
default:
return new $collectionClass();
}
}
}
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
The factory class must then be registered in the Configuration
: