IOC和DI之刨根问底 -- 第一节

很多freshman上来就想搞清楚什么是IOC和DI,其实很多先进的理论和技术都在老的基础上升华出来的,最终目的是为了解放生产力。

所以先来说说下面两点基础知识:

  • Direct Dependency(直接依赖)
  • Inverted Dependency(反向依赖)

Direct Dependency

应用程序中的依赖关系方向应该是抽象的方向,而不是实现详细信息的方向。 大部分应用程序都是这样编写的:编译时依赖关系顺着运行时执行的方向流动,从而生成一个直接依赖项关系图。 也就是说,如果类 A 调用类 B 的方法,类 B 调用 C 类的方法,则在编译时,类 A 将取决于类 B,而 B 类又取决于类 C,如图1所示。

 直接依赖项关系图

假设一个A通过朋友B和C找超级富婆的故事,A只有B朋友的关系,B只有C朋友的关系,C朋友才能帮忙找到超级富婆!条件是“身体好”!

对应的代码块如下:

  public class ClassA
    {
        /// <summary>
        /// Find super rich woman
        /// </summary>
        /// <returns>return super rich woman</returns>
        public string FindRichWoman()
        {
            var criteria = "身体好";
            return new ClassB().FindRichWoman(citeria);
        }
    }
   public class ClassB
    {
        /// <summary>
        /// Find super rich woman by citeria.
        /// </summary>
        /// <returns>return super rich woman</returns>
        public string FindRichWoman(string criteria)
        {return new ClassC().FindRichWoman(criteria);
        }
    }
   public class ClassC
    {
        /// <summary>
        /// Find super rich woman by criteria.
        /// </summary>
        /// <returns>return super rich woman</returns>
        public string FindRichWoman(string criteria)
        {
            if (criteria.Equals("身体好"))
                return "Super Rich Woman";

            return string.Empty;
        }
    }

编译时和运行时的依赖关系和控制关系都是A=》B=》C

  •  高层次类对底层次类正向依赖 - A要想找到富婆就必须要找到B,B需要去找C
  •  高层次类对低层次类的正向控制 - B什么时候要富婆由A来决定,C什么时候去找富婆由B来决定

Inverted Dependency

应用依赖关系反转原则后,A 可以调用 B 实现的抽象上的方法,让 A 可以在运行时调用 B,而 B 又在编译时依赖于 A 控制的接口(因此,典型的编译时依赖项发生反转)。 运行时,程序执行的流程保持不变,但接口引入意味着可以轻松插入这些接口的不同实现。

反转依赖项关系图

  • 依赖关系反转原则之抽象接口和工厂模式应用 - 上面那段话可以这么理解,A发现找个富婆还要自己亲自去找B,还得管B的吃喝拉撒,所以是否可以找机器人中心(工厂模式)帮忙搭线,自己只要找到由B抽象出来的虚拟机器人就可以了

 使用工厂模式后,只需要将抽象接口B给到工厂就能找到想要的方法,A不用去关注对象B是怎么产生的。

  public class ClassA
    {
        /// <summary>
        /// Find super rich woman
        /// </summary>
        /// <returns>return super rich woman</returns>
        public string FindRichWoman()
        {
            var criteria = "身体好";
            return factory.CreateInstance(InterfaceB).FindRichWomanByB(criteria);
        }
    }

 

  •    接口引入意味着可以轻松插入这些接口的不同实现 如果A突然不想通过B找富婆,假设D也不需要任何条件就可以现实找到富婆,那A肯定很乐意换成通过D,我只需要把InterfaceB换成InterfaceD丢给工厂。
   public class ClassD: InterfaceD
    {
        /// <summary>
        /// Find super rich woman
        /// </summary>
        /// <returns>return super rich woman</returns>
        public string FindRichWoman()
        {
           return "Super Rich Woman";
        }
    }
}
  public class ClassA
    {
        /// <summary>
        /// Find super rich woman
        /// </summary>
        /// <returns>return super rich woman</returns>
        public string FindRichWoman()
        {
            return factory.CreateInstance(InterfaceD).FindRichWoman(); 
     }
   }

 发现很多博主在讲解IOC时,就将此处的概念就定义为IOC, 其实此处:

  •  高层次的类不再正向依赖于低层次的类,两者都依赖于抽象接口  - 类A和类B依赖了InterfaceB
  •  低层次类依赖于高层次类的需求抽象 类A有找富婆的需求,那么低层次抽象出来的InterfaceB和InterfaceD都有实现找富婆的办法,所以他们就都有依赖于来自高层次类A抽象出来找富婆的需求

最大的优势就是解耦了高层次模块对于低层次模块的紧密依赖,可以灵活扩展!

到了锻炼身体的时间了,下一节再来说IOC和DI的概念和对应的场景

posted @ 2021-10-28 22:44  ~扎克伯格  阅读(503)  评论(0编辑  收藏  举报