This article is the first chapter of the series on the implementation of PHP dependency injection container.
Today, let’s not talk about the container (container). First, we will use some specific examples to introduce the concept of dependency injection to prove what problems the dependency injection model can solve and what benefits it can bring to developers.
If you already know the concept of dependency injection, you can skip this article.
Dependency injection may be one of the simplest design patterns I know. In many cases, you may have used dependency injection unconsciously. But it's also one of the hardest to explain. I think part of the reason is that most of the examples that introduce dependency injection lack practical meaning and are difficult to understand. Because PHP is mainly used for web development, let's first look at a simple web development example.
HTTP itself is a stateless connection protocol. In order to support the application to store user information when the client initiates a WEB request, we need to use a technology to achieve storage state interaction. Of course the easiest way is to use cookies, and a better way is PHP's built-in Session mechanism.
$_SESSION['language'] = 'fr';
The above code stores the user language in the Session variable named language, so in subsequent requests by the user, the language can be obtained through the global array $_SESSION:
$user_language = $_SESSION['language'];
Dependencies Injection is mainly used for object-oriented development. Now let us assume that we have a SessionStorage class, which encapsulates the PHP Session mechanism:
class SessionStorage { function __construct($cookieName = 'PHP_SESS_ID') { session_name($cookieName); session_start(); } function set($key, $value) { $_SESSION[$key] = $value; } function get($key) { return $_SESSION[$key]; } // ... }
There is also a User class that provides a more advanced encapsulation:
class User { protected $storage; function __construct() { $this->storage = new SessionStorage(); } function setLanguage($language) { $this->storage->set('language', $language); } function getLanguage() { return $this->storage->get('language'); } // ... }
The code is very simple, and it is also very simple to use the User class:
$user = new User(); $user->setLanguage('fr'); $user_language = $user->getLanguage();
Everything is beautiful, unless your program needs better scalability. Suppose now you want to change the COOKIE key value that saves session_id. Here are some alternative methods:
When creating a SessionStorage instance in the User class, use the string 'SESSION_ID' hard-coded in the SessionStorage constructor:
class User { function __construct() { $this->storage = new SessionStorage('SESSION_ID'); } // ... }
Set a constant outside the User class (named STORAGE_SESSION_NAME)
class User { function __construct() { $this->storage = new SessionStorage(STORAGE_SESSION_NAME); } // ... } define('STORAGE_SESSION_NAME', 'SESSION_ID');
Pass the Session name through the parameters in the User class constructor
class User { function __construct($sessionName) { $this->storage = new SessionStorage($sessionName); } // ... } $user = new User('SESSION_ID');
Still pass the Session name through the parameters in the User class constructor, but this time the parameters are in the form of an array
class User { function __construct($storageOptions) { $this->storage = new SessionStorage($storageOptions['session_name']); } // ... } $user = new User(array('session_name' => 'SESSION_ID'));
The above methods are all bad.
Hardcoding the session name in the user class does not really solve the problem. If you need to change the COOKIE key value that saves the session_id in the future, you have to modify the user class again (the User class should not care about the COOKIE key value) .
The way to use constants is also bad, causing the User class to depend on a constant setting.
It is relatively better to pass the session name through the parameters or array of the User class constructor, but it is not perfect. This interferes with the parameters of the User class constructor, because how to store the Session is not what the User class needs to care about. The User class should not be associated with them.
In addition, there is another problem that is not easy to solve: how do we change the SessionStorage class. There are many application scenarios for this, for example, you want to use a Session simulation class for testing, or you want to store the Session in a database or memory. With the current implementation, it is difficult to do this without changing the User class.
Now, let’s use dependency injection. Recall that we created the SessionStorage object inside the User class before. Now we modify it and pass the SessionStorage object through the constructor of the User class.
class User { function __construct($storage) { $this->storage = $storage; } // ... }
This is the most classic case of dependency injection, no one. Now there are some small changes to using the User class. First you need to create the SessionStorage object.
$storage = new SessionStorage('SESSION_ID'); $user = new User($storage);
Now, it is very simple to configure the session storage object, and it is also very simple to change the session storage object. All of this does not need to update the User class, reducing the coupling between business classes.
Pico Container's website describes dependency injection like this:
Dependency injection is a method of passing dependent components to a class through the class's constructor, method, or direct writing.
So dependency injection is not limited to constructor injection. Let’s take a look at several injection methods:
Constructor injection
class User { function __construct($storage) { $this->storage = $storage; } // ... }
setter method injection
class User { function setSessionStorage($storage) { $this->storage = $storage; } // ... }
Direct attribute injection
class User { public $sessionStorage; } $user->sessionStorage = $storage;
現(xiàn)在,大多數(shù)流行的PHP框架都采用了依賴注入的模式實(shí)現(xiàn)業(yè)務(wù)組件間的高內(nèi)聚低耦合。
// symfony: 構(gòu)造函數(shù)注入的例子 $dispatcher = new sfEventDispatcher(); $storage = new sfMySQLSessionStorage(array('database' => 'session', 'db_table' => 'session')); $user = new sfUser($dispatcher, $storage, array('default_culture' => 'en')); // Zend Framework: setter方式注入的例子 $transport = new Zend_Mail_Transport_Smtp('smtp.gmail.com', array( 'auth' => 'login', 'username' => 'foo', 'password' => 'bar', 'ssl' => 'ssl', 'port' => 465, )); $mailer = new Zend_Mail(); $mailer->setDefaultTransport($transport);
?以上就是理解PHP依賴注入容器系列(一) 什么是的內(nèi)容,更多相關(guān)內(nèi)容請(qǐng)關(guān)注PHP中文網(wǎng)(www.miracleart.cn)!

Hot AI Tools

Undress AI Tool
Undress images for free

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

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

Clothoff.io
AI clothes remover

Video Face Swap
Swap faces in any video effortlessly with our completely free AI face swap tool!

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

To merge two PHP arrays and keep unique values, there are two main methods. 1. For index arrays or only deduplication, use array_merge and array_unique combinations: first merge array_merge($array1,$array2) and then use array_unique() to deduplicate them to finally get a new array containing all unique values; 2. For associative arrays and want to retain key-value pairs in the first array, use the operator: $result=$array1 $array2, which will ensure that the keys in the first array will not be overwritten by the second array. These two methods are applicable to different scenarios, depending on whether the key name is retained or only the focus is on

exit() is a function in PHP that is used to terminate script execution immediately. Common uses include: 1. Terminate the script in advance when an exception is detected, such as the file does not exist or verification fails; 2. Output intermediate results during debugging and stop execution; 3. Call exit() after redirecting in conjunction with header() to prevent subsequent code execution; In addition, exit() can accept string parameters as output content or integers as status code, and its alias is die().

The rational use of semantic tags in HTML can improve page structure clarity, accessibility and SEO effects. 1. Used for independent content blocks, such as blog posts or comments, it must be self-contained; 2. Used for classification related content, usually including titles, and is suitable for different modules of the page; 3. Used for auxiliary information related to the main content but not core, such as sidebar recommendations or author profiles. In actual development, labels should be combined and other, avoid excessive nesting, keep the structure simple, and verify the rationality of the structure through developer tools.

When you encounter the prompt "This operation requires escalation of permissions", it means that you need administrator permissions to continue. Solutions include: 1. Right-click the "Run as Administrator" program or set the shortcut to always run as an administrator; 2. Check whether the current account is an administrator account, if not, switch or request administrator assistance; 3. Use administrator permissions to open a command prompt or PowerShell to execute relevant commands; 4. Bypass the restrictions by obtaining file ownership or modifying the registry when necessary, but such operations need to be cautious and fully understand the risks. Confirm permission identity and try the above methods usually solve the problem.

There are two ways to create an array in PHP: use the array() function or use brackets []. 1. Using the array() function is a traditional way, with good compatibility. Define index arrays such as $fruits=array("apple","banana","orange"), and associative arrays such as $user=array("name"=>"John","age"=>25); 2. Using [] is a simpler way to support since PHP5.4, such as $color

The way to process raw POST data in PHP is to use $rawData=file_get_contents('php://input'), which is suitable for receiving JSON, XML, or other custom format data. 1.php://input is a read-only stream, which is only valid in POST requests; 2. Common problems include server configuration or middleware reading input streams, which makes it impossible to obtain data; 3. Application scenarios include receiving front-end fetch requests, third-party service callbacks, and building RESTfulAPIs; 4. The difference from $_POST is that $_POST automatically parses standard form data, while the original data is suitable for non-standard formats and allows manual parsing; 5. Ordinary HTM

To safely handle PHP file uploads, you need to verify the source and type, control the file name and path, set server restrictions, and process media files twice. 1. Verify the upload source to prevent CSRF through token and detect the real MIME type through finfo_file using whitelist control; 2. Rename the file to a random string and determine the extension to store it in a non-Web directory according to the detection type; 3. PHP configuration limits the upload size and temporary directory Nginx/Apache prohibits access to the upload directory; 4. The GD library resaves the pictures to clear potential malicious data.

InPHP,variablesarepassedbyvaluebydefault,meaningfunctionsorassignmentsreceiveacopyofthedata,whilepassingbyreferenceallowsmodificationstoaffecttheoriginalvariable.1.Whenpassingbyvalue,changestothecopydonotimpacttheoriginal,asshownwhenassigning$b=$aorp
