ioc框架 java_从零开始实现一个简易的Java MVC框架(三)--实现IOC

Spring中的IOC

IoC全称是Inversion of Control,就是控制反转,他其实不是spring独有的特性或者说也不是java的特性,他是一种设计思想。而DI(Dependency Injection),即依赖注入就是Ioc的一种实现方式。关于Ioc和DI的具体定义和优缺点等大家可以自行查找资料了解一下,这里就不详细赘述,总之spring的IoC功能很大程度上便捷了我们的开发工作。

在实现我们的Ioc之前,我们先了解一下spring的依赖注入,在spring中依赖注入有三种方式,分别是:

接口注入(Interface Injection)

设值方法注入(Setter Injection)

构造注入(Constructor Injection)

@Component

public class ComponentA {

@Autowired // 1.接口注入

private ComponentB componentB;

@Autowired // 2.设值方法注入

public void setComponentB(ComponentB componentB) {

this.componentB = componentB;

}

@Autowired // 3.构造注入

public ComponentA(ComponentB componentB) {

this.componentB = componentB;

}

}

循环依赖注入

如果只是实现依赖注入的话实际上很简单,只要利用java的反射原理将对应的属性‘注入’进去就可以了。但是必须要注意一个问题,那就是循环依赖问题。循环依赖就是类之间相互依赖形成了一个循环,比如A依赖于B,同时B又依赖于A,这就形成了相互循环。

// ComponentA

@Component

public class ComponentA {

@Autowired

private ComponentB componentB;

}

// ComponentB

@Component

public class ComponentB {

@Autowired

private ComponentA componentA;

}

那么在spring中又是如何解决循环依赖问题的呢,我们大致说一下原理。

如果要创建一个类,先把这个类放进'正在创建池'中,通过反射等创建实例,创建成功的话就把这个实例放入创建池中,并移除'正在创建池'中的这个类。每当实例中有依赖需要注入的话,就从创建池中找对应的实例注入进去,如果没有找到实例,则先创建这个依赖。

利用了这个正在创建的中间状态缓存,让Bean的创建的时候即使有依赖还没有实例化,可以先把Bean放进这个中间状态,然后跑去创建那个依赖,假如那个依赖的类又依赖与这个Bean,那么只要在'正在创建池'中再把这个Bean拿出来,注入到这个依赖中,就可以保证Bean的依赖能够实例化完成。再回头来把这个依赖注入到Bean中,那么这个Bean也实例化完成了,就把这个Bean从'正在创建池'移到'创建完成池'中,就解决了循环依赖问题。

虽然spring巧妙的避免了循环依赖问题,但是事实上构造注入是无法避免循环依赖问题的。因为在实例化ComponentA的构造函数的时候必须得到ComponentB的实例,但是实例化ComponentB的构造函数的时候又必须有ComponentA的实例。这两个Bean都不能通过反射实例化然后放到'正在创建池',所以无法解决循环依赖问题,这时候spring就会主动抛出BeanCurrentlyInCreationException异常避免死循环。

* 注意,前面讲的这些都是基于spring的单例模式下的,如果是多例模式会有所不同,大家有兴趣可以自行了解。

实现IOC

现在可以开始实现IOC功能了

增加注解

先在zbw.ioc包下创建一个annotation包,然后再创建一个Autowired的注解。这个注解的Target只有一个ElementType.FIELD,就是只能注解在属性上。意味着我们目前只实现接口注入的功能。这样可以避免构造注入造成的循环依赖问题无法解决,而且接口注入也是用的最多的方式了。如果想要实现设值方式注入大家可以自己去实现,实现原理几乎都一样。

package com.zbw.ioc.annotation;

import ...

@Target(ElementType.FIELD)

@Retention(RetentionPolicy.RUNTIME)

public @interface Autowired {

}

实现IOC类

package com.zbw.ioc;

import ...

@Slf4j

public class Ioc {

/**

* Bean容器

*/

private BeanContainer beanContainer;

public Ioc() {

beanContainer = BeanContainer.getInstance();

}

/**

* 执行Ioc

*/

public void doIoc() {

for (Class> clz : beanContainer.getClasses()) { //遍历Bean容器中所有的Bean

final Object targetBean = beanContainer.getBean(clz);

Field[] fields = clz.getDeclaredFields();

for (Field field : fields) { //遍历Bean中的所有属性

if (field.isAnnotationPresent(Autowired.class)) {// 如果该属性被Autowired注解,则对其注入

final Class> fieldClass = field.getType();

Object fieldValue = getClassInstance(fieldClass);

if (null != fieldValue) {

ClassUtil.setField(field, targetBean, fieldValue);

} else {

throw new RuntimeException("无法注入对应的类,目标类型:" + fieldClass.getName());

}

}

}

}

}

/**

* 根据Class获取其实例或者实现类

*/

private Object getClassInstance(final Class> clz) {

return Optional

.ofNullable(beanContainer.getBean(clz))

.orElseGet(() -> {

Class> implementClass = getImplementClass(clz);

if (null != implementClass) {

return beanContainer.getBean(implementClass);

}

return null;

});

}

/**

* 获取接口的实现类

*/

private Class> getImplementClass(final Class> interfaceClass) {

return beanContainer.getClassesBySuper(interfaceClass)

.stream()

.findFirst()

.orElse(null);

}

}

