浅谈控制反转(IoC)

article/2025/11/6 21:58:08

Inversion of Control

什么是控制反转?

程序的流程控制权相对于传统的面向过程编程而言发生了反转。下面是维基百科的描述

In software engineering, inversion of control (IoC) is a programming principle. IoC inverts the flow of control as compared to traditional control flow.

看到这里大家可能会觉得云里雾里的…控制反转(Inversion of Control)实际是控制(Control)和反转(Inversion)两个词的组合,所以拨开云雾的关键在于理解控制反转

怎么理解“控制”?

程序的流程控制权,决定程序的运行方式和流程。

举个例子,想必大家都知道,要把大象放进冰箱需要三步:1)打开冰箱门;2)把大象放进冰箱;3)关上冰箱门。用一段代码用表示的话,代码可能长这样

void putElephantToRefrigerator() {打开冰箱门;放进大象;关闭冰箱门;
}int main() {putElephantToRefrigerator();return 0;
}

不过,冰箱可能已经被塞满放不下大象了,我们需要把冰箱里没用的东西清理掉,再把大象团成团放进去,代码会变成这样

void putElephantToRefrigerator() {打开冰箱门;清理冰箱;把大象团成团;放进大象;关闭冰箱门;
}
...

当然,我们完全可以先把大象团成团,像这样

void putElephantToRefrigerator() {把大象团成团;打开冰箱门;清理冰箱;放进大象;关闭冰箱门;
}
...

所以,我们要把大象放进冰箱里都会经过一个流程,目前这个流程是由我们自己控制的。

怎么理解“反转”?

相对于传统的面向过程编程实践而言

  • 程序的流程控制权发生了转变
  • 应用程序与第三方代码之间的调用关系发生了转变

继续看上面的例子,假如我们有一款极其智能的冰箱,不需要手动开门、关门,只要关心放什么东西进去、怎么放进去(nice,事情变得更简单了),代码会变成这样(考虑设计模式-模板方法)

typedef struct AutoRefrigerator;struct MyRefrigerator : AutoRefrigerator {private void putSomething() {...}
}int main() {MyRefrigerator refrigerator;refrigerator.put(elephant);return 0;
}

代码中“refrigerator.put(elephant);”由框架提供,包含多条子语句(open、close、putSomething…),但putSomething没具体实现,需要我们自己定制。

此时,控制程序流程的不是我们的代码而是框架提供的AutoRefrigerator,程序流程控制权发生了反转,控制权由我们自己的代码转移到了框架中

控制反转前后有哪些变化呢?

  • 反转前:我们自己的代码决定程序的工作流程,并调用第三方代码(我们自己的代码是甲方,第三方代码是乙方)
  • 反转后:第三方代码(框架)决定程序的工作流程,并调用我们写的代码(我们自己的代码是乙方,第三方代码是甲方)

可见,在控制反转前我们写的代码会调用第三方代码,而控制反转后则是由第三方代码(框架)调用我们的代码,这种代码调用关系的反转也是控制反转的意义

有什么好处?

好处?怎么可以这么物质?这个问题超纲了,暂且不谈…笔者是个物质的人,所以只谈谈有什么好处。

好处用两个字概括:复用用Paul Graham的话来说,复用就是软件圣杯)。

复用代码有三种方式:类库、框架、设计模式。

  • 类库强调代码复用
    定义一组可复用的代码,供其他程序调用——拿来主义,别人的东西拿来用,用别人的锤子砸核桃。
  • 框架强调设计复用
    定义程序的体系结构,开发人员通过预留的接口插入代码(做填空题)——把自己的锤子装在流水线上,让它砸核桃。
  • 设计模式复用解决方案
    设计模式提供了解决一类问题的有效经验,复用这些经验往往可以很好地解决问题——看别人是怎么砸核桃的,依葫芦画瓢模仿一遍。

控制反转为复用框架提供了可能性。框架定义了解决一类问题的流程、体系结构,但程序的特定细节需要定制,因此框架需要调用我们写的代码,控制反转提供了这种可能性。控制反转和编码原则之一的好莱坞原则(It calls me rather me calling the framework) 也存在着紧密的联系。

