The class instantiator tries to create an object of the passed class recursively.
<?php
use HJerichen\ClassInstantiator\ClassInstantiator;
class ClassA {
}
class ClassB {
public function __construct(ClassA $instanceA) {}
}
$instantiator = new ClassInstantiator();
$object = $instantiator->instantiateClass(ClassB::class);
Some arguments for a constructor cannot be created automatically.
For example primitive types like integer or interfaces.
Or you want to inject a specific object, that should not be created automatically, like a PDO object.
For those scenarios, there are multiple solutions.
Pass predefined arguments:
<?php
use HJerichen\ClassInstantiator\ClassInstantiator;
class ClassA {
public function __construct(PDO $database, int $id) {}
}
class ClassB {
public function __construct(ClassA $instanceA) {}
}
$predefinedArguments = [
'database' => new PDO('dsn'),
'id' => 5
];
$instantiator = new ClassInstantiator();
$object = $instantiator->instantiateClass(ClassB::class, $predefinedArguments);
Extend the class instantiator:
<?php
use HJerichen\ClassInstantiator\ClassInstantiator;
class ClassA {
public function __construct(PDO $database, int $id) {}
}
class MyInstantiator extends ClassInstantiator {
# The class instantiator will match the property name to the constructor parameter name
private $id = 5;
# The class instantiator will match the return type to the constructor parameter type
public function getDatabase(): PDO
{
return new PDO('dsn');
}
}
$instantiator = new MyInstantiator();
$object = $instantiator->instantiateClass(ClassA::class);
Use an attribute to define a class instantiator extension:
<?php
use HJerichen\ClassInstantiator\Attribute\Instantiator;
use HJerichen\ClassInstantiator\ClassInstantiator;
#[Instantiator(class: MyInstantiator::class)]
class ClassA {
public function __construct(PDO $database, int $id) {}
}
class ClassB {
public function __construct(ClassA $a) {}
}
class MyInstantiator extends ClassInstantiator {
# The class instantiator will match the property name to the constructor parameter name
private $id = 5;
# The class instantiator will match the return type to the constructor parameter type
public function getDatabase(): PDO
{
return new PDO('dsn');
}
}
$instantiator = new ClassInstantiator();
$object = $instantiator->instantiateClass(ClassB::class);
Instantiate classes with an extension has a higher priority then using an attribute.
This allows overwriting the attribute and instantiate classes depending on your current needs.
You can also use the instantiator attribute to let the class build from a method in a specific instantiator:
<?php
use HJerichen\ClassInstantiator\Attribute\Instantiator;
use HJerichen\ClassInstantiator\ClassInstantiator;
#[Instantiator(class: MyInstantiator::class)]
class ClassA {
public function __construct(PDO $database, int $id) {}
}
class ClassB {
public function __construct(ClassA $a) {}
}
class MyInstantiator extends ClassInstantiator {
# The class instantiator will match the return type to the class type
public function buildClassA(): ClassA
{
$database = new PDO('dsn');
return new ClassA($database, 55);
}
}
$instantiator = new ClassInstantiator();
$object = $instantiator->instantiateClass(ClassB::class);
The attribute can also be used on a constructor parameter:
<?php
use HJerichen\ClassInstantiator\Attribute\Instantiator;
use HJerichen\ClassInstantiator\ClassInstantiator;
class ClassA {
public function __construct(PDO $database, int $id) {}
}
class ClassB {
public function __construct(
#[Instantiator(class: MyInstantiator::class)] ClassA $a
) {
}
}
class MyInstantiator extends ClassInstantiator {
# The class instantiator will match the return type to the class type
public function buildClassA(): ClassA
{
$database = new PDO('dsn');
return new ClassA($database, 55);
}
}
$instantiator = new ClassInstantiator();
$object = $instantiator->instantiateClass(ClassB::class);
The attribute on the parameter has a higher priority as the one on the class itself.
It is also possible to specify a method of the instantiator in the attribute.
<?php
use HJerichen\ClassInstantiator\Attribute\Instantiator;
use HJerichen\ClassInstantiator\ClassInstantiator;
class ClassA {
public function __construct(PDO $database, int $id) {}
}
class ClassB {
public function __construct(
#[Instantiator(class: MyInstantiator::class, method: 'someMethod')] ClassA $a
) {
}
}
class MyInstantiator extends ClassInstantiator {
# The class instantiator will not use this method to build a ClassA insance.
public function buildClassA(): ClassA
{
$database = new PDO('dsn');
return new ClassA($database, 55);
}
# Is uses this method, because is is specified in the attribute.
public function someMethod(): ClassA
{
$database = new PDO('dsn');
return new ClassA($database, 60);
}
}
$instantiator = new ClassInstantiator();
$object = $instantiator->instantiateClass(ClassB::class);
It is possible to sore objects permanently with an attribute.
Those objects will then be returned when a matching class is requested instead of creating a new one every time.
<?php
use HJerichen\ClassInstantiator\Attribute\Instantiator;
#[Instantiator(class: MyInstantiator::class, store: true)]
class ClassA {
public function __construct(PDO $database, int $id) {}
}
Storing objects with attribute does not work when the instantiation is done in the extension.
This is because of the higher priority for extensions.
But you can store objects as well when using the method "injectObject" inside the creation method of the extension.
It is possible to inject objects into the class instantiator.
Those objects will be returned when a matching class is requested.
<?php
use HJerichen\ClassInstantiator\ClassInstantiator;
interface InterfaceA {}
class ClassA implements InterfaceA {}
$instantiator = new ClassInstantiator();
# only using the object resolves to this object when exactly this class is requested
$instantiator->injectObject(new ClassA());
$object = $instantiator->instantiateClass(ClassA::class);
# using the object and an interface as second parameter
# resolves to this object when exactly this interface is requested
$instantiator->injectObject(new ClassA(), InterfaceA::class);
$object = $instantiator->instantiateClass(InterfaceA::class);
There is also a PSR-11 Container Implementation.
You can use this container in frameworks and libraries who support those.
Example for GraphQLite:
<?php
use HJerichen\ClassInstantiator\ClassInstantiator;
use HJerichen\ClassInstantiator\ClassInstantiatorContainer;
use TheCodingMachine\GraphQLite\SchemaFactory;
# you can also inject an extension.
$container = new ClassInstantiatorContainer(new ClassInstantiator());
$factory = new SchemaFactory($cache, $container);
This project is free and under the MIT Licence. Responsible for this project is Heiko Jerichen ([email protected]).