美文网首页IT@程序员猿媛
学习Spring,手撸控制反转

学习Spring,手撸控制反转

作者: 祥哥去哪里 | 来源:发表于2019-06-16 16:43 被阅读87次

    前言 :

    image.png

    往往在各路博客中浏览Spring相关的内容。总是能看到高度总结的内容

    Spring的核心就是 AOP(面向切面) 和 IOC (控制反转)

    就像我们经常能看到的一句话

    编程的核心就是,数据结构和算法

    哎。这些话当然是大佬经过多年摸索总结出来的经验,看着俏皮。但是我们这种小白怎么又能明白里面的内涵呢?

    今天就通过自己手撸一个控制反转来,来学习到底什么是IOC

    什么是控制反转?

    引用开涛老师的ioc的几个问题。总结的非常的精辟

    Ioc—Inversion of Control,即“控制反转”,不是什么技术,而是一种设计思想。在Java开发中,Ioc意味着将你设计好的对象交给容器控制,而不是传统的在你的对象内部直接控制。如何理解好Ioc呢?理解好Ioc的关键是要明确“谁控制谁,控制什么,为何是反转(有反转就应该有正转了),哪些方面反转了”,那我们来深入分析一下:

    谁控制谁,控制什么:传统Java SE程序设计,我们直接在对象内部通过new进行创建对象,是程序主动去创建依赖对象;而IoC是有专门一个容器来创建这些对象,即由Ioc容器来控制对 象的创建;谁控制谁?当然是IoC 容器控制了对象;控制什么?那就是主要控制了外部资源获取(不只是对象包括比如文件等)。

    为何是反转,哪些方面反转了:有反转就有正转,传统应用程序是由我们自己在对象中主动控制去直接获取依赖对象,也就是正转;而反转则是由容器来帮忙创建及注入依赖对象;为何是反转?因为由容器帮我们查找及注入依赖对象,对象只是被动的接受依赖对象,所以是反转;哪些方面反转了?依赖对象的获取被反转了。

    先来个正转的例子

    可能看了开涛老师的总结,还是有点似懂非懂,既然反转没明白,我么就先看看正转
    我们现在假设有一个人叫王五,他拥有回家这个动作。但是他回家比较远,那么他需要一辆车。
    所以在回家的方法实现的时候,他在方法中new了一台宝马,然后开着宝马回家了

    public class WangWu {
        //回家
        public void goHome(){
            //实例化一辆宝马车
            BMW bmw=new BMW();
            //宝马启动
            bmw.start();
            //左转
            bmw.left();
            //熄火
            bmw.stop();
        }
    }
    

    我们观察到,wangwu这里对象要实现回家的方法,就依赖宝马这个对象。这时,对象内部主动的去实例化依赖的对象。这种控制依赖对象的方式就是正转

    约定IOC

    Spring框架在易用性上面也得到了开发者的认可。可是框架是怎么做到去掉复杂的配置的呢?那就是约定!
    按照框架和开发者约定好的规矩,就能帮开发者去除很多配置的步骤。那我们现在要手撸ioc,我们也需要约定下ioc的规矩
    我们立下3个规矩

    既然是控制反转,那么我们的bean就不能由对象内部去创建,而是构造函数传入
    1. 依赖对象都以构造函数的方式传入
    还有就是所有的对象都有IOC去管理,也就是对象的全生命周期都交给IOC
    2. 所有bean的生命周期交给IOC管理
    第三就很明显了。刚刚的正转也看到了。被依赖对象是后于依赖对象实例化的。我们现在是反转,那么被依赖对象需要先被创建
    3. 被依赖的bean需要优先创建

    故事背景

    我们需要两个人,张三,李四
    他们都是有车一族。他们都开车回家。
    他们一个有辆奥迪,一个有辆宝马。
    宝马和奥迪都有启动,左转,右转,停止 四个功能

    创建故事背景中的角色

    • 人类抽象接口。很简单每个人都具有回家的方法
    /**
     * 人类接口,抽象方法 回家
     */
    public interface Human {
       void  goHome();
    }
    
    • 汽车抽象接口。每个汽车都有的功能
    /**
     * 汽车抽象接口
     */
    public interface Car {
        void start();
        void left();
        void right();
        void stop();
    }
    
    • 有车一族抽象类,有车一族代表部分人类,他们有个特别的属性。那就是车,这个车按照我们我们约定的方式,通过构造函数传入
    /**
     * 有车一族,继承人类
     */
    public abstract class HumenWithCar implements Human {
        protected  Car car;
    
        public HumenWithCar(Car car) {
            this.car = car;
        }
    
        /**
         * 由于每个人回家的路线是不同的,有车一族这个类不能给具体的实现。交由具体的某个人来实现回家方法
         */
        public abstract void goHome();
    }
    
    • 奥迪车(宝马车)代码类似,我就不放了。继承Car接口,实现接口中定义的内容
    public class Audi implements Car {
        public void start() {
            System.out.println("Audi start");
        }
    
        public void left() {
            System.out.println("Audi left");
        }
    
        public void right() {
            System.out.println("Audi right");
        }
    
        public void stop() {
            System.out.println("Audi stop");
        }
    }
    
    • 张三(李四)代码类似,我就不放了。由于父类构造函数中需要传递一辆车,那么张三的构造对象也需要接收一个Car对象交个父类。并且通过Car对象实现张三的回家方法
    public class ZhangSan extends HumenWithCar {
        public ZhangSan(Car car) {
            super(car);
        }
    
        public void goHome() {
            car.start();
            car.left();
            car.right();
            car.stop();
        }
    }
    

    构建IOC控制器

    IOC控制器,由3个部分组成:

    1. 控制器的私有属性,用来存放各种bean的容器,Map类型。
      //创建一个储存bean的容器 使用线程安全的Map(String 就是每个bean的ID,代表一个对象的键值)
        private Map<String, Object> beans = new ConcurrentHashMap<String, Object>();
    
    1. 获取bean的方法,只需传入bean的Id就返回容器中的bean
     public Object getBean(String beanId) {
            //传入键值,到容器中取对应的bean
            Object bean=beans.get(beanId);
            if (bean==null) {
                throw new RuntimeException("所需对象在容器中没有找到");
            }
            return bean;
        }
    
    1. 设置bean,将需要生成的类,beanId,依赖对象所需的参数传入
     /**
         * 将所需要的bean放入容器
         * @param clazz 所需要的对象的类
         * @param beanId 创建bean对象在容器中的id
         * @param relyBeanIds 依赖对象在容器的beanId(有可能有多个,就传多个)
         */
        public void setBean(Class<?> clazz, String beanId, String... relyBeanIds) {
            //首先我们所有的bean都通过构造方法去创建,构造方法就需要所需的参数
            //参数其实就是这个类所依赖的类的对象。那我们先去取所依赖的对象。然后传进去就好了
            //考虑到有多个依赖就会有多个构造方法
            Object[] paramValues = new Object[relyBeanIds.length];
            for (int i = 0; i < paramValues.length; i++) {
                //这里
                paramValues[i] = getBean(relyBeanIds[i]);
            }
            Object bean = null;
            //实例化我们要创建的bean。通过构造方法
            for (Constructor<?> constructor : clazz.getConstructors()) {
                //catch异常,但不处理的原因是,如果构造方法没有构造出一个bean的话。走到异常里面。则 bean对象任然是一个null。我们留在最后判断
                try {
                    bean = constructor.newInstance(paramValues);
                } catch (InstantiationException ignored) {
                } catch (IllegalAccessException ignored) {
                } catch (InvocationTargetException ignored) {
                }
            }
            //如果循环了一轮之后,容器还是没有值
            if (bean == null) {
                throw new RuntimeException("传入依赖对象时出错");
            }
            //将实例化好的对象放到容器中
            beans.put(beanId,bean);
        }
    

    使用IOC

    一切就绪就需要使用下我们的代码了
    项目呢,事先引入了junit 用于做单元测试
    首先搞个单测类


    image.png

    类里面使用一个私有属性,用来存放IOC控制器的实例
    private IocController iocController=new IocController();

    通过注解声明一个前置方法,用来初始化容器,将所需要的bean都注册进去

        @Before
        public void before(){
            //把bean依次注册到容器中
            //bean的id我就占时使用的类的名字,这个其实是不规范的,应该是一个不会重复的字符串
            iocController.setBean(BMW.class,BMW.class.getName());
            iocController.setBean(Audi.class,Audi.class.getName());
            //因为张三这个对象依赖Audi这个对象,所以第三个参数需要传入Audi对象的beanId,来控制依赖
            iocController.setBean(ZhangSan.class,ZhangSan.class.getName(),Audi.class.getName());
            //李四同理
            iocController.setBean(LiSi.class,LiSi.class.getName(),BMW.class.getName());
        }
    

    将需要的bean都注册好了,我们就可以快乐使用它了

        @Test
        public void test(){
            //在容器中取出张三对象
            Human zhangsan= (Human) iocController.getBean(ZhangSan.class.getName());
            //调用回家方法
            zhangsan.goHome();
        }
    

    执行结果


    执行回家方法的结果

    尾巴

    本次内容只能算是学习总结。内容比较简单,好在是一些思想性的东西。能为读者带来一些帮助那是更好,如果文中有解释错误的地方,希望大佬能够指出。感谢
    完整代码:以下链接
    gihub地址

    相关文章

      网友评论

        本文标题:学习Spring,手撸控制反转

        本文链接:https://www.haomeiwen.com/subject/slbwfctx.html