另外,按照Martin Fowler的描述,控制反转还有Seperating Configuration from Use的作用。实际上这是一些容器的主要作用(如Spring),而这些容器又不可避免地使用了控制反转。

怎么实践?

说了这么多,又有这么诱人的好处,怎么才能得到它呢?

最简单的实践就是设计模式-模板方法,不过最常见的实践方法还是依赖注入依赖查询
依赖注入(DI:Dependency Inject):被动接收依赖对象,由容器将被依赖对象注入到对象内部;
依赖查询(DL:Dependency Lookup):主动查询依赖对象,由对象自身通过 服务定位器 查询被依赖对象;依赖查询也经常以服务定位器模式(Service Locator)的形式出现。

DIDL
在这里插入图片描述在这里插入图片描述
Context依赖各事例各实例依赖context
通过构造函数、setter函数等容易查看依赖关系依赖关系不容易查看,需要分析调用locator的源码分析依赖关系
装配关系在业务类外的配置文件中装配关系在业务类中
被动接收依赖对象主动查询依赖对象

这两种方式的主要区别在于配置方式不同、获得依赖对象的方式不同。

该怎么决定在代码中使用DI还是DL呢?Martin Fowler有话说

So the decision between locator and injector depends on whether that dependency is a problem.

参考连接

https://www.martinfowler.com/bliki/InversionOfControl.html
https://www.martinfowler.com/articles/injection.html
https://en.wikipedia.org/wiki/Inversion_of_control


http://chatgpt.dhexx.cn/article/3VD7kONq.shtml

相关文章

Spring---浅谈IOC

概念 IOC(Inversion of Control 控制反转)是spring的核心,贯穿始终。所谓IOC,对于spring框架来说,就是由spring来负责控制对象的生命周期和对象间的关系。 传统开发模式与IOC开发模式的对比 传统开发模式:对…

控制反转-Inversion Of Control

