search
HomeBackend DevelopmentPHP TutorialPHP Inversion of Control and Dependency Injection

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 &#39;b&#39;;
	}
}

class C
{
	public function C()
	{
		//TODO
	}
	public function Method()
	{
		//TODO
		echo &#39;c&#39;;
	}
}

$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 the

function 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 B

objectThe creation of class b and C object c has been moved outside class A, so once class B and class C are changed, class A does not need to be modified, just change it in the client class

If 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 &#39;b&#39;;
	}
}
class B1 extends B
{
	public function B1()
	{
		//TODO
	}
	public function Method()
	{
		echo &#39;b1&#39;;
	}
}
class B2 extends B
{
	public function B2()
	{
		//TODO
	}
	public function Method()
	{
		echo &#39;b2&#39;;
	}
}

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 &#39;B&#39;:
			{
				return new B();
				break;
			}
			case &#39;C&#39;:
			{
				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(&#39;B&#39;);
		$this->c=$f->create(&#39;C&#39;);
		
		$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


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 &#39;b&#39;;
	}
}

class C implements IMethod
{
	public function C()
	{
		//TODO
	}
	public function Method()
	{
		//TODO
		echo &#39;c&#39;;
	}
}

Summary Several points:

1. We move the creation of class B objects and class C objects in class A Out of Class A

2. 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!

Statement
The content of this article is voluntarily contributed by netizens, and the copyright belongs to the original author. This site does not assume corresponding legal responsibility. If you find any content suspected of plagiarism or infringement, please contact admin@php.cn
手把手带你了解Angular中的依赖注入手把手带你了解Angular中的依赖注入Dec 02, 2022 pm 09:14 PM

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

在Phalcon框架中使用依赖注入(Dependency Injection)的方法在Phalcon框架中使用依赖注入(Dependency Injection)的方法Jul 30, 2023 pm 09:03 PM

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

Golang函数参数传递中的依赖注入模式Golang函数参数传递中的依赖注入模式Apr 14, 2024 am 10:15 AM

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

使用JUnit单元测试框架进行依赖注入使用JUnit单元测试框架进行依赖注入Apr 19, 2024 am 08:42 AM

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

深入理解Go语言中的控制反转深入理解Go语言中的控制反转Apr 08, 2024 am 08:51 AM

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

PHP 函数的依赖注入和服务容器PHP 函数的依赖注入和服务容器Apr 27, 2024 pm 01:39 PM

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

如何在 Golang 中使用依赖注入进行单元测试?如何在 Golang 中使用依赖注入进行单元测试?Jun 02, 2024 pm 08:41 PM

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

Go语言:依赖注入指南Go语言:依赖注入指南Apr 07, 2024 pm 12:33 PM

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

See all articles

Hot AI Tools

Undresser.AI Undress

Undresser.AI Undress

AI-powered app for creating realistic nude photos

AI Clothes Remover

AI Clothes Remover

Online AI tool for removing clothes from photos.

Undress AI Tool

Undress AI Tool

Undress images for free

Clothoff.io

Clothoff.io

AI clothes remover

AI Hentai Generator

AI Hentai Generator

Generate AI Hentai for free.

Hot Tools

SublimeText3 English version

SublimeText3 English version

Recommended: Win version, supports code prompts!

SAP NetWeaver Server Adapter for Eclipse

SAP NetWeaver Server Adapter for Eclipse

Integrate Eclipse with SAP NetWeaver application server.

WebStorm Mac version

WebStorm Mac version

Useful JavaScript development tools

SublimeText3 Linux new version

SublimeText3 Linux new version

SublimeText3 Linux latest version

MinGW - Minimalist GNU for Windows

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.