Spring IOC

Stella981
• 阅读 585

spring ioc是spring的核心之一,也是spring体系的基础,那么spring ioc所依赖的底层技术是什么的?反射,以前我们开发程序的时候对象之间的相互调用需要用new来实现,现在所有的bean都是通过spring容器来管理。这样做有什么好处呢?解耦!以前程序直接的调用用new直接给写死了,现在我们可以通过注入不同的接口实现类来完成对象直接的调用。

首先来聊聊Java的反射机制

1、反射机制的作用:

反编译:.class-->.java

通过反射机制访问java对象的属性,方法,构造方法等;

2、Java反射机制用途:

在运行时判断任意一个对象所属的类

在运行时构造任意一个类的对象

在运行时判断任意一个类所具有的成员变量和方法

在运行时调用任意一个对象的方法

3、sun为我们提供了那些反射机制中的类:

  java.lang.Class;                
  java.lang.reflect.Constructor;           
  java.lang.reflect.Field;        
  java.lang.reflect.Method;
  java.lang.reflect.Modifier;

4、反射实现的方式

Class c=Class.forName("className");
注明:className必须为全名,也就是得包含包名,比如,cn.xx.UserInfo;

Object obj=c.newInstance();
//创建对象的实例 

Constructor getConstructor(Class[] params)
//根据指定参数获得public构造器 

Constructor[] getConstructors()
//获得public的所有构造器 

Constructor getDeclaredConstructor(Class[] params)
//根据指定参数获得public和非public的构造器 

Constructor[] getDeclaredConstructors()
//获得public的所有构造器 

ewInstance();
//创建对象的实例 

获得类方法的方法
Method getMethod(String name, Class[] params),
根据方法名,参数类型获得方法 

Method[] getMethods()
//获得所有的public方法 

Method getDeclaredMethod(String name, Class[] params)
//根据方法名和参数类型,获得public和非public的方法 

Method[] getDeclaredMethods()
//获得所以的public和非public方法

获得类中属性的方法
Field getField(String name)
//根据变量名得到相应的public变量 

Field[] getFields()
//获得类中所以public的方法 

Field getDeclaredField(String name)
//根据方法名获得public和非public变量 

Field[] getDeclaredFields()
//获得类中所有的public和非public方法

总结一句,以前写的代码,类的熟悉、方法什么东西的都固定了,如果使用反射我们就可以在运行的时候动态的去修改、增删对象的熟悉、方法等等。

spring IOC是如果使用反射来完成对象的注入呢?

1、读取配置文件,或者扫描注解属性

2、根据配置文件,通过反射实例化对象

3、给对象注入依赖的属性

4、放到类似hashMap结构中,供系统调用

/**
* 学习版容器
*
*/

public class LeamClassPathXMLApplicationContext {
    private List<Definition> beanDefines = new ArrayList<Definition>();
    private Map<String, Object> sigletons = new HashMap<String, Object>();

    

    public LeamClassPathXMLApplicationContext(String filename){
        this.readXML(filename);
        this.instanceBeans();
        this.injectObject();
    }

