当前位置: 首页 > news >正文

1、Spring_IOC

IOC

1.概述

  • IOC:Inversion of Control 控制反转,可以让容器负责对象的创建以及销毁操作,对象在容器中叫 bean
    在这里插入图片描述

2.回顾问题

  • 问题:写了太多与业务无关的代码

    • 耦合度非常高,写了很多和业务无关的代码
    • 不利于项目的升级迭代
  • 思考的解决方案

    • 能够直接获取 mapper 接口,而不必去关心底层的获取方式

3.bean 配置

3.1.创建 spring01 项目

  • 项目结构如下

    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-Ox468aeu-1692499957253)(picture/image-20221026180412182.png)]

  • 添加 Spring 依赖

    <dependencies>
    <!--        添加 spring 依赖--><dependency><groupId>org.springframework</groupId><artifactId>spring-context</artifactId><version>5.2.17.RELEASE</version></dependency>
    <!--        添加 junit 依赖--><dependency><groupId>junit</groupId><artifactId>junit</artifactId><version>4.12</version>
    <!--            <scope>test</scope>--></dependency></dependencies>
    

3.2.添加 Student 类

  • 创建空学生类

    public class Student {
    }
    

3.3.添加 Spring 配置文件

  • 添加方式如下

    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-5AxTNnQW-1692499957257)(picture/image-20221026180840332.png)]

  • 开始配置 javabean

    <?xml version="1.0" encoding="UTF-8"?>
    <beans xmlns="http://www.springframework.org/schema/beans"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">
    <!--    配置 Student 学生类--><bean id="student" name="stu" class="cn.sycoder.domian.Student"></bean>
    </beans>
    

4.属性介绍

属性名称说明
id给 bean 起名字(定义id 不能重复)
name给 bean 起别名
class类全限定类名

4.容器创建

4.1.ClassPathXmlApplicationContext

  • 从classPath下寻找
ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
System.out.println(context);

4.2.FileSystemXmlApplicationContext

  • 从系统文件下寻找
ApplicationContext ctx = new FileSystemXmlApplicationContext("绝对路径地址");
  • 使用 ClassPathXmlApplicationContext 获取方式会出现如下问题

    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-6kjqHuNO-1692499957258)(picture/image-20221026182206916.png)]

5.从容器中获取 bean

5.1.根据id 获取

ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");//从容器中根据 id 获取 bean 对象
Student stu = (Student)context.getBean("student");//通过别名获取 bean 对象
Student stuByName = (Student)context.getBean("stu");
  • 注意:如果id重复会有如下问题
    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-jBKxVpJX-1692499957259)(picture/image-20221026183925757.png)]

5.2.根据id和类型

ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml"); 
final Student student = context.getBean("student", Student.class);

5.3.根据类型获取bean

ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
final Student bean = context.getBean(Student.class);
  • 注意:使用类型获取的时候,一定要保证容器中只有一个 javabean 对象
    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-qno96Rak-1692499957261)(picture/image-20221026183708897.png)]

5.4.注意点

  • bean 的配置spring 是使用反射调用对象的无参构造器实现的。所以必须提供无参构造器,否则会出现如下错误

    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-xg1jELcK-1692499957262)(picture/image-20221026184148621.png)]

6.设计模式

6.1.单例模式

  • 概述:属于创建型模式,提供了创建对象的最佳方式。单例模式只能有一个单一的类

  • 懒汉式单例模式:需要用的时候,再去把对象创建出来

    public class SingleObject {//线程不安全的懒汉式单例模式private static SingleObject instance;public static SingleObject getInstance(){if(instance == null){instance =new SingleObject();}return instance;}}
    
  • 饿汉式单例模式:不管你有没有,我先创建出来

    public class SingleObjectE {//线程不安全的饿汉式单例模式private static SingleObjectE instance = new SingleObjectE();public static SingleObjectE getInstance(){return instance;}
    }
    

6.2.工厂模式

  • 概述:也是属于创建型模式,目的也是提供创建对象的最佳方式

  • 静态工厂

    public class BeanFactory {public static Student getBean() {return new Student();}public static Object getBean(String name) {if ("Student".equals(name))return new Student();else if("SingleObject".equals(name)) {return new SingleObject();}else{return new Object();}}}
    
  • 实例工厂

    public class BeanFactory {public Object getBean(){return new Student();}
    }
    

6.3.通过反射获取 xml 配置创建对象、

  • 模拟实现IOC获取对象

  • 导依赖

    <dependency><groupId>org.dom4j</groupId><artifactId>dom4j</artifactId><version>2.1.3</version>
    </dependency>
    <dependency><groupId>jaxen</groupId><artifactId>jaxen</artifactId><version>1.2.0</version>
    </dependency>
    
  • 代码实现

    public class XmlCreateBean {/*** 模拟spring 容器创建 javabean* @param configName* @param id* @return*/public static Object getBean(String configName,String id){final SAXReader saxReader = new SAXReader();try {final Document dom = saxReader.read(configName);final Element element = dom.getRootElement();final List<Element> beans = element.elements("bean");//获取每一个 beanfor (Element bean : beans) {if(id.equals(bean.attributeValue("id"))){final String aClass = bean.attributeValue("class");final Class<?> clz = Class.forName(aClass);return clz.newInstance();}}} catch (Exception e) {e.printStackTrace();}return null;}
    }
    

7. bean实例化

  • bean 交给 spring 创建,底层究竟是怎么创建的?
  • 实例化 bean 三种方式:
    • 构造器(常用)
    • 静态工厂方法
    • 实例工厂方法
    • 实现 FactoryBean(常用)

1.无参构造器实例化

  • 新建 person 类,底层是通过 clz.getDeclaredClasses() 获取构造器

    public class Person {public Person(){}}
    
  • 配置 Person bean

    <bean id="person" class="cn.sycoder.domian.Person"></bean>
    
  • 从容器中获取 bean

    @Testpublic void testConstructorInit(){final ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");Person p = (Person) context.getBean("person");}
    
  • 修改构造器,添加参数测试,提示找不到无参构造器

    在这里插入图片描述
    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-IvDyDbZO-1692517619467)(picture/image-20221027095741037.png)]

2.静态工厂方法实例化

  • 创建 clintServer 类,提供静态工厂方法

    public class ClientServer {//创建自身对象并且私有化private static ClientServer clientServer = new ClientServer();private ClientServer() {}public static ClientServer createInstance(){return clientServer;}
    }
    
  • 配置bean 的 xml

    <bean id="clientServer" class="cn.sycoder.domian.ClientServer" factory-method="createInstance"></bean>
    
  • 获取 bean

    @Testpublic void testFactoryStaticMethodInit(){final ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");final ClientServer bean = context.getBean(ClientServer.class);}
    

    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-8eLi3vVy-1692517619469)(picture/image-20221027100822350.png)]

  • 配置关系
    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-noejAEeq-1692517619470)(picture/image-20221027102210357.png)]

