Unity应用架构设计(8)——使用ServiceLocator实现对象的注入

简介:

对象的 『注入』 是企业级软件开发经常听到的术语。如果你是一个 Java 程序员,一定对注入有着深刻的映像。不管是SSH框架还是SSM框架,Spring 全家桶永远是绕不过去的弯。通过依赖注入,可以有效的解耦应用程序。在uMVVM框架中,我提供了另外一种对象注入的方式,称为Service Locator 『服务定位模式』 。与Spring的依赖注入不同的是,Service Locator 内部以字典的形式维护了对象的依赖关系,外部通过Key的形式获取 『Resolve』 到对应的Value,从而达到解耦。

为什么要注入对象

简而言之,为了解耦,达到 不去依赖 具体的对象。

实际上解耦是个非常 『虚』 的概念,只有软件到达一定的复杂度之后才会明白解耦和的好处,对于一个简单如『Hello World』程序而言,你很难理解为什么需要解耦。

假设有个 Foo 类,需要通过调用 SomeService 对象的方法去执行一些任务。很简单的需求,你可能会这样写:

public class Foo
{
    ISomeService _service;

    public Foo()
    {
        _service = new SomeService();
    }

    public void DoSomething()
    {
        _service.PerformTask();
        ...
    }
}

这当然没问题,但有隐患,Foo 紧耦合了 SomeService,当需求变了,你不得不打开 Foo 类,然后找到构造函数,重新调用另外的 Service,改完之后编译,然后部署、测试等等。如果是Web程序,你还得在等到晚上去部署。

既然紧耦合了,那就解耦,你可能会这样写:

public class Foo
{
    ISomeService _service;

    public Foo(ISomeService service)
    {
        _service = service;
    }

    public void DoSomething()
    {
        _service.PerformTask();
        ...
    }
}

这样很不错,Foo 与具体的 Service 解耦了,那怎样去实例化 Foo 呢?比如有一个 Bar 类:

public class Bar
{
    public void DoSomething()
    {
        var foo = new Foo(new SomeService());
        foo.DoSomething();
        ...
    }
}

遗憾的是,BarSomeService 又耦合了。然后你又改成这样:

public class Bar
{
    ISomeService _service;

    public Bar(ISomeService service)
    {
        _service = service;
    }

    public void DoSomething()
    {
        var foo = new Foo(_service);
        foo.DoSomething();
        ...
    }
}

通过构造函数传递参数,BarSomeService 解耦了。但你打算怎样去实例化 Bar 呢?

额...(-。-;)

Spring中的依赖注入

Spring中将上述 Foo、Bar 类对SomeService的依赖关系,通过构造函数或者setter方法来实现对象的注入。

<!-- 创建对象:Bar-->
<bean id="barId" class="com.xxx.Bar" >
    <property name="service" ref="someServiceId"></property>
</bean>

<!-- 创建SomeService实例 -->
<bean id="someServiceId" class="com.xxx.SomeService"></bean>

可以看到Spring将依赖关系配置到XML中,在运行时,从IoC容器工厂获取 『Bean(即:实例)』 并将依赖关系注入。

难道我们需要在Unity3D 中定义XML来配置吗?这会不会太麻烦了?

使用ServiceLocator实现对象的注入

其实对象的 『注入』 有很多实现方式,依赖注入 『DI』 只是其中一种,大名鼎鼎的Spring框架就是非常优秀的依赖注入框架,而uMVVM中实现的注入式通过ServiceLocator实现。

什么是ServiceLocator?

简单说ServiceLocator内部以字典的形式维护了对象的依赖关系,外部通过Key的形式获取到对应的Value,从而达到解耦,如下图所示:

要实现对象的 『注入』 ,还缺一个非常重要的对象,就是IoC容器工厂,所有需要被注入的对象都是由容器工厂创建。那我们哪里去找工厂呢?还记得上篇文章的内容了吗?我们已经预先定义了3种不同创建对象的工厂,他们分别为 Singleton Factory,Transient Factory以及 Pool Factory,这些就是我们需要的IoC工厂。

既然 ServiceLocator内部以字典的形式维护了依赖关系,那么首先需要创建一个字典:

private static readonly Dictionary<Type, Func<object>> Container = new Dictionary<Type, Func<object>>();

注意到字典的Value了吗,这是一个 Fun ,本质上是一段匿名函数,只有当真正需要的时候,执行这段匿名函数,返回对象。这是一个非常好的设计,也是懒加载的核心。Swift 和 C# 4.0 的Lazy 核心和代码就是匿名函数。

我们再对Service Locator进行增强,既然要通过字典来维护依赖关系,那我们必须往字典里注册它们,结合我们的工厂,通过ServiceLocator获取的对象可以是单例Singleton对象或者临时Transient对象:

/// <summary>
/// 对每一次请求,只返回唯一的实例
/// </summary>
/// <typeparam name="TInterface"></typeparam>
/// <typeparam name="TInstance"></typeparam>
public static void RegisterSingleton<TInterface, TInstance>() where TInstance : class, new()
{
    Container.Add(typeof(TInterface), Lazy<TInstance>(FactoryType.Singleton));
}
/// <summary>
/// 对每一次请求,只返回唯一的实例
/// </summary>
/// <typeparam name="TInstance"></typeparam>
public static void RegisterSingleton<TInstance>() where TInstance : class, new()
{
    Container.Add(typeof(TInstance), Lazy<TInstance>(FactoryType.Singleton));
}
/// <summary>
/// 对每一次请求,返回不同的实例
/// </summary>
/// <typeparam name="TInterface"></typeparam>
/// <typeparam name="TInstance"></typeparam>
public static void RegisterTransient<TInterface, TInstance>() where TInstance : class, new()
{
    Container.Add(typeof(TInterface),Lazy<TInstance>(FactoryType.Transient));
}
/// <summary>
/// 对每一次请求,返回不同的实例
/// </summary>
/// <typeparam name="TInstance"></typeparam>
public static void RegisterTransient<TInstance>() where TInstance : class, new()
{
    Container.Add(typeof(TInstance),Lazy<TInstance>(FactoryType.Transient));
}

