详解PHP设计模式之依赖注入模式
依赖注入模式:依赖注入是控制反转的一种实现方式。要实现控制反转,通常的解决方案是将创建被调用者实例的工作交由 IoC 容器来完成,然后在调用者中注入被调用者(通过构造器 / 方法注入实现),这样我们就实现了调用者与被调用者的解耦,该过程被称为依赖注入。
目的
实现了松耦合的软件架构,可得到更好的测试,管理和扩展的代码
用法
DatabaseConfiguration 被注入 DatabaseConnection 并获取所需的 $config 。如果没有依赖注入模式, 配置将直接创建 DatabaseConnection 。这对测试和扩展来说很不好。
例子
Doctrine2 ORM 使用依赖注入。 例如,注入到 Connection 对象的配置。 对于测试而言, 可以轻松的创建可扩展的模拟数据并注入到 Connection 对象中。
Symfony 和 Zend Framework 2 已经有了依赖注入的容器。他们通过配置的数组来创建对象,并在需要的地方注入 (在控制器中)。
UML 图
代码DatabaseConfiguration.php
- <?php
- namespace DesignPatterns\Structural\DependencyInjection;
- class DatabaseConfiguration
- {
- /**
- * @var string
- */
- private $host;
- /**
- * @var int
- */
- private $port;
- /**
- * @var string
- */
- private $username;
- /**
- * @var string
- */
- private $password;
- public function __construct(string $host, int $port, string $username, string $password)
- {
- $this->host = $host;
- $this->port = $port;
- $this->username = $username;
- $this->password = $password;
- }
- public function getHost(): string
- {
- return $this->host;
- }
- public function getPort(): int
- {
- return $this->port;
- }
- public function getUsername(): string
- {
- return $this->username;
- }
- public function getPassword(): string
- {
- return $this->password;
- }
- }
- ?>
DatabaseConnection.php
- <?php
- namespace DesignPatterns\Structural\DependencyInjection;
- class DatabaseConnection
- {
- /**
- * @var DatabaseConfiguration
- */
- private $configuration;
- /**
- * @param DatabaseConfiguration $config
- */
- public function __construct(DatabaseConfiguration $config)
- {
- $this->configuration = $config;
- }
- public function getDsn(): string
- {
- // 这仅仅是演示,而不是一个真正的 DSN
- // 注意,这里只使用了注入的配置。 所以,
- // 这里是关键的分离关注点。
- return sprintf(
- '%s:%s@%s:%d',
- $this->configuration->getUsername(),
- $this->configuration->getPassword(),
- $this->configuration->getHost(),
- $this->configuration->getPort()
- );
- }
- }
- ?>
测试Tests/DependencyInjectionTest.php
- <?php
- namespace DesignPatterns\Structural\DependencyInjection\Tests;
- use DesignPatterns\Structural\DependencyInjection\DatabaseConfiguration;
- use DesignPatterns\Structural\DependencyInjection\DatabaseConnection;
- use PHPUnit\Framework\TestCase;
- class DependencyInjectionTest extends TestCase
- {
- public function testDependencyInjection()
- {
- $config = new DatabaseConfiguration('localhost', 3306, 'domnikl', '1234');
- $connection = new DatabaseConnection($config);
- $this->assertEquals('domnikl:1234@localhost:3306', $connection->getDsn());
- }
- }
- ?>