Let’s look at an example first:
<?php class A { public $b; public $c; public function A() { //TODO } public function Method() { $this->b=new B(); $this->c=new C(); $this->b->Method(); $this->c->Method(); //TODO } } class B { public function B() { //TODO } public function Method() { //TODO echo 'b'; } } class C { public function C() { //TODO } public function Method() { //TODO echo 'c'; } } $a=new A(); $a->Method(); ?>
The above code, we can easily understand one sentence:
A classDepend on Class B and Class C
In other words, if in the future development process, Class B or Class C needs to be modified, once thefunction is renamed, the function If the number of parameters changes, or even if the entire class structure is adjusted, we must also make corresponding adjustments to Class A. The independence of Class A is lost, which is very inconvenient during the development process, which is what we call "One thing affects the whole body." If the two categories are written by two people separately, conflicts often arise at this time. . .
If we really need to change categories B and C, is there any way to not change the code of category A or to change it as little as possible? Inversion of control is used here.High-level modules should not depend on low-level modules, both should rely on abstractions.Inversion of Control (IOC) is an idea,
Dependency Injection (DI) is a method to implement this idea.
The first method is called: constructor injection (this method is not recommended, but it is better than not using it)class A { public $b; public $c; public function A($b,$c) { $this->b=$b; $this->c=$c; } public function Method() { $this->b->Method(); $this->c->Method(); } }The client class is written like this :
$a=new A(new B(),new C()); $a->Method();The constructor of class A depends on class B and class C. It is passed in through the parameters of the constructor. At least one thing is achieved, which is class BIf One day, we need to expand Class B and make two subclasses of Class B
class B { public function B() { //TODO } public function Method() { //TODO echo 'b'; } } class B1 extends B { public function B1() { //TODO } public function Method() { echo 'b1'; } } class B2 extends B { public function B2() { //TODO } public function Method() { echo 'b2'; } }is also very simple. The client class is written like this:
$a=new A(new B2(),new C()); $a->Method();So class A does not need to care about which subclasses class B has. It only needs to be concerned about in the client class. The second method is called:
Factory modeInjection (recommended)
class Factory { public function Factory() { //TODO } public function create($s) { switch($s) { case 'B': { return new B(); break; } case 'C': { return new C(); break; } default: { return null; break; } } } }Our Class A code is changed to:
class A { public $b; public $c; public function A() { //TODO } public function Method() { $f=new Factory(); $this->b=$f->create('B'); $this->c=$f->create('C'); $this->b->Method(); $this->c->Method(); //TODO } }In fact, a small part has been decoupled. At least if the
constructor of classes B and C changes, such as modifying function parameters, etc., we Just change the Factory class.
Abstraction should not depend on details, details should depend on abstraction.Abstract the methods in classes B and C and make an
interface IMethod { public function Method(); }In this way, A The $b
variables and $c variables in the class are no longer a specific variable, but an abstract class type variable. Until the moment of running, their identity is unknown. How is the Method method implemented?
class B implements IMethod { public function B() { //TODO } public function Method() { //TODO echo 'b'; } } class C implements IMethod { public function C() { //TODO } public function Method() { //TODO echo 'c'; } }
Summary Several points:
1. We move the creation of class B objects and class C objects in class A Out of Class A2. Originally Class A relied on Class B and Class C, but now A depends on Factory, and Factory depends on B and C.The above is the detailed content of PHP Inversion of Control and Dependency Injection. For more information, please follow other related articles on the PHP Chinese website!

本篇文章带大家了解一下依赖注入,介绍一下依赖注入解决的问题和它原生的写法是什么,并聊聊Angular的依赖注入框架,希望对大家有所帮助!

在Phalcon框架中使用依赖注入(DependencyInjection)的方法引言:在现代的软件开发中,依赖注入(DependencyInjection)是一种常见的设计模式,旨在提高代码的可维护性和可测试性。而Phalcon框架作为一个快速、低耗的PHP框架,也支持使用依赖注入来管理和组织应用程序的依赖关系。本文将向您介绍如何在Phalcon框架中

在Go中,依赖注入(DI)模式通过函数参数传递实现,类型包括值传递和指针传递。在DI模式中,依赖项通常以指针传递,以提高解耦性、减少锁争用和支持可测试性。通过使用指针,函数与具体实现解耦,因为它只依赖于接口类型。指针传递还可以减少传递大对象的开销,从而减少锁争用。此外,DI模式可以轻松地为使用DI模式的函数编写单元测试,因为可以轻松地模拟依赖项。

针对使用JUnit测试依赖注入,摘要如下:使用模拟对象创建依赖项:@Mock注解可创建依赖项的模拟对象。设置测试数据:@Before方法在每个测试方法前运行,用于设置测试数据。配置模拟行为:Mockito.when()方法配置模拟对象的预期行为。验证结果:assertEquals()断言检查实际结果与预期值是否匹配。实际应用:可使用依赖注入框架(如SpringFramework)注入依赖项,通过JUnit单元测试验证注入的正确性和代码的正常运行。

控制反转(IoC)是软件设计模式,将对象依赖关系分离为硬编码耦合。在Go中,可以通过接口和依赖注入(DI)实现IoC:接口:定义方法集,遵循该接口的类型必须实现这些方法。依赖注入:外部配置或代码生成设置对象依赖关系。技巧包括:构造函数注入:在构造函数中指定依赖关系。字段注入:使用反射或代码生成向字段注入依赖关系。接口注入:将接口类型作为参数传递给函数或方法。

答案:PHP中的依赖注入和服务容器有助于灵活地管理依赖项,提高代码可测试性。依赖注入:通过容器传递依赖项,避免在函数内直接创建,提升灵活性。服务容器:存储依赖项实例,方便在程序中访问,进一步增强松散耦合。实战案例:示例应用程序演示依赖注入和服务容器的实际应用,将依赖项注入到控制器,体现松散耦合优势。

在Golang单元测试中使用依赖注入(DI)可以隔离要测试的代码,简化测试设置和维护。流行的DI库包括wire和go-inject,它们可以生成依赖项桩或模拟,供测试使用。DI测试的步骤包括设置依赖项、设置测试用例和断言结果。使用DI测试HTTP请求处理函数的示例表明,它可以轻松隔离和测试代码,无需实际依赖项或通信。

答案:在Go语言中,依赖注入可以通过接口和结构体实现。定义一个描述依赖项行为的接口。创建一个实现该接口的结构体。在函数中通过接口作为参数注入依赖项。允许在测试或不同场景中轻松替换依赖项。


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

SublimeText3 English version
Recommended: Win version, supports code prompts!

SAP NetWeaver Server Adapter for Eclipse
Integrate Eclipse with SAP NetWeaver application server.

WebStorm Mac version
Useful JavaScript development tools

SublimeText3 Linux new version
SublimeText3 Linux latest version

MinGW - Minimalist GNU for Windows
This project is in the process of being migrated to osdn.net/projects/mingw, you can continue to follow us there. MinGW: A native Windows port of the GNU Compiler Collection (GCC), freely distributable import libraries and header files for building native Windows applications; includes extensions to the MSVC runtime to support C99 functionality. All MinGW software can run on 64-bit Windows platforms.
