spring-ioc

三层结构的耦合性问

  • 存在的问题

    • 各层级接口问题,使用依赖倒置原则解决

    • 对象创建使用构造器,如果构造器发生改变,所有使用到的地方都要修改,使用工厂模式解决

    • 尽量减少导包操作,使用反射创建对象

    • 尽量减少字符串硬编码问题,使用配置文件存储对应的字符串

通过工厂模式获取对象

public class MyApplicationContext {


    public Object getBean(String beanName){
        if ("userService".equals(beanName)) {
            return new UserServiceImpl();
        } else if ("userDao".equals(beanName)) {
            return new UserDaoImpl();
        } else if ("user".equals(beanName)) {
            return new User();
        }

        return null;
    }

}

存在的问题 : if-else 分支太多 当需要获取其他对象时需要改动代码

                        import太多 

解决方案 通过反射获取对象

public class MyApplicationContext {


    /**
     *
     * @param className : 全限定类名
     * @return
     */
    public Object getBean(String className){

        try {
            Object obj = Class.forName(className).newInstance();
            return obj;
        } catch (Exception e) {
            e.printStackTrace();
        }

        return null;
    }

}

存在的问题: 存在大量的字符串硬编码 因为获取对象时需要传入类的全限定名

解决方案: 引入xml配置 通过dom4j 解析配置文件 获取对象

public class MyApplicationContext {

    private String configLocation;
    private Map<String, Object> map = new HashMap<String, Object>();

    public MyApplicationContext(String configLocation) {
        this.configLocation = configLocation;
        //1,MyApplicationContext一初始化就立马解析beans.xml
        parseXML(configLocation);
    }

    /**
     * 解析xml
     */
    private void parseXML(String configLocation) {
        //创建核心解析器
        SAXReader saxReader = new SAXReader();
        //加载beans.xml,获取对应inputStream
        InputStream inputStream = MyApplicationContext.class.getClassLoader().getResourceAsStream(configLocation);
        try {
            //读取该inputStream,获取对应xml文档对象
            Document document = saxReader.read(inputStream);
            Element rootElement = document.getRootElement();
            List<Element> beanEles = rootElement.elements("bean");
            for (Element beanEle : beanEles) {
                //获取bean对象的唯一标识
                String id = beanEle.attributeValue("id");
                //获取bean对象的全限定类名
                String className = beanEle.attributeValue("class");
                //2,根据xml配置的class属性使用反射创建对象,并根据id属性存储对象
                Object obj = Class.forName(className).getConstructor().newInstance();
                map.put(id, obj);

            }
        } catch (Exception e) {
            e.printStackTrace();
        }

    }

    /**
     * 获取对象
     * @param beanName : 对象名称
     * @return
     */
    public Object getBean(String beanName) {
        return map.get(beanName);
    }

}

在使用此工具类时 通过解析传入的xml文件 创建一个Map 作为存储对象的容器 从map中获取对象

IoC控制反转

  • 概述

    • IoC : inversion of controller , 控制反转

    • 对资源(对象)的控制权进行反转

  • 控制反转

使用Spring的IoC解决程序耦合

  • 控制反转

    • 控制

      • 原来,由程序直接控制使用资源

      • 现在,由spring容器控制和提供资源,程序等待资源被提供

    • 反转

      • 资源的控制权由原来的程序交给Spring容器

  • 好处

    • 降低资源之间的耦合度

    • 可以对资源进行合理有效地管理

Spring-IOCSpring框架的核心部分之一,它是一种设计模式,全称为Inversion of Control(控制反转)。它通过将对象的创建、依赖关系的管理和对象的生命周期交给Spring容器来实现,从而降低了组件之间的耦合度,提高了代码的可重用性和可维护性。Spring-IOC的实现主要依靠Spring容器,Spring容器是Spring框架的核心,它负责创建、管理和装配Bean对象,其中Bean是Spring框架中最基本的组件。 Spring-IOC的实现主要有两种方式:BeanFactory和ApplicationContext。其中,BeanFactory是Spring-IOC的基本实现,而ApplicationContext是BeanFactory的子接口,提供了更多高级特性。ApplicationContext是Spring框架中最常用的IOC容器,它除了提供BeanFactory的所有功能外,还提供了更多的企业级特性,例如AOP、事务管理、国际化、事件传播等。 下面是一个简单的Spring-IOC的例子,假设我们有一个UserService接口和一个UserServiceImpl实现类,我们可以通过Spring-IOC容器来创建和管理UserServiceImpl对象: 1.定义UserService接口和UserServiceImpl实现类 ```java public interface UserService { void addUser(User user); } @Service public class UserServiceImpl implements UserService { @Override public void addUser(User user) { // 添加用户的具体实现 } } ``` 2.在Spring配置文件中配置UserService实例 ```xml <bean id="userService" class="com.example.service.UserServiceImpl"/> ``` 3.在代码中获取UserService实例并使用 ```java ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml"); UserService userService = context.getBean("userService", UserService.class); User user = new User(); userService.addUser(user); ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值