在知道IOC的原理之后发现其实真的是非常简单,这里用了几十行的代码就实现了IOC的功能。

首先在Ioc类构造的时候先获取到我们之前已经单例化的BeanContainer容器。

然后在doIoc()方法中就是正式实现IOC功能的了。

遍历在BeanContainer容器的所有Bean

对每个Bean的Field属性进行遍历

如果某个Field属性被Autowired注解,则调用getClassInstance()方法对其进行注入

getClassInstance()会根据Field的Class尝试从Bean容器中获取对应的实例,如果获取到则返回该实例,如果获取不到,则我们认定该Field为一个接口,我们就调用getImplementClass()方法来获取这个接口的实现类Class,然后再根据这个实现类Class在Bean容器中获取对应的实现类实例。

测试用例

为了测试我们的Ioc和之前写的BeanContainer编写正确,我们写一下测试用例测试一下。

先在pom.xml添加junit的依赖

...

4.12

...

junit

junit

${junit.version}

test

然后在test包下添加DoodleController、DoodleService、DoodleServiceImpl三个类方便测试

// DoodleController

package com.zbw.bean;

@Controller

@Slf4j

public class DoodleController {

@Autowired

private DoodleService doodleService;

public void hello() {

log.info(doodleService.helloWord());

}

}

// DoodleService

package com.zbw.bean;

public interface DoodleService {

String helloWord();

}

// DoodleServiceImpl

package com.zbw.bean;

@Service

public class DoodleServiceImpl implements DoodleService{

@Override

public String helloWord() {

return "hello word";

}

}

再编写IocTest的测试用例

package com.zbw.ioc;

import ...

@Slf4j

public class IocTest {

@Test

public void doIoc() {

BeanContainer beanContainer = BeanContainer.getInstance();

beanContainer.loadBeans("com.zbw");

new Ioc().doIoc();

DoodleController controller = (DoodleController) beanContainer.getBean(DoodleController.class);

controller.hello();

}

}

8ba9472efd2e2d72d17873432cda1680.png

看到在DoodleController中输出了DoodleServiceImpl的helloWord()方法里的字符串,说明DoodleController中的DoodleService已经成功注入了DoodleServiceImpl。那么我们的IOC的功能也完成了。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/246430.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

b树索引 java_B树索引最通俗易懂的介绍

先来一段有莫的对话:前几天下班回到家后正在处理一个白天没解决的bug,厕所突然传来对象的声音:对象:xx,你有《时间简史》吗?我:我去!妹子,你这啥癖好啊,我有时…

jython mysql_Jython

gomysql###介绍gomysql是基于go-sql-driver基础上开发的orm,这是一个轻量级的库。它会使数据库的增删改查变得非常容易。当然也是测试开发版,会一直优化和更新!请时刻关注我们###安装go get github.com/go-sql-driver/mysqlgo get github.com…

java人种_Java面向对象练习题继承之人种

//People类package 继承;public class People {private String name;private double height;private double weight;public String getName() {return name;}public void setName(String name) {this.name name;}public double getHeight() {return height;}public void setHe…

java高校职工工资管理论文_毕业设计论文java大学工资管理系统

毕业设计论文java大学工资管理系统 本 科 生 毕 业 论 文(设 计)题 目:大学工资管理系统 学 号: _________姓 名: ____ ___年 级: ___________学 院: ____ ___系 别: ___ ____专 业: ___ __完成…

mysql 冷热表_Redis+MySQL冷热数据交换

场景:某网站需要对其项目做一个投票系统,投票项目上线后一小时之内预计有100万用户进行投票,希望用户投票完就能看到实时的投票情况这个场景可以使用redismysql冷热数据交换来解决。何为冷热数据交换?冷数据:之前使用的…

关于包装java_[java初探09]__关于java的包装类

前言在Java语言的学习过程中,我们逐渐的理解了Java面向对象的思想,与类和对象的应用.但是在基本数据类型的使用上,我们无法将其定义为一个对象,通过使用对象的方法来使用它们,但是Java语言的思想又是面向对象的.那么在Java语言中,是否能够解决这个问题,使基本数据类型能够以对象…

能跑java的服务器_一台java服务器可以跑多少个线程?

一台java服务器能跑多少个线程?这个问题来自一次线上报警如下图,超过了我们的配置阈值。京东自研UMP监控分析打出jstack文件,通过IBM Thread and Monitor Dump Analyzer for Java工具查看如下:IBM Thread and Monitor Dump Analyz…