    /**
     * 为bean对象的属性注入值
     */
    private void injectObject() {
        for(Definition beanDefinition : beanDefines){
           Object bean = sigletons.get(beanDefinition.getId());
            if(bean!=null){
                try {
                    PropertyDescriptor[] ps = Introspector.getBeanInfo(bean.getClass()).getPropertyDescriptors();
                    for(ProsDefinition propertyDefinition : beanDefinition.getPropertys()){
                        for(PropertyDescriptor properdesc : ps){
                            if(propertyDefinition.getName().equals(properdesc.getName())){
                        Method setter = properdesc.getWriteMethod();//获取属性的setter方法 
                               if(setter!=null){
                                    Object value = sigletons.get(propertyDefinition.getRef());
                                    setter.setAccessible(true);
                                    setter.invoke(bean, value);//把引用对象注入到属性                                }
                               break;
                           }
                        }
                    }
                } catch (Exception e) {
                }
            }
        }
    }

    /**
     * 完成bean的实例化
     */
    private void instanceBeans() {
        for(Definition beanDefinition : beanDefines){
            try {
                if(beanDefinition.getClassName()!=null && !"".equals(beanDefinition.getClassName().trim()))
                   sigletons.put(beanDefinition.getId(), 
                   Class.forName(beanDefinition.getClassName()).newInstance());
            } catch (Exception e) {
               e.printStackTrace();
            }
        }       
    }

    /**
     * 读取xml配置文件
     * @param filename
     */
    private void readXML(String filename) {
           SAXReader saxReader = new SAXReader();   
            Document document=null;   
          try{
           URL xmlpath = this.getClass().getClassLoader().getResource(filename);
           document = saxReader.read(xmlpath);
           Map<String,String> nsMap = new HashMap<String,String>();
           nsMap.put("ns","http://www.springframework.org/schema/beans");//加入命名空间
             XPath xsub = document.createXPath("//ns:beans/ns:bean");//创建beans/bean查询路径
             xsub.setNamespaceURIs(nsMap);//设置命名空间
             List<Element> beans = xsub.selectNodes(document);//获取文档下所有bean节点 
             for(Element element: beans){
              String id = element.attributeValue("id");//获取id属性值
                String clazz = element.attributeValue("class"); //获取class属性值        
                Definition beanDefine = new Definition(id, clazz);
               XPath propertysub =  element.createXPath("ns:property");
               propertysub.setNamespaceURIs(nsMap);//设置命名空间
                  List<Element> propertys = propertysub.selectNodes(element);
               for(Element property : propertys){                    
                    String propertyName = property.attributeValue("name");//元素内部引用的属性也获取
                    String propertyref = property.attributeValue("ref");
                 ProsDefinition propertyDefinition = new ProsDefinition(propertyName, propertyref);
                    beanDefine.getPropertys().add(propertyDefinition);
                }
                beanDefines.add(beanDefine);
             } 
            }catch(Exception e){   
                e.printStackTrace();
            }

    }

    /**

     * 获取bean实例
     * @param beanName
     * @return
     */
    public Object getBean(String beanName){
        return this.sigletons.get(beanName);
    }

}

spring ioc核心思想

ioc的思想最核心的地方在于,资源不由使用资源的双方管理,而由不使用资源的第三方管理,这可以带来很多好处。第一,资源集中管理,实现资源的可配置和易管理。第二,降低了使用资源双方的依赖程度,也就是我们说的耦合度。

点赞
收藏
评论区
推荐文章
blmius blmius
2年前
MySQL:[Err] 1292 - Incorrect datetime value: ‘0000-00-00 00:00:00‘ for column ‘CREATE_TIME‘ at row 1
文章目录问题用navicat导入数据时,报错:原因这是因为当前的MySQL不支持datetime为0的情况。解决修改sql\mode:sql\mode:SQLMode定义了MySQL应支持的SQL语法、数据校验等,这样可以更容易地在不同的环境中使用MySQL。全局s
Karen110 Karen110
2年前
一篇文章带你了解JavaScript日期
日期对象允许您使用日期(年、月、日、小时、分钟、秒和毫秒)。一、JavaScript的日期格式一个JavaScript日期可以写为一个字符串:ThuFeb02201909:59:51GMT0800(中国标准时间)或者是一个数字:1486000791164写数字的日期,指定的毫秒数自1970年1月1日00:00:00到现在。1\.显示日期使用
Jacquelyn38 Jacquelyn38
2年前
2020年前端实用代码段,为你的工作保驾护航
有空的时候,自己总结了几个代码段,在开发中也经常使用,谢谢。1、使用解构获取json数据let jsonData  id: 1,status: "OK",data: 'a', 'b';let  id, status, data: number   jsonData;console.log(id, status, number )
皕杰报表之UUID
​在我们用皕杰报表工具设计填报报表时,如何在新增行里自动增加id呢?能新增整数排序id吗?目前可以在新增行里自动增加id,但只能用uuid函数增加UUID编码,不能新增整数排序id。uuid函数说明:获取一个UUID,可以在填报表中用来创建数据ID语法:uuid()或uuid(sep)参数说明:sep布尔值,生成的uuid中是否包含分隔符'',缺省为
Easter79 Easter79
2年前
SpringIOC官方文档解读
IoC容器本章介绍了Spring的控制反转(IoC)容器。1.1。SpringIoC容器和Bean简介本章介绍了反转控制(IoC)原则的Spring框架实现。IoC也称为依赖注入(DI)。在此过程中,对象可以通过①构造函数参数(),②工厂方法的参数③或在构造或从工厂方法返回后在对象实例上设置的属性来定义其依
Wesley13 Wesley13
2年前
Java日期时间API系列31
  时间戳是指格林威治时间1970年01月01日00时00分00秒起至现在的总毫秒数,是所有时间的基础,其他时间可以通过时间戳转换得到。Java中本来已经有相关获取时间戳的方法,Java8后增加新的类Instant等专用于处理时间戳问题。 1获取时间戳的方法和性能对比1.1获取时间戳方法Java8以前
Easter79 Easter79
2年前
Twitter的分布式自增ID算法snowflake (Java版)
概述分布式系统中,有一些需要使用全局唯一ID的场景,这种时候为了防止ID冲突可以使用36位的UUID,但是UUID有一些缺点,首先他相对比较长,另外UUID一般是无序的。有些时候我们希望能使用一种简单一些的ID,并且希望ID能够按照时间有序生成。而twitter的snowflake解决了这种需求,最初Twitter把存储系统从MySQL迁移
Wesley13 Wesley13
2年前
MySQL部分从库上面因为大量的临时表tmp_table造成慢查询
背景描述Time:20190124T00:08:14.70572408:00User@Host:@Id:Schema:sentrymetaLast_errno:0Killed:0Query_time:0.315758Lock_
为什么mysql不推荐使用雪花ID作为主键
作者:毛辰飞背景在mysql中设计表的时候,mysql官方推荐不要使用uuid或者不连续不重复的雪花id(long形且唯一),而是推荐连续自增的主键id,官方的推荐是auto_increment,那么为什么不建议采用uuid,使用uuid究
Python进阶者 Python进阶者
3个月前
Excel中这日期老是出来00:00:00,怎么用Pandas把这个去除
大家好,我是皮皮。一、前言前几天在Python白银交流群【上海新年人】问了一个Pandas数据筛选的问题。问题如下:这日期老是出来00:00:00,怎么把这个去除。二、实现过程后来【论草莓如何成为冻干莓】给了一个思路和代码如下:pd.toexcel之前把这