springioc原理和设计模式(SpringSpring核心)

控制反转总结目录1. 控制反转描述

控制反转IoC(Inversion of Control) 是一个概念,是一种思想。指将传统上由程序代码直接操控的对象调用权交给容器,通过容器来实现对象的装配和管理。控制反转就是对对象控制权的转移,从程序代码本身反转到了外部容器。通过容器实现对象的创建,属性赋值, 依赖的管理。

IoC 是一个概念,是一种思想,其实现方式多种多样。当前比较流行的实现方式是依赖注入。应用广泛。依赖:classA 类中含有 classB 的实例,在 classA 中调用 classB 的方法完成功能,即 classA对 classB 有依赖。

Ioc 的实现:

依赖注入:DI(Dependency Injection),程序代码不做定位查询,这些工作由容器自行完成

依赖注入 DI 是指程序运行过程中,若需要调用另一个对象协助时,无须在代码中创建被调用者,而是依赖于外部容器,由外部容器创建后传递给程序。

Spring 的依赖注入对调用者与被调用者几乎没有任何要求,完全支持对象之间依赖关系的管理。Spring 框架使用依赖注入(DI)实现IoC。

Spring 容器是一个超级大工厂,负责创建、管理所有的 Java 对象,这些 Java 对象被称为 Bean。Spring 容器管理者容器中 Bean 之间的依赖关系,Spring 使用“依赖注入”的方式来管理 Bean 之间的依赖关系。使用 IoC 实现对象之间的解耦和。

2. 创建Spring程序

说明:本程序中需要了解的地方有:

  • 创建实体类中要有无参构造方法和setXXX()方法
  • 在xml文件中使用set注入简单类型的方法,创建对象
  • 测试类中ApplicationContext 用于加载 Spring 的配置文件,实现类使用的是ClassPathXmlApplicationContext,在容器启动之时,xml文件中所有的对象就会被全部创建
2.1 创建Maven项目 配置文件

Maven的配置地址

配置好Maven的前提下创建Maven模块地址

pom.xml配置

添加spring的依赖,以及指定资源文件

<dependencies> <dependency> <groupId>junit</groupId> <artifactId>junit</artifactId> <version>4.11</version> <scope>test</scope> </dependency> <!--添加spring的依赖--> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-context</artifactId> <version>5.2.5.RELEASE</version> </dependency> </dependencies> <build> <resources> <resource> <directory>/src/main/java</directory> <includes> <include>**/*.xml</include> <include>**/*.properties</include> </includes> </resource> <resource> <directory>/src/test/java</directory> <includes> <include>**/*.xml</include> <include>**/*.properties</include> </includes> </resource> </resources> </build>

2.2 创建实体类
  • 使用spring容器创建对象的前提是实体类中需要有无参构造方法 setXXX()方法

public class Student { private String name; private int age; public void setName(String name) { this.name = name; } public void setAge(int age) { this.age = age; } public Student() { System.out.println("学生的无参构造方法被调用了......"); } @Override public String toString() { return "Student{" "name='" name '\'' ", age=" age '}'; } }

2.3 配置Spring的xml文件,使用Spring 容器创建对象
  • 这里就是一个Bean工厂,只要在这里配置bean标签,在容器启动的时候,对象就会被创建
  • 下面方式使用的是set注入的方式实现

<?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 stu = new Student(); 启动容器的同时,对象就被创建了 --> <bean id="stu" class="com.lcl.pojo.Student"> <property name="name" value="张三"></property> <property name="age" value="21"></property> </bean> </beans>

2.4 测试类
  • ApplicationContext 用于加载 Spring 的配置文件,在程序中充当“容器”的角色。其实现类使用的是ClassPathXmlApplicationContext
  • ApplicationContext 容器,会在容器对象初始化时,将其中的所有对象一次性全部装配好。以后代码中若要使用到这些对象,只需从内存中直接获取即可。执行效率较高。但占用内存。Spring初始化对象时要使用无参的构造方法,切记保证类中有无参构造方法。