php取json子对象属性,php中输出json对象的值(实现方法)

实例如下所示:$json {"report":{"date":"2012-04-10","content":"abcdefght"}};$arr (array) json_decode($json,true);echo 当前日期是:. $arr[report][date];echo "";echo ;print_r($a…

php mysql 表关联,mysql的多表关联_MySQL

bitsCN.commysql的多表关联数据库中经常要用到多个表的关联。mysql的关联主要包括inner join,left join,right join三种,下面分别加以介绍,并举例说明。顾名思义,inner join集合了两个表的信息,只有都包含的…

ulink php,【转载】15款USB数字界面横向评测(对比顶级CD转盘)!多看点!

还有一则有趣的回帖,一位纽约的烧友认为作者没有尝试何庆华最新的数字界面非常可惜,他认为何先生自主开发的DI-V3电源套件非常之牛,甚至超过了AP1PP的水准(不少人认为这个组合有一点点过于分析,不如何先生的作品水润)。他认为以他…

php取掉字符串第一位支付,php怎样去掉字符串中的第一个字符

php去掉字符串中的第一个字符的方法:可以利用substr()函数来实现。substr()函数可以返回字符串的提取部分,如果失败则返回false,或者返回一个空字符串。substr() 函数返回字符串的提取部分,如果失败则返回 FALSE,或者返…

matlab求距离判别函数,求MATLAB的逐步判别程序 - 仿真模拟 - 小木虫 - 学术 科研 互动社区...

somomo91你不觉得信息量太少了么?摸不着头脑,zhouxiaobo是啊,信息量太少,LZ能详细描述一下你的问题吗或者直接给出你的数据和要求2012jxyl引用回帖:zhouxiaobo at 2013-06-06 09:44:49是啊,信息量太少,LZ能…

php验证码背景图是数字,ThinkPHP5.0.20验证码背景图片

tp5配置验证码相关问题一、验证码背景图片及tp中文验证码5.0.1及以上支持者在应用配置目录(application)下面 extra 子目录内配置captcha.php文件;配置参数如下:2345678abcdefhijkmnpqrstuvwxyzABCDEFGHJKLMNPQRTUVWXY,// 验证码字体大小(px)fontSize &g…

c mysql安装教程,Mysql安装教程_完成版(吐血式安装)

每次在不同操作系统中安装oracle和mysql这些常规数据库,步骤就那么点儿,但是遇见的错误却是千差万别。。记一次耗时两天的mysql数据库安装新得,有耐心,有毅力,憋生气。1.官网上下载免安装的版本(也就是下载下来直接解压…

oracle替代变量输出,【Oracle】替代变量

1.替代变量通常而言,替代变量的前缀是&或者&&区别:& 用来创建一个临时变量,每当遇到这个临时变量时,都会提示你输入一个值&&用来创建一个持久变量,当用&&命令引用这个变量时&#xff…

oracle 常用故障,Oracle常见问题解决方案汇总

1、Oracle 11g ORA-12514:TNS:监听程序当前无法识别连接描述符中请求的服务数据库服务器崩了,而且尝试重启服务和重启机器都解决不了问题打开cmd窗口C:\Users\hxt>sqlplus / as sysdbaSQL*Plus: Release 11.2.0.1.0 Production on 星期三 12月 5 11:39:54 2018Co…

特洛伊木马脚本linux,手动查杀特洛伊木马

首先也是最重要的,重新启动电脑到安全模式下,让所有文件都可见。然后进入到C:看看根目录是否存在不熟悉的文件,如果有,且日期为发现中毒现象当天,则删除之。接着到c:\windows,首先按照修改时间顺序排列图标…

linux添加虚拟硬盘命令,虚拟机linux扩盘命令操作

虚拟机linux扩展硬盘分原有硬盘上增加及新增硬盘,二者实现差不多。扩盘:fdisk -l查看是哪个盘扩盘了,如sdb重启后,将sdb刷新pv: pvresize /dev/sdb,pvdisplay查看是否增加了,再将逻辑卷分区VG(L…

mac ssh远程登录linux,MAC使用SSH远程登录

8种机械键盘轴体对比本人程序员,要买一个写代码的键盘,请问红轴和茶轴怎么选?Sun 20 December 2015tags: 备忘录MAC使用SSH远程登录打开sshMac Terminal是自带SSH的,可以用whereis来看看$ whereis ssh但是在现有进程中是找不到ssh…

四叶草引导windows和linux,Windows环境下使用Clover四叶草引导双硬盘安装OSX 10.11.5原版镜像...

作为一个穷逼大学生,想搞iOS开发 买不起Mac只能鼓捣鼓捣黑苹果啦。。。。。。。。之前我的电脑通过变色龙引导的方式装了个OSX10.10和win8.1双系统,因为自学的是Swift语言之前装的OSX10.10.4的Xcode(6.多版本的)只支持到Swift1.2,所以现在要装…