private static Func<object> Lazy<TInstance>(FactoryType factoryType) where TInstance : class, new()
{
    return () =>
    {
        switch (factoryType)
        {
            case FactoryType.Singleton:
                return _singletonObjectFactory.AcquireObject<TInstance>();
            default:
                return _transientObjectFactory.AcquireObject<TInstance>();
        }
        
    };
}

可以看到,其实本质上真的很简单,就是一个 Key:Value 形式的字典,最后提供一个Resolve方法,可以通过Key来获取对应的对象:

/// <summary>
/// 从容器中获取一个实例
/// </summary>
/// <returns></returns>
private static object Resolve(Type type)
{
    if (!Container.ContainsKey(type))
    {
        return null;
    }
    return Container[type]();
 }

使用起来也非常方便,在一个全局初始化文件中去定义这些依赖关系:

 ServiceLocator.RegisterSingleton<IUnitRepository,UnitRepository>();

然后在你的任何业务代码中以Resolve的形式获取对象:

 ServiceLocator.Resolve<IUnitRepository>();

小结

使用构造函数或者setter方法依赖注入也好,还是使用ServiceLocator也罢,它们本质上都是去解耦。对象的注入一般需要结合IoC容器,我们已经定义了3种不同的IoC工厂容器。详细可以翻阅前一篇文章:『Unity 3D Framework Designing(7)——IoC工厂理念先行』。这两篇文章对于初学者来说是有难度的,因为很多概念都是Web开发经常遇到的,如果需要额外的资料,建议翻阅 《设计模式》 相关书籍。

源代码托管在Github上,点击此了解

本博客为 木宛城主原创,基于 Creative Commons Attribution 2.5 China Mainland License发布,欢迎转载,演绎或用于商业目的,但是必须保留本文的署名 木宛城主(包含链接)。如您有任何疑问或者授权方面的协商,请给我留言。

本文转自木宛城主博客园博客,原文链接:http://www.cnblogs.com/OceanEyes/p/using_service_locator.html,如需转载请自行联系原作者
目录
相关文章
|
2天前
|
人工智能 物联网
VideoPainter:开源视频修复神器!双分支架构一键修复,对象身份永久在线
VideoPainter 是由香港中文大学、腾讯ARC Lab等机构联合推出的视频修复和编辑框架,基于双分支架构和预训练扩散模型,支持任意长度视频的修复与编辑,具备背景保留、前景生成、文本指导编辑等功能,为视频处理领域带来新的突破。
45 12
|
12天前
|
存储 缓存 API
类似ComfyUI和Midjourney这样的文生图图生图应用的API与服务架构该怎么设计
文生图图生图应用的API与服务架构分析。或和微服务类似,但是不同。ComfyUI其 API 架构设计为我们理解此类应用提供了很好的参考模型。但距离生产级别的应用差距还有很远。
42 0
|
12天前
|
消息中间件 人工智能 自然语言处理
基于 RocketMQ 事件驱动架构的 AI 应用实践
基于 RocketMQ 事件驱动架构的 AI 应用实践
|
2月前
|
容灾 网络协议 数据库
云卓越架构:云上网络稳定性建设和应用稳定性治理最佳实践
本文介绍了云上网络稳定性体系建设的关键内容,包括面向失败的架构设计、可观测性与应急恢复、客户案例及阿里巴巴的核心电商架构演进。首先强调了网络稳定性的挑战及其应对策略,如责任共担模型和冗余设计。接着详细探讨了多可用区部署、弹性架构规划及跨地域容灾设计的最佳实践,特别是阿里云的产品和技术如何助力实现高可用性和快速故障恢复。最后通过具体案例展示了秒级故障转移的效果,以及同城多活架构下的实际应用。这些措施共同确保了业务在面对网络故障时的持续稳定运行。
|
3月前
|
运维 Cloud Native 持续交付
深入理解云原生架构及其在现代企业中的应用
随着数字化转型的浪潮席卷全球,企业正面临着前所未有的挑战与机遇。云计算技术的迅猛发展,特别是云原生架构的兴起,正在重塑企业的IT基础设施和软件开发模式。本文将深入探讨云原生的核心概念、关键技术以及如何在企业中实施云原生策略,以实现更高效的资源利用和更快的市场响应速度。通过分析云原生架构的优势和面临的挑战,我们将揭示它如何助力企业在激烈的市场竞争中保持领先地位。
101 13
|
4月前
|
运维 Kubernetes Docker
深入理解容器化技术及其在微服务架构中的应用
深入理解容器化技术及其在微服务架构中的应用
135 1
|
4月前
|
Cloud Native 安全 持续交付
深入理解微服务架构及其在现代软件开发中的应用
深入理解微服务架构及其在现代软件开发中的应用
128 33
|
4月前
|
监控 持续交付 API
深入理解微服务架构及其在现代应用开发中的应用
深入理解微服务架构及其在现代应用开发中的应用
57 4
|
4月前
|
边缘计算 监控 自动驾驶
揭秘云计算中的边缘计算:架构、优势及应用场景
揭秘云计算中的边缘计算:架构、优势及应用场景
|
4月前
|
存储 监控 API
深入解析微服务架构及其在现代应用中的实践
深入解析微服务架构及其在现代应用中的实践
124 12

热门文章

最新文章