@Test public void testStudentSpring(){ //有spring容器创建学生对象 //如果想从spring容器中取出对象,则需要先创建容器对象,并且启动 ApplicationContext ac = new ClassPathXmlApplicationContext("s01/applicationContext.xml"); Student stu = (Student) ac.getBean("stu"); System.out.println(stu); }

3. 注入的分类3.1 set注入
  • set 注入也叫设值注入,通过 setter 方法传入被调用者的实例。这种注入方式简单、直观,因而在 Spring 的依赖注入中大量使用。
3.1.1 简单类型的注入

springioc原理和设计模式(SpringSpring核心)(1)

springioc原理和设计模式(SpringSpring核心)(2)

set注入简单类型的使用

<!--创建学校实体类的对象--> <bean id="school" class="com.lcl.pojo2.School"> <property name="name" value="郑州大学"></property> <property name="address" value="郑州市"></property> </bean>

3.1.2 引用类型的注入
  • set注入引用类型的使用:参数school属于引用类型【和实体类中属性名是一致的】,所以school属性值使用的是标签的ref属性

<!--创建学生对象 完成依赖注入 --> <bean id="stu" class="com.lcl.pojo2.Student"> <property name="name" value="李华"></property> <property name="age" value="22"></property> <property name="school" ref="school" ></property> </bean>

3.2 构造方法的注入3.2.1 使用构造方法参数名称注入
  • 这里同样使用Student类和School类,唯一不一样的就是这里在实体类中不再需要setXXX()方法

<!--使用有参构造方法注入值创建对象 配置中的name需要和====构造方法====中传入的参数名称一致 --> <bean id="school" class="com.lcl.pojo3.School"> <constructor-arg name="name" value="郑州大学"></constructor-arg> <constructor-arg name="address" value="郑州市"></constructor-arg> </bean>

3.2.2 使用构造方法的参数索引下标注入
  • constructor-arg标签中的属性分别是:index,value

<!--使用构造方法参数下标的方法注入值 创建学生类的对象 --> <bean id="student" class="com.lcl.pojo3.Student"> <constructor-arg index="0" value="李华"></constructor-arg> <constructor-arg index="1" value="22"></constructor-arg> <constructor-arg index="2" ref="school"></constructor-arg> </bean>

3.2.3 使用默认顺序注入
  • 这里的顺序要和实体类中的有参构造器的顺序一致,不然会报错

<!--使用构造器默认下表顺序注入值创建对象--> <bean id="stu" class="com.lcl.pojo3.Student"> <constructor-arg value="张三"></constructor-arg> <constructor-arg value="22"></constructor-arg> <constructor-arg ref="school"></constructor-arg> </bean>

3.4 注解注入
  • 依赖注入:DI(Dependency Injection),对于 DI 使用注解,将不再需要在 Spring 配置文件中声明bean 实例。Spring 中使用注解, 需要在原有 Spring 运行环境基础上再做一些改变。需要在 Spring 配置文件中配置组件扫描器,用于在指定的基本包中扫描注解。
  • 基于注解的方式注入,创建对象和基于set注入的方式可以说成是换汤不换药
    • 药:创建对象并依赖注入
    • 汤:set注入的方式是在:xml文件中的bean标签创建 注解方式:通过注解注入值,只需要在xml文件中配置容器需要扫描的包即可【发现有相关注解就创建对象并注入值】
  • 使用注解注入的方式创建对象,在创建实体类的需要有无参构造方法
3.4.1 注解类型创建对象的注解
  • @Component:可以创建任意对象.创建的对象的默认名称是类名的驼峰命名法.也可以指定对象的名称@Component(“指定名称”).
  • @Controller:专门用来创建控制器的对象(Servlet),这种对象可以接收用户的请求,可以返回处理结果给客户端.
  • @Service:专门用来创建业务逻辑层的对象,负责向下访问数据访问层,处理完毕后的结果返回给界面层.
  • @Repository:专门用来创建数据访问层的对象,负责数据库中的增删改查所有操作.
依赖注入的注解

说明:

实际上没那么麻烦,是简单类型就是用Value,是引用类型就是用Autowired,其他的都是体会之后的了解内容

  • 简单类型(8种基本类型 String)的注入
    • @Value:用来给简单类型注入值
  • 引用类型的注入
    • A.@Autowired:使用类型注入值,从整个Bean工厂中搜索同源类型的对象进行注入.同源类型也可注入.
    • B.@Autowired@Qualifier(“名称”):使用名称注入值,从整个Bean工厂中搜索相同名称的对象进行注入.
      • 什么是同源类型
      • :
      • a.被注入的类型(Student中的school)与注入的类型是完全相同的类型
      • b.被注入的类型(Student中的school父)与注入的类型(子)是父子类
      • c.被注入的类型(Student中的school接口)与注入的类型(实现类)是接口和实现类的类型
  • 注意:在有父子类的情况下,使用按类型注入,就意味着有多个可注入的对象.此时按照名称进行二次筛选,选中与被注入对象相同名称的对象进行注入.
3.4.2 注解类型的使用
  • 创建实体类对象,并且采用注解的方式创建对象,指明简单类型和因引用类型的属性值

学校类

@Component public class School { @Value("郑州大学") private String name; @Value("郑州市") private String address; public School() { System.out.println("School的无参构造方法被调用......"); } @Override public String toString() { return "School{" "name='" name '\'' ", address='" address '\'' '}'; } }

学生类

@Component public class Student { @Value("李华") private String name; @Value("22") private int age; /* //引用类型 @Autowired private School school;*/ //引用类型按名称注入, //使用名称注入时需要子啊Qualifier注解中说明注入名 //两个注解都不能少 @Autowired @Qualifier("school") private School school; public Student() { System.out.println("Student的无参构造方法被调用了......"); } @Override public String toString() { return "Student{" "name='" name '\'' ", age=" age ", school=" school '}'; } }

xml文件中添加配置,spring容器需要扫描的包

<!--使用注解方式创建对象并注入值,此处配置参数,使spring能够扫描到对应的包--> <context:component-scan base-package="com.lcl.s02"></context:component-scan>

测试:

@Test public void testStudent(){ ApplicationContext ac = new ClassPathXmlApplicationContext("s02/applicationContext.xml"); Student student = (Student) ac.getBean("student"); System.out.println(student); }

原文 https://blog.csdn.net/weixin_44606952/article/details/126645637

,

免责声明:本文仅代表文章作者的个人观点,与本站无关。其原创性、真实性以及文中陈述文字和内容未经本站证实,对本文以及其中全部或者部分内容文字的真实性、完整性和原创性本站不作任何保证或承诺,请读者仅作参考,并自行核实相关内容。文章投诉邮箱:anhduc.ph@yahoo.com

    分享
    投诉
    首页