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.

FAQ

After the download, you have to make one include require_once('vendor/autoload.php');. After that you have to import the classes with use statements.

Example:
If you use only one package a project is not needed. But if you use more then one package, without a project it is not possible to import the classes with use statements.

In general, it is recommended to use always a project to download your libraries. In an application normally there is more than one library needed.
Some PHP packages are not free to download and because of that hosted in private repositories. In this case some credentials are needed to access such packages. Please use the auth.json textarea to insert credentials, if a package is coming from a private repository. You can look here for more information.

  • Some hosting areas are not accessible by a terminal or SSH. Then it is not possible to use Composer.
  • To use Composer is sometimes complicated. Especially for beginners.
  • Composer needs much resources. Sometimes they are not available on a simple webspace.
  • If you are using private repositories you don't need to share your credentials. You can set up everything on our site and then you provide a simple download link to your team member.
  • Simplify your Composer build process. Use our own command line tool to download the vendor folder as binary. This makes your build process faster and you don't need to expose your credentials for private repositories.
Please rate this library. Is it a good library?

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:

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:

  1. The class name (string).
  2. 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:

  1. Resolve ClassA automatically.
  2. Assign null to the nullable int.
  3. 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:

  1. Class types (ClassA, ClassB, etc.)
  2. Built-in types (int, string, array, etc.)
  3. Union types (ClassA|ClassB, ClassC|ClassD, etc.)
  4. Intersection types (ClassA&ClassB, ClassC&ClassD, etc.)
  5. Any leftover arguments are passed as-is, as it may be untyped or not resolved by the container.
  6. 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:

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.

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.


All versions of container with dependencies

PHP Build Version
Package Version
Requires php Version >=8.4
psr/container Version ^2.0.2
Composer command for our command line client (download client) This client runs in each environment. You don't need a specific PHP version etc. The first 20 API calls are free. Standard composer command

The package rammewerk/container contains the following files

Loading the files please wait ....