一、控制反转 控制反转(Inversion of Control,英文缩写为IoC)把创建对象的权利交给框架,是框架的重要特征,并非面向对象编程的专业术语。它包括依赖注入(Dependency Injection,简称DI&#xff…

微信小程序hamburgers汉堡包css动画

微信小程序hamburgers汉堡包css动画 源码在我的csdn里下载 https://download.csdn.net/download/ozhy111/12201373

汉堡式折叠html,美味的CSS动画汉堡包:hamburgers_html/css_WEB-ITnose

s if you insist, but they’re not accessible as a menu button. Append the class name of the type of hamburger you’re craving: Here’s the list of hamburger-type classes you can choose from: hamburger--arrowhamburger--arrow-rhamburger--arrowalthamburger--a…

LSB 题解

今天来刷一道Misc的题目,LSB原理进行图片隐写 LSB原理 LSB是一种利用人类视觉的局限性设计的幻术 PNG和BMP图片中的图像像素一般是由RGB(RED红 GREEN绿 BLUE蓝)三原色组成 记住,JPG图片是不适合使用LSB隐写的,JPG图片对像数进行了有损压缩…

LSB利器-zsteg

一、Usage # zsteg -h Usage: zsteg [options] filename.png [param_string] -c, --channels X channels (R/G/B/A) or any combination, comma separated valid values: r,g,b,a,rg,bgr,rgba,r3g2b3,... -l, --limi…

【linux】lsb_release -a命令

查看linux发行版本:lsb_release -a 运行该命令时若提示lsb_release命令不存在 不存在,则表明系统未安装该命令,报错如下: 解决方法: 查看 lsb_release 命令属于哪个软件包,命令为yum provides */lsb_rel…

图片LSB隐写(java)

一、隐写原理 LSB隐写原理就是图片中的像素一般是由三种颜色组成,即三原色(红绿蓝)。由这三种原色可以组成其他各种颜色,在png图片的存储中,每个颜色占有8bit,即有256种颜色,一共包含256的三次方颜色,即16777216中颜色。人类的眼睛…

LSB隐写术

前言 LSB全称为 least significant bit,是最低有效位的意思。Lsb图片隐写是基于lsb算法的一种图片隐写术,以下统称为lsb隐写,这是一种常见的信息隐藏方法。当然关于图像的隐写的方法有很多,统称为隐写术, lsb隐写很实…

Linux开发标准LSB简介:Linux Standard Base

目录 Unix/Linux 标准化历史 POSIX Open Group Austin Group LSB LSB 简介 组织架构 工作组 LSB 的标准化流程 认证 认证问题报告 LSB 的历史、现状和将来 实例:lsb_release 的规范定义和实现 结束语 Unix/Linux 标准化历史 标准化目前已经成为 Linu…

LSB 简介

Unix/Linux 标准化历史 标准化目前已经成为 Linux 系统上的一个热门话题。实际上,在 Linux 诞生之初,这个问题就得到了重视。当 Linus 在开发 0.01 版本的 Linux 内核时,就开始关注 POSIX 标准的发展,他在 /include/unistd.h 文件…

图片隐写之LSB(Least Significant Bit)原理及其代码实现

1. 什么是隐写? 隐写术是一门关于信息隐藏的技巧与科学,所谓信息隐藏指的是不让除预期的接收者之外的任何人知晓信息的传递事件或者信息的内容。隐写术的英文叫做Steganography,来源于特里特米乌斯的一本讲述密码学与隐写术的著作Steganogra…

MetaPhlAn2:宏基因组物种组成分析

简介 MetaPhlAn2是分析微生物群落(细菌、古菌、真核生物和病毒)组成的工具,它在宏基因组研究中非常有用,只需一条完命令即可获得微生物的物种丰度信息(扩增子物种组成需要质控、拼接、拆样本、切除引物、比对等步骤,此软件居然分析宏基因组这…

你想要的宏基因组-微生物组知识全在这(2023.4)

欢迎点击上方蓝色”宏基因组”关注我们! 宏基因组/微生物组是当今世界科研最热门的研究领域之一,为加强宏基因组学技术和成果交流传播,推动全球华人微生物组领域发展,中科院青年科研人员创立“宏基因组”公众号,联合海…

Kraken2:宏基因组快速物种注释神器

简介 kraken是基于k-mer精确比对,并采用最LCA投票结果快速宏基因组DNA序列进行物种注释的软件。 图. Kraken2分类基本原理 该文章于2014年发表于Genome Biology,目前引用过两千次[1]。详见《Kraken:使用精确比对的超快速宏基因组序列分类软件…

Nature综述: 宏基因组关联分析-深入研究微生物组

本文由谢忠杰编译,董小橙、江舜尧编辑,本文较长,建议用电脑阅读。 “微生太”原创微文,转载已获授权。 导读 问题1:哪些疾病与人体微生物明确相关? 问题2:如何研究人体微生物与健康的关系&#…

Nature子刊:HUMAnN2实现宏基因组和宏转录组种水平功能组成分析

HUMAnN2实现宏基因组和宏转录组种水平功能组成分析 Species-level functional profiling of metagenomes and metatranscriptomes Nature Methods, [IF 26.919], Article, 2018-10-30 DOI: http://dx.doi.org/10.1038/s41592-018-0176-y 第一作者:Eric A. Franzosa…

从测序到宏基因组:聚焦菌群生信分析方法最前沿

今天是第2439期日报。 Nature子刊:基于三代测序的宏基因组分析助力完整微生物进化研究 Nature Methods[IF:47.99] ① 基于Pacific Biosciences或Oxford Nanopore的三代测序技术已成为获得闭环微生物基因组的常规手段,即使对特殊大量数据的读取&#xff0…

Protein Cell:扩增子和宏基因组数据分析实用指南

扩增子和宏基因组数据分析实用指南 A practical guide to amplicon and metagenomic analysis of microbiome data Protein Cell [IF: 10.164] DOI: https://doi.org/10.1007/s13238-020-00724-8 Review: 2020-5-11 第一作者:刘永鑫1,2,3, 秦媛1,2,3,4, 陈同5 通讯作…

凌恩生物文献分享|颠覆性的宏基因组新思路,速来get!

非人灵长类动物(NHP)是人类的近亲,为宿主-微生物互作的研究提供了一个很好的例子。近年来研究主要集中在野生灵长类动物的肠道微生物群,这将有助于了解灵长类及其肠道微生物群的进化,但仍然缺乏关于野生种群肠道微生物…