Analysis of the running process of PHP container Pimple
This article mainly introduces the analysis of the running process of PHP container Pimple, which has certain reference value. Now I share it with everyone. Friends in need can refer to it
Required knowledge points
Closure
Closures and anonymous functions were introduced in PHP5.3.0.
A closure refers to a function that encapsulates the surrounding state when created. Even if the environment in which the closure is located no longer exists, the state encapsulated in the closure still exists.
Theoretically, closures and anonymous functions are different concepts. But PHP treats it as the same concept.
Actually, closures and anonymous functions are objects disguised as functions. They are instances of the Closure class.
Closures, like strings and integers, are first-class value types.
Create closure:
<?php $closure = function ($name) { return 'Hello ' . $name; }; echo $closure('nesfo');//Hello nesfo var_dump(method_exists($closure, '__invoke'));//true
The reason why we can call the$closure
variable is because the value of this variable is a closure, and the closure object implements__invoke()
Magic method. As long as there is()
after the variable name, PHP will find and call the__invoke()
method.
Usually PHP closures are used as callbacks of functions.
array_map()
, preg_replace_callback()
methods all use callback functions. This is the best time to use closures!
For example:
<?php $numbersPlusOne = array_map(function ($number) { return $number + 1; }, [1, 2, 3]); print_r($numbersPlusOne);
Get the result:
[2, 3, 4]
Before closures appeared, you could only create named functions individually and then reference that function by name. By doing this, the code execution will be slightly slower, and the implementation of the callback will be isolated from the usage scenario.
<?php function incrementNum ($number) { return $number + 1; } $numbersPlusOne = array_map('incrementNum', [1, 2, 3]); print_r($numbersPlusOne);
SPL
ArrayAccess
Implementing the ArrayAccess interface allows objects to operate like arrays. The ArrayAccess interface contains four methods that must be implemented:
interface ArrayAccess { //检查一个偏移位置是否存在 public mixed offsetExists ( mixed $offset ); //获取一个偏移位置的值 public mixed offsetGet( mixed $offset ); //设置一个偏移位置的值 public mixed offsetSet ( mixed $offset ); //复位一个偏移位置的值 public mixed offsetUnset ( mixed $offset ); }
SplObjectStorage
The SplObjectStorage class implements a map with objects as keys or a collection of objects (if the object corresponding to the key is ignored data) this data structure. An instance of this class is much like an array, but the objects it stores are all unique. Another feature of this class is that you can directly delete the specified object from it without traversing or searching the entire collection.
::class
Syntax
Because ::class
represents a string. The advantage of using ::class
is that you can directly rename a class in the IDE, and then the IDE will automatically handle the related references.
At the same time, when PHP executes the relevant code, it will not load the relevant class first.
Similarly, automated code inspection inspect can also correctly identify classes.
A brief analysis of Pimple container process
Pimpl is a popular container in the PHP community. There is not a lot of code, see https://github.com/silexphp/P... for details.
Our application can be developed based on Pimple:
namespace EasyWeChat\Foundation; use Pimple\Container; class Application extends Container { /** * Service Providers. * * @var array */ protected $providers = [ ServiceProviders\ServerServiceProvider::class, ServiceProviders\UserServiceProvider::class ]; /** * Application constructor. * * @param array $config */ public function __construct($config) { parent::__construct(); $this['config'] = function () use ($config) { return new Config($config); }; if ($this['config']['debug']) { error_reporting(E_ALL); } $this->registerProviders(); } /** * Add a provider. * * @param string $provider * * @return Application */ public function addProvider($provider) { array_push($this->providers, $provider); return $this; } /** * Set providers. * * @param array $providers */ public function setProviders(array $providers) { $this->providers = []; foreach ($providers as $provider) { $this->addProvider($provider); } } /** * Return all providers. * * @return array */ public function getProviders() { return $this->providers; } /** * Magic get access. * * @param string $id * * @return mixed */ public function __get($id) { return $this->offsetGet($id); } /** * Magic set access. * * @param string $id * @param mixed $value */ public function __set($id, $value) { $this->offsetSet($id, $value); } }
How to use our application:
$app = new Application([]); $user = $app->user;
After that we can use the method of the $user
object . We found that there is no $this->user
attribute, but it can be used directly. Mainly the role of these two methods:
public function offsetSet($id, $value){} public function offsetGet($id){}
Below we will explain what Pimple does when executing these two lines of code. But before explaining this, let’s look at some core concepts of containers.
Service provider
The service provider is the bridge connecting the container and the specific function implementation class. Service providers need to implement the interface ServiceProviderInterface
:
namespace Pimple; /** * Pimple service provider interface. * * @author Fabien Potencier * @author Dominik Zogg */ interface ServiceProviderInterface { /** * Registers services on the given container. * * This method should only be used to configure services and parameters. * It should not get services. * * @param Container $pimple A container instance */ public function register(Container $pimple); }
All service providers must implement the interface register
method.
There are 2 service providers by default in our application:
protected $providers = [ ServiceProviders\ServerServiceProvider::class, ServiceProviders\UserServiceProvider::class ];
Taking UserServiceProvider as an example, we look at its code implementation:
namespace EasyWeChat\Foundation\ServiceProviders; use EasyWeChat\User\User; use Pimple\Container; use Pimple\ServiceProviderInterface; /** * Class UserServiceProvider. */ class UserServiceProvider implements ServiceProviderInterface { /** * Registers services on the given container. * * This method should only be used to configure services and parameters. * It should not get services. * * @param Container $pimple A container instance */ public function register(Container $pimple) { $pimple['user'] = function ($pimple) { return new User($pimple['access_token']); }; } }
We see that this service provides The user's registration method will add the attribute user
to the container, but what is returned is not an object, but a closure. I will explain this later.
Service registration
We use $this->registerProviders();
in the constructor of Application to register all service providers:
private function registerProviders() { foreach ($this->providers as $provider) { $this->register(new $provider()); } }
Looking carefully, we find that the service provider is instantiated here and the register
method of the container Pimple is called:
public function register(ServiceProviderInterface $provider, array $values = array()) { $provider->register($this); foreach ($values as $key => $value) { $this[$key] = $value; } return $this; }
And here the register of the service provider is called
method, which is what we mentioned in the previous section: the registration method adds attributes user
to the container, but it returns not an object, but a closure.
When we add attributes user
to the container Pimple, the offsetSet($id, $value)
method will be called: values to the attributes of the container Pimple
, keys
are assigned respectively:
$this->values[$id] = $value; $this->keys[$id] = true;
Up to this point, we have not instantiated the class EasyWeChat\User\Usr
that actually provides actual functions. However, the service provider registration has been completed.
When we run here:
$user = $app->user;
will call offsetGet($id)
and instantiate the real class:
$raw = $this->values[$id]; $val = $this->values[$id] = $raw($this); $this->raw[$id] = $raw; $this->frozen[$id] = true; return $val;
$raw
obtains the closure:
$pimple['user'] = function ($pimple) { return new User($pimple['access_token']); };
$raw($this)
returns the instantiated object User. That is to say, only the actual call will instantiate the specific class. Later we can call methods in the User class through $this['user']
or $this->user
.
Of course, there are many features in Pimple that are worthy of our in-depth study, so we won’t explain them too much here.
The above is the entire content of this article. I hope it will be helpful to everyone's study. For more related content, please pay attention to the PHP Chinese website!
Related recommendations:
How to use wp_head() function in wordpress
Scope, global, static of PHP variables Wait for keywords
The above is the detailed content of Analysis of the running process of PHP container Pimple. For more information, please follow other related articles on the PHP Chinese website!

Hot AI Tools

Undresser.AI Undress
AI-powered app for creating realistic nude photos

AI Clothes Remover
Online AI tool for removing clothes from photos.

Undress AI Tool
Undress images for free

Clothoff.io
AI clothes remover

AI Hentai Generator
Generate AI Hentai for free.

Hot Article

Hot Tools

Notepad++7.3.1
Easy-to-use and free code editor

SublimeText3 Chinese version
Chinese version, very easy to use

Zend Studio 13.0.1
Powerful PHP integrated development environment

Dreamweaver CS6
Visual web development tools

SublimeText3 Mac version
God-level code editing software (SublimeText3)

Hot Topics



PHP 8.4 brings several new features, security improvements, and performance improvements with healthy amounts of feature deprecations and removals. This guide explains how to install PHP 8.4 or upgrade to PHP 8.4 on Ubuntu, Debian, or their derivati

Visual Studio Code, also known as VS Code, is a free source code editor — or integrated development environment (IDE) — available for all major operating systems. With a large collection of extensions for many programming languages, VS Code can be c

If you are an experienced PHP developer, you might have the feeling that you’ve been there and done that already.You have developed a significant number of applications, debugged millions of lines of code, and tweaked a bunch of scripts to achieve op

This tutorial demonstrates how to efficiently process XML documents using PHP. XML (eXtensible Markup Language) is a versatile text-based markup language designed for both human readability and machine parsing. It's commonly used for data storage an

JWT is an open standard based on JSON, used to securely transmit information between parties, mainly for identity authentication and information exchange. 1. JWT consists of three parts: Header, Payload and Signature. 2. The working principle of JWT includes three steps: generating JWT, verifying JWT and parsing Payload. 3. When using JWT for authentication in PHP, JWT can be generated and verified, and user role and permission information can be included in advanced usage. 4. Common errors include signature verification failure, token expiration, and payload oversized. Debugging skills include using debugging tools and logging. 5. Performance optimization and best practices include using appropriate signature algorithms, setting validity periods reasonably,

A string is a sequence of characters, including letters, numbers, and symbols. This tutorial will learn how to calculate the number of vowels in a given string in PHP using different methods. The vowels in English are a, e, i, o, u, and they can be uppercase or lowercase. What is a vowel? Vowels are alphabetic characters that represent a specific pronunciation. There are five vowels in English, including uppercase and lowercase: a, e, i, o, u Example 1 Input: String = "Tutorialspoint" Output: 6 explain The vowels in the string "Tutorialspoint" are u, o, i, a, o, i. There are 6 yuan in total

Static binding (static::) implements late static binding (LSB) in PHP, allowing calling classes to be referenced in static contexts rather than defining classes. 1) The parsing process is performed at runtime, 2) Look up the call class in the inheritance relationship, 3) It may bring performance overhead.

What are the magic methods of PHP? PHP's magic methods include: 1.\_\_construct, used to initialize objects; 2.\_\_destruct, used to clean up resources; 3.\_\_call, handle non-existent method calls; 4.\_\_get, implement dynamic attribute access; 5.\_\_set, implement dynamic attribute settings. These methods are automatically called in certain situations, improving code flexibility and efficiency.
