mirror of
https://github.com/janunger/rheinwerk-video-training.git
synced 2024-11-25 08:33:41 +01:00
253 lines
6.9 KiB
Markdown
253 lines
6.9 KiB
Markdown
|
# DeepCopy
|
||
|
|
||
|
DeepCopy helps you create deep copies (clones) of your objects. It is designed to handle cycles in the association graph.
|
||
|
|
||
|
[![Build Status](https://travis-ci.org/myclabs/DeepCopy.png?branch=master)](https://travis-ci.org/myclabs/DeepCopy) [![Coverage Status](https://coveralls.io/repos/myclabs/DeepCopy/badge.png?branch=master)](https://coveralls.io/r/myclabs/DeepCopy?branch=master) [![Scrutinizer Quality Score](https://scrutinizer-ci.com/g/myclabs/DeepCopy/badges/quality-score.png?s=2747100c19b275f93a777e3297c6c12d1b68b934)](https://scrutinizer-ci.com/g/myclabs/DeepCopy/)
|
||
|
[![Total Downloads](https://poser.pugx.org/myclabs/deep-copy/downloads.svg)](https://packagist.org/packages/myclabs/deep-copy)
|
||
|
|
||
|
|
||
|
## How?
|
||
|
|
||
|
Install with Composer:
|
||
|
|
||
|
```json
|
||
|
composer require myclabs/deep-copy
|
||
|
```
|
||
|
|
||
|
Use simply:
|
||
|
|
||
|
```php
|
||
|
use DeepCopy\DeepCopy;
|
||
|
|
||
|
$deepCopy = new DeepCopy();
|
||
|
$myCopy = $deepCopy->copy($myObject);
|
||
|
```
|
||
|
|
||
|
|
||
|
## Why?
|
||
|
|
||
|
- How do you create copies of your objects?
|
||
|
|
||
|
```php
|
||
|
$myCopy = clone $myObject;
|
||
|
```
|
||
|
|
||
|
- How do you create **deep** copies of your objects (i.e. copying also all the objects referenced in the properties)?
|
||
|
|
||
|
You use [`__clone()`](http://www.php.net/manual/en/language.oop5.cloning.php#object.clone) and implement the behavior yourself.
|
||
|
|
||
|
- But how do you handle **cycles** in the association graph?
|
||
|
|
||
|
Now you're in for a big mess :(
|
||
|
|
||
|
![association graph](doc/graph.png)
|
||
|
|
||
|
### Using simply `clone`
|
||
|
|
||
|
![Using clone](doc/clone.png)
|
||
|
|
||
|
### Overridding `__clone()`
|
||
|
|
||
|
![Overridding __clone](doc/deep-clone.png)
|
||
|
|
||
|
### With DeepCopy
|
||
|
|
||
|
![With DeepCopy](doc/deep-copy.png)
|
||
|
|
||
|
|
||
|
## How it works
|
||
|
|
||
|
DeepCopy traverses recursively all your object's properties and clones them.
|
||
|
|
||
|
To avoid cloning the same object twice (and thus, keep you object graph), it keeps a hash-map of all instances.
|
||
|
|
||
|
|
||
|
## Going further
|
||
|
|
||
|
You can add filters to customize the copy process.
|
||
|
|
||
|
The method to add a filter is `$deepCopy->addFilter($filter, $matcher)`,
|
||
|
with `$filter` implementing `DeepCopy\Filter\Filter`
|
||
|
and `$matcher` implementing `DeepCopy\Matcher\Matcher`.
|
||
|
|
||
|
We provide some generic filters and matchers.
|
||
|
|
||
|
### Matchers
|
||
|
|
||
|
- `DeepCopy\Matcher` applies on a object attribute.
|
||
|
- `DeepCopy\TypeMatcher` applies on any element found in graph, including array elements.
|
||
|
|
||
|
#### Property name
|
||
|
|
||
|
The `PropertyNameMatcher` will match a property by its name:
|
||
|
|
||
|
```php
|
||
|
use DeepCopy\Matcher\PropertyNameMatcher;
|
||
|
|
||
|
$matcher = new PropertyNameMatcher('id');
|
||
|
// will apply a filter to any property of any objects named "id"
|
||
|
```
|
||
|
|
||
|
#### Specific property
|
||
|
|
||
|
The `PropertyMatcher` will match a specific property of a specific class:
|
||
|
|
||
|
```php
|
||
|
use DeepCopy\Matcher\PropertyMatcher;
|
||
|
|
||
|
$matcher = new PropertyMatcher('MyClass', 'id');
|
||
|
// will apply a filter to the property "id" of any objects of the class "MyClass"
|
||
|
```
|
||
|
|
||
|
#### Type
|
||
|
|
||
|
The `TypeMatcher` will match any element by its type (instance of a class or any value that could be parameter of [gettype()](http://php.net/manual/en/function.gettype.php) function):
|
||
|
|
||
|
```php
|
||
|
use DeepCopy\TypeMatcher\TypeMatcher;
|
||
|
|
||
|
$matcher = new TypeMatcher('Doctrine\Common\Collections\Collection');
|
||
|
// will apply a filter to any object that is an instance of Doctrine\Common\Collections\Collection
|
||
|
```
|
||
|
|
||
|
### Filters
|
||
|
|
||
|
- `DeepCopy\Filter` applies a transformation to the object attribute matched by `DeepCopy\Matcher`.
|
||
|
- `DeepCopy\TypeFilter` applies a transformation to any element matched by `DeepCopy\TypeMatcher`.
|
||
|
|
||
|
#### `SetNullFilter`
|
||
|
|
||
|
Let's say for example that you are copying a database record (or a Doctrine entity), so you want the copy not to have any ID:
|
||
|
|
||
|
```php
|
||
|
use DeepCopy\DeepCopy;
|
||
|
use DeepCopy\Filter\SetNullFilter;
|
||
|
use DeepCopy\Matcher\PropertyNameMatcher;
|
||
|
|
||
|
$myObject = MyClass::load(123);
|
||
|
echo $myObject->id; // 123
|
||
|
|
||
|
$deepCopy = new DeepCopy();
|
||
|
$deepCopy->addFilter(new SetNullFilter(), new PropertyNameMatcher('id'));
|
||
|
$myCopy = $deepCopy->copy($myObject);
|
||
|
|
||
|
echo $myCopy->id; // null
|
||
|
```
|
||
|
|
||
|
#### `KeepFilter`
|
||
|
|
||
|
If you want a property to remain untouched (for example, an association to an object):
|
||
|
|
||
|
```php
|
||
|
use DeepCopy\DeepCopy;
|
||
|
use DeepCopy\Filter\KeepFilter;
|
||
|
use DeepCopy\Matcher\PropertyMatcher;
|
||
|
|
||
|
$deepCopy = new DeepCopy();
|
||
|
$deepCopy->addFilter(new KeepFilter(), new PropertyMatcher('MyClass', 'category'));
|
||
|
$myCopy = $deepCopy->copy($myObject);
|
||
|
|
||
|
// $myCopy->category has not been touched
|
||
|
```
|
||
|
|
||
|
#### `ReplaceFilter`
|
||
|
|
||
|
1. If you want to replace the value of a property:
|
||
|
|
||
|
```php
|
||
|
use DeepCopy\DeepCopy;
|
||
|
use DeepCopy\Filter\ReplaceFilter;
|
||
|
use DeepCopy\Matcher\PropertyMatcher;
|
||
|
|
||
|
$deepCopy = new DeepCopy();
|
||
|
$callback = function ($currentValue) {
|
||
|
return $currentValue . ' (copy)'
|
||
|
};
|
||
|
$deepCopy->addFilter(new ReplaceFilter($callback), new PropertyMatcher('MyClass', 'title'));
|
||
|
$myCopy = $deepCopy->copy($myObject);
|
||
|
|
||
|
// $myCopy->title will contain the data returned by the callback, e.g. 'The title (copy)'
|
||
|
```
|
||
|
|
||
|
2. If you want to replace whole element:
|
||
|
|
||
|
```php
|
||
|
use DeepCopy\DeepCopy;
|
||
|
use DeepCopy\TypeFilter\ReplaceFilter;
|
||
|
use DeepCopy\TypeMatcher\TypeMatcher;
|
||
|
|
||
|
$deepCopy = new DeepCopy();
|
||
|
$callback = function (MyClass $myClass) {
|
||
|
return get_class($myClass);
|
||
|
};
|
||
|
$deepCopy->addTypeFilter(new ReplaceFilter($callback), new TypeMatcher('MyClass'));
|
||
|
$myCopy = $deepCopy->copy(array(new MyClass, 'some string', new MyClass));
|
||
|
|
||
|
// $myCopy will contain ['MyClass', 'some stirng', 'MyClass']
|
||
|
```
|
||
|
|
||
|
|
||
|
The `$callback` parameter of the `ReplaceFilter` constructor accepts any PHP callable.
|
||
|
|
||
|
#### `ShallowCopyFilter`
|
||
|
|
||
|
Stop *DeepCopy* from recursively copying element, using standard `clone` instead:
|
||
|
|
||
|
```php
|
||
|
use DeepCopy\DeepCopy;
|
||
|
use DeepCopy\TypeFilter\ShallowCopyFilter;
|
||
|
use DeepCopy\TypeMatcher\TypeMatcher;
|
||
|
use Mockery as m;
|
||
|
|
||
|
$this->deepCopy = new DeepCopy();
|
||
|
$this->deepCopy->addTypeFilter(
|
||
|
new ShallowCopyFilter,
|
||
|
new TypeMatcher(m\MockInterface::class)
|
||
|
);
|
||
|
|
||
|
$myServiceWithMocks = new MyService(m::mock(MyDependency1::class), m::mock(MyDependency2::class));
|
||
|
// all mocks will be just cloned, not deep-copied
|
||
|
```
|
||
|
|
||
|
#### `DoctrineCollectionFilter`
|
||
|
|
||
|
If you use Doctrine and want to copy an entity, you will need to use the `DoctrineCollectionFilter`:
|
||
|
|
||
|
```php
|
||
|
use DeepCopy\DeepCopy;
|
||
|
use DeepCopy\Filter\Doctrine\DoctrineCollectionFilter;
|
||
|
use DeepCopy\Matcher\PropertyTypeMatcher;
|
||
|
|
||
|
$deepCopy = new DeepCopy();
|
||
|
$deepCopy->addFilter(new DoctrineCollectionFilter(), new PropertyTypeMatcher('Doctrine\Common\Collections\Collection'));
|
||
|
$myCopy = $deepCopy->copy($myObject);
|
||
|
```
|
||
|
|
||
|
#### `DoctrineEmptyCollectionFilter`
|
||
|
|
||
|
If you use Doctrine and want to copy an entity who contains a `Collection` that you want to be reset, you can use the `DoctrineEmptyCollectionFilter`
|
||
|
|
||
|
```php
|
||
|
use DeepCopy\DeepCopy;
|
||
|
use DeepCopy\Filter\Doctrine\DoctrineEmptyCollectionFilter;
|
||
|
use DeepCopy\Matcher\PropertyMatcher;
|
||
|
|
||
|
$deepCopy = new DeepCopy();
|
||
|
$deepCopy->addFilter(new DoctrineEmptyCollectionFilter(), new PropertyMatcher('MyClass', 'myProperty'));
|
||
|
$myCopy = $deepCopy->copy($myObject);
|
||
|
|
||
|
// $myCopy->myProperty will return an empty collection
|
||
|
```
|
||
|
|
||
|
## Contributing
|
||
|
|
||
|
DeepCopy is distributed under the MIT license.
|
||
|
|
||
|
### Tests
|
||
|
|
||
|
Running the tests is simple:
|
||
|
|
||
|
```php
|
||
|
phpunit
|
||
|
```
|