Download the PHP package rammewerk/container without Composer
On this page you can find all versions of the php package rammewerk/container. It is possible to download/install these versions without Composer. Possible dependencies are resolved automatically.
Download rammewerk/container
More information about rammewerk/container
Files in rammewerk/container
Package container
Short Description Simple and performant dependency injection container for PHP
License MIT
Homepage https://rammewerk.com
Informations about the package container
Rammewerk Container
Rammewerk Container is an elegant, high-performance dependency injection container for PHP, designed to simplify development with minimal to no configuration. Built exclusively for PHP 8.4+, it embraces modern coding standards and avoids legacy baggage.
With its fully-cached Reflection mechanism and native lazy objects, it resolves complex dependencies efficiently while deferring initialization to boost performance. Proven to be one of the fastest DI containers in benchmarks, Rammewerk Container offers a smart, streamlined solution for modern PHP projects.
Key features include:
- Easy to Use: Zero-config for basic functions.
- Lightweight: A single PHP file with less than 300 lines of code, no dependencies.
- Immutable Config: Fluent and predictable setups.
- Autowire Dependencies: Less boilerplate, clearer code.
- Highly Performant: Caches reflection data for speed, proven well in benchmarks
- Built-In Lazy Loading Objects only initialize on demand.
- IDE & Tools Friendly: Thorough docblocks and strict PHPStan checks let IDEs (e.g. PhpStorm) accurately autocomplete and hint returned classes.
- PSR-11 Support: PSR-11 ContainerInterface support through an extended adapter.
Installation
Install this package using Composer:
Requires PHP 8.4 or above.
Container API
Rammewerk Container provides three essential methods for managing dependencies:
Create
Instantiates and auto-wires a fully resolved class. Can also create instances with arguments.
Share (singleton)
Registers shared (singleton) classes, ensuring the same instance is returned on every request. By default, instances are not shared unless explicitly defined.
Bind
bind
/ bindings
: Maps interfaces or abstract types to concrete implementations, offering fine-grained control over
dependency resolution.
To use PSR-11 container interface, see details at the bottom of this README
Basic Usage
Rammewerk Container automatically resolves and instantiates class dependencies. If PaymentGateway needs PaymentProcessor, and PaymentProcessor needs Logger, the container figures it all out with no extra setup.
Consider these 3 classes:
Without a DI container:
With Rammewerk Container:
All dependencies are automatically resolved — no extra wiring required.
Non-Class Parameters
Not all dependencies are classes. Sometimes you need to pass other values (like strings, numbers or arrays). Here’s how:
The DI container will require a value for $processName. Provide it during creation:
create()
accepts two arguments:
- The class name (string).
- An optional array of constructor parameters.
Here, "PayPal" automatically satisfies the string parameter. You don’t need to manually specify PaymentProcessor — the container still resolves that class for you.
Understanding the Lazy Object Feature
By default, Rammewerk Container creates lazy objects, meaning classes aren’t initialized until you actually use them.
Let's look at an example:
Without Lazy Loading
Output:
With Lazy Loading
Output:
This example shows that ClassC doesn’t initialize until you actually use it. Likewise, ClassB remains uninitialized unless it’s needed—even though it’s declared in ClassC. This boosts efficiency by loading only what’s truly required.
Shared Instances
By default, each call to create() returns a new instance:
To make instances shared (singletons), class name must be defined as shared:
share() takes an array of class names and ensures you get the same instance each time. Because the container is immutable, calling share() returns a new container, preventing unwanted side effects.
Bindings / Implementations
Not all classes have concrete type-hints. Some might depend on interfaces or abstract classes, while others accept scalars or arrays. In these cases, you can guide the container by binding specific implementations or values to those parameters.
For example, suppose you have a NewsMailer interface and a MailChimp class that implements it:
To tell Rammewerk Container which concrete class to use for NewsMailer, do:
Now, whenever the container encounters NewsMailer, it will use MailChimp.
Multiple Bindings at Once
You can also define multiple bindings together:
This is more performant and keeps your setup concise.
Closure for More Custom Setup
Sometimes, you need extra configuration before returning a class. In these cases, Rammewerk Container lets you pass a \Closure as the implementation:
Here, FilesystemLoader needs a template directory (TEMPLATE_DIR), so we define a closure. The container is passed in automatically, letting you create or retrieve other dependencies as needed. This way, whenever the container encounters Twig\Loader\LoaderInterface, it returns a fully configured FilesystemLoader.
How the Container Resolves Dependencies
The container resolves dependencies by processing each constructor parameter in order. It tries to automatically resolve parameters when possible, using class types, built-in types, or default values. If needed, you can pass an array of arguments to the create() method, and the container will intelligently match each argument to the first compatible parameter.
For example, given a class with dependencies:
The container will:
- Resolve ClassA automatically.
- Assign null to the nullable int.
- Use 'string-value' for the string parameter.
This is especially useful with bindings defined as closures. Since the closure receives the container, it can call create() with custom arguments:
This allows custom values to be injected while relying on the container for automatic resolution of other dependencies, significantly reducing the amount of logic needed to set up the system. Unlike many other DI containers, which often require multiple bindings, factories, or definitions for similar functionality, this approach simplifies the process by minimizing the need for extensive manual setup.
This is the processing order that the container uses to resolve dependencies:
- Class types (ClassA, ClassB, etc.)
- Built-in types (int, string, array, etc.)
- Union types (ClassA|ClassB, ClassC|ClassD, etc.)
- Intersection types (ClassA&ClassB, ClassC&ClassD, etc.)
- Any leftover arguments are passed as-is, as it may be untyped or not resolved by the container.
- Default values (if available and not anymore arguments)
Class dependencies
If the constructor parameter is a class type and an argument matching the type is provided, the container will use that argument. If no such argument is provided:
- If a closure binding is defined for the class, the closure will be used to resolve the dependency.
- If the class type is the container itself, the container instance is returned.
- Otherwise, an instance of the class will be created unless the parameter allows null, in which case null is returned.
Built-in types
If the parameter is a built-in type (int, string, array etc.), the container will search for the first argument that matches the built-in type and use it. If no matching argument is found, it continues to the next resolution strategy.
Union types
The container handles union types (ClassA|string, ClassB|ClassC, etc.) by searching through the provided arguments and returning the first argument that matches any type in the union. This includes both class types and built-in types.
- If the argument matches a class type in the union (e.g., ClassA), it will return that class instance.
- If the argument matches a built-in type in the union (e.g., string, int), it will return the argument as-is.
- If no matching argument is found, and the parameter allows null, it will return null.
Intersection types
The container will search for the first argument that implements all the required classes or interfaces in the intersection type. If no such argument is found, it continues to the next resolution strategy. Intersection types are not autowired; use the bind() method to handle them properly.
Unresolved arguments
If the parameter has no type hint, the container will use the next available argument as-is.
Default values:
If no argument is provided and none of the above strategies resolve the parameter, the container will return the default value of the parameter if available. If no default value is defined, null will be returned. This may provoke an error if the parameter is required and no argument is provided and isn't nullable. THe container will not be able to resolve the dependency.
This approach ensures that the container can handle a wide range of parameter types, including built-in types, union types, and intersection types, while offering flexibility for custom bindings through closures.
Exception Handling
The Rammewerk Component Container library uses Rammewerk\Component\Container\Error\ContainerException
for exceptions
thrown during the execution. The exceptions provide information about issues such as failing to reflect a class or
instantiate an interface.
A Note on Container Caching
Rammewerk Container uses Reflection the first time it encounters a class to map its dependencies. While reflection has a cost, it’s still very efficient — and the container caches those results. Next time you request the same class (directly or indirectly), Rammewerk container reuses the cached data. This also extends to lazy objects, making it one of the fastest DI containers available.
For benchmark results see: https://github.com/rammewerk/php-di-container-benchmarks
PSR-11 Support
The container supports PSR-11: ContainerInterface through an extended implementation called PsrContainer
. Since PSR-11
only defines get()
and has()
methods and does not dictate how dependencies should be resolved, we chose not to make
it
the default implementation. Including PSR-11 as the default could lead to confusion about how to properly use the
create()
method, which offers greater flexibility in resolving dependencies by allowing additional arguments to be
passed during instantiation (read more on this below).
If you prefer to use the PSR-11 interface, you can do so by using PsrContainer
, which provides standard has()
and
get()
methods. However, note that this approach requires more explicit setup using the bind()
method to define how
dependencies should be resolved.
Note that the has()
method will most likely always return true
, since the container doesn't require you to define
a factory/binding for every class. I would recommend you to view the source code of PsrContainer
to see how it works.
Both PsrContainer
and the base Container
are fully extendable, allowing you to implement your own strategies for
setting up and managing dependencies as needed.
Contribution
If you have any issues or would like to contribute to the development of this library, feel free to open an issue or pull request.
License
The Rammewerk Container is open-sourced software licensed under the MIT license.