3.实例工厂方法实例化

  • 创建实例工厂类

    public class ClientServiceFactory {private static ClientService instance = new ClientService();private ClientServiceFactory(){}public ClientService getInstance(){return instance;}
    }public class ClientService {
    }
    
  • 配置 bean

    <!--    配置工厂--><bean id="clientFactory" class="cn.sycoder.domian.ClientServiceFactory"></bean>
    <!--    配置 clientService--><bean id="clientService" factory-bean="clientFactory" factory-method="getInstance"></bean>
    
  • 获取bean

    @Testpublic void testFactoryInstanceMethodInit(){final ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");final ClientService bean = context.getBean(ClientService.class);}
    
  • 配置关系

    [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-eumHoxKB-1692517915886)(picture/image-20221027102041002.png)]

4.FactoryBean

  • 创建员工类

    public class Employee {public void check(){System.out.println("检查是否能够拿到员工类对象");}
    }
    
  • 创建员工 factory 类实现 FactoryBean

    public class EmployeeFactory implements FactoryBean<Employee> {public Employee getObject() throws Exception {System.out.println("获取 emp 对象");return new Employee();}public Class<?> getObjectType() {return Employee.class;}public boolean isSingleton() {return false;}
    }
    
  • 配置工厂类(并没有直接配置 emp 类)

    <bean id="employee" class="cn.sycoder.domian.EmployeeFactory"></bean>
    
  • 获取 emp 对象

    @Testpublic void testFactoryBeanInit(){final ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");final Employee emp = (Employee)context.getBean("employee");emp.check();}
    
  • 实现方法说明

    • isSingleton:如果是 true 返回单例的对象

      public boolean isSingleton() {return true;}
      
    • getObject:进行对象创建的

      public Employee getObject() throws Exception {System.out.println("获取 emp 对象");return new Employee();
      }
      
http://www.lryc.cn/news/131614.html

相关文章:

  • Socks5、IP代理在爬虫开发与HTTP通信中的应用
  • 重新认识小米
  • react之react-redux的介绍、基本使用、获取状态、分发动作、数据流、reducer的分离与合并等
  • 滑块验证码-接口返回base64数据
  • 智能文件改名,一键与上上级目录名称同步,让文件整理更加便捷
  • RK3399平台开发系列讲解(内核调试篇)Valgrind使用案例
  • 07_缓存预热缓存雪崩缓存击穿缓存穿透
  • 常见前端基础面试题(HTML,CSS,JS)(三)
  • CSS(JavaEE初阶系列14)
  • 学习笔记230810--get请求的两种传参方式
  • 游戏找不到msvcr100.dll解决方法,常见的三种解决方法
  • 机器学习知识点总结:什么是GBDT(梯度提升树)
  • SpringBoot + Vue 微人事权限组管理模块 (十四)
  • Liunx系统编程:进程信号的概念及产生方式
  • 宝塔端口监听不到端口
  • 机器学习入门的概念
  • 插入排序优化——超越归并排序的超级算法
  • 面试之快速学习STL-容器适配器
  • 性能比较 - Spring Boot 应用程序中的线程池与虚拟线程 (Project Loom)
  • rust学习-打印结构体中的vec
  • FPGA: RS译码仿真过程
  • PostgreSQL 查询数据表、视图信息
  • 手撕vector容器
  • PyMuPDF`库实现PDF旋转功能
  • 微人事 登录问题完善
  • 【业务功能篇64】安装docker容器,在docker上安装mysql
  • MyBatis的基本概念和核心组件
  • sql update执行返回0,能否判断数据不存在
  • 数据分析 | 调用Optuna库实现基于TPE的贝叶斯优化 | 以随机森林回归为例
  • stm32单片机开关输入控制蜂鸣器参考代码(附PROTEUS电路图)