- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
我正在学习依赖注入(inject)和控制反转,我想我开始理解它是如何工作的:
如果一切正确,我是否可以不再在我的对象中使用我所谓的“引用方法”?
这就是我所说的引用方法。假设我有两个模型用于家庭和家庭成员。我发现创建引用与该模型相关的对象的方法非常有帮助。在下面的示例中,当调用 $family->members()
时,我可以快速访问所有家庭成员。但是,这意味着我的 family
对象正在实例化 family_member
类……这不会违反 IoC 规则吗?
如果 family_member
类具有超出 family
类范围的依赖项怎么办?在这里输入将非常有用!
<?php
class family
{
public $id;
public function members()
{
// Return an array of family_member objects
}
}
class family_member
{
public $family_id;
public $first_name;
public $last_name;
public $age;
}
最佳答案
免责声明:我只是自己学习 DI。对答案持保留态度。
依赖注入(inject)只是注入(inject)依赖。如果您的面向对象设计导致 Family
对象负责创建 Member
的实例,那么一定要让 Family
对象创建Member
,因为在那种情况下,Member
不再被视为 Family
的依赖项,而是 责任。因此:
class Family
{
/**
* Constructor.
*
* Since you have decided in your OO design phase that this
* object should have the responsibility of creating members,
* Member is no longer a dependency. MySQLi is, since you need
* it to get the information to create the member. Inject it.
*
*/
public function __construct($id, MySQLi $mysqli)
{
$this->id = $id;
$this->mysqli = $mysqli;
}
/**
* Query the database for members data, instantiates them and
* return them.
*
*/
public function getMembers()
{
// Do work using MySQLi
}
}
但是如果你仔细想想,Family
真的应该有创建Member
的责任吗? 更好的设计是有另一个对象,例如 FamilyMapper
创建 Family
及其成员。像这样:
class FamilyMapper
{
/**
* Constructor.
*
* A better OO design, imho is using the DataMapper pattern.
* The mapper's responsibility is instantiating Family,
* which means it's going to have to connect to the database,
* which makes MySQLi its dependency. So we inject it.
*
*/
public function __construct(MySQLi $mysqli)
{
$this->mysqli = $mysqli;
}
public function findByID($familyID)
{
// Query database for family and members data
// Instantiate and return them
}
}
class Family
{
/**
* Constructor.
*
* Family is an object representing a Family and its members,
* along with methods that *operate* on the data, so Member
* in this OO design is a dependency. Inject it.
*
*/
public function __construct($id, MemberCollection $members)
{
$this->id;
$this->members;
}
public function getMembers()
{
return $this->members;
}
}
使用此模式,您的域对象及其方法(可能包含业务逻辑)将与您的数据访问代码分离。这就是依赖注入(inject)的好处 - 它迫使您重新考虑您的 OO 设计,以便您最终得到更清晰的代码。
许多人认为使用依赖注入(inject)意味着不使用工厂等。 这是错误的!依赖注入(inject)只是注入(inject)依赖。您也可以对工厂对象使用依赖注入(inject),方法是将依赖注入(inject)工厂而不是让工厂实例化自己的依赖。
有用的链接:
同样,对下面的内容持保留态度。
另请注意,依赖注入(inject) 和依赖注入(inject)容器 之间存在差异。第一个是注入(inject)依赖项而不是让对象自己创建依赖项的简单概念(这会导致非常高的耦合)。我们从上面的例子中看到了这一点。
后者是处理依赖注入(inject)的框架/库的术语,因此您不必进行手动注入(inject)。容器的职责是连接依赖项,因此您不必做脏活累活。这个想法是你定义一个依赖注入(inject)配置,它告诉容器 Foo
对象有什么依赖,以及如何注入(inject)它们。容器读取文档并为您执行注入(inject)。 Pimple、SimpleDIC 等 DIC 库就是这样做的。
您可以将依赖注入(inject)容器与工厂进行比较,因为两者都是创建对象,其唯一职责是创建对象。虽然工厂通常是专门的(即 FamilyMemberFactory
创建 MemberInterface
的实例),但依赖注入(inject)容器更为通用。有人说使用依赖注入(inject)容器可以让您摆脱对工厂的需求,但您应该记住,这意味着您必须创建和维护依赖注入(inject)配置文件,这可能是数千行 XML/PHP 行。
希望对您有所帮助。
关于php - 了解 IoC、DI 和引用方法,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/7881378/
在我的 ZF2 应用程序中,我使用 Zend\Di\Di创建我所有的类实例。使用 Zend\Di\Definition\CompilerDefinition 扫描 DI 定义并使用 APC 缓存。这避
为什么人们使用 Spring DI 而不是 JSR330 DI?我看到许多项目仍在高速推进,而 spring DI 却忽视了 JSR330 规范。许多人甚至不知道它的存在。是不是它的营销力度不够,而
我正在使用 spring 制作一个 Web 应用程序,在 web.xml 中我定义了 context-param 来查找 application-context.xml 文件,该文件扫描除 Contr
给定的是 Intel 8086 处理器的汇编程序,它将数组中的数字相加: .model small .stack 100h .data array dw 1,2,3,1,2 sum
.NET 和 Java 都有大量可用的 DI/IoC 容器,并且每个 有许多我发现在不同方面非常有用的模式 与他们合作。我现在正处于我想做等价的地步 JavaScript 中的东西。由于 JavaSc
我有一个使用 Spring 进行 DI 的 Swing 项目,现在我正在尝试迁移到 Eclipse 4 和 OSGi. 使用 Spring 的配置文件,用户可以注释/取消注释 bean,以添加/删除功
Spring 有两种两种类型的 DI:setter DI 和构造 DI。 基于构造函数的 DI 修复了需要注入(inject)依赖项的顺序。基于 Setter 的 DI 不提供此功能。 基于 Sett
TL;博士 在 JCenter 访问 Kodein 核心包是未经授权的。 详情 我们正在使用 Kodein 进行依赖注入(inject),但是当 Gradle 尝试下载任何 org.kodein.*
我已经使用 NInject 一段时间了,现在我将在 Asp.net Core 中开始一个项目。好像NInject cannot be used with Asp.net Core .所以现在我的问题是
目前缺乏有关 DI 主题的文档 - Dependency Injection 。与现有解决方案(Ninject、Autofac、StructureMap)相比,使用内置 DI 有何优点/缺点?默认依赖
我想做的是将两个 Actor (木乃伊 Actor 和爸爸 Actor )传递给小 Actor 。由于使用 actor 引用而不是 actor 是最佳实践,因此我使用 IActorRef 将木乃伊 a
我是 MQL4 的菜鸟,我正在编写我的第一个 EA。 我的目标是获取 ADX 指标的 +DI 和 -DI 的变量。 我使用了 iADX() 函数,如下所示: double a; int OnInit(
我有一个环境,有 4 个相同的设备,我必须连接到这些设备并通过 TCP 连接请求一些参数(每个设备都有其 IP 地址)。 我为需要一些参数的单个设备实现了一个类(如 IP 地址、端口、轮询间隔等...
我正在尝试将 DI(使用 Autofac)引入现有的 Windows 窗体应用程序。 此应用程序有一个基本的插件架构,其中每个插件都显示自己的表单。在启动时,应用程序会扫描已注册的程序集以查找实现 I
我有一个基于 .NET Core API Gateway 的项目。我想引入依赖注入(inject)(di),因为我需要引入的很多包都是基于这种模式的,所以需要使用 IServiceCollection
我正在尝试使用蛋糕模式进行依赖注入(inject),如下所示: trait FooComponent { val foo: Foo trait Foo; } trait AlsoNeedsFo
我即将创建一个桌面应用程序(使用 .NET windows 窗体) 本质上,我想创建一个 n 层应用程序,但我也想要层之间的松散耦合。但是,我不太确定这是否是 Windows 窗体的好方法 现在我只是
我想在我的一个项目中使用依赖注入(inject) (DI)。我写了一个基本的 DI 库,其工作原理如下 let di = new DI(); di.register('screen', Screen)
在: module.directive 'name', -> (scope, element, attr) -> # Whatever implemenation 链接函数的 scope、
我使用这个库https://github.com/smsapi/smsapi-php-client从站点发送短信。但是当我尝试处理服务中的基类时遇到了问题。所以我的问题是有没有办法用参数调用静态方法?
我是一名优秀的程序员,十分优秀!