一,Spring的简介和安装,深入理解IOC容器及测试(springioc容器)

  本篇文章为你整理了一,Spring的简介和安装,深入理解IOC容器及测试(springioc容器)的详细内容,包含有spring框架中ioc容器有哪几种 springioc容器 什么是spring ioc容器,优点是什么 spring中有几种类型的ioc容器 一,Spring的简介和安装,深入理解IOC容器及测试,希望能帮助你了解 一,Spring的简介和安装,深入理解IOC容器及测试。

  3)声明式事务

  官网地址:spring是一家公司-商业化运作。apache是一家基金会组织,接收社会的捐赠。

  https://spring.io/

  io表示开发者平台

  

  

  

  

  1.SpringFramework

  Spring 基础框架,可以视为 Spring 基础设施,基本上任何其他 Spring 项目都是以 Spring Framework 为基础的。

  Spring5,基于jdk1.8 。

  1.1SpringFramework的优良特性

  1非侵入式:使用 Spring Framework 开发应用程序时,Spring 对应用程序本身的结构影响非常小。对领域模型(JavaBean)可以做到零污染;对功能性组件也只需要使用几个简单的注解进行标记,完全不会破坏原有结构,反而能将组件结构进一步简化。这就使得基于 Spring Framework 开发应用程序时结构清晰、简洁优雅。

  2控制反转:IOC——Inversion of Control,翻转资源获取方向。把自己创建资源、向环境索取资源变成环境将资源准备好,我们享受资源注入。

  3面向切面编程:AOP——Aspect Oriented Programming,在不修改源代码的基础上增强代码功能。

  4容器:Spring IOC 是一个容器,因为它包含并且管理组件对象的生命周期。组件享受到了容器化的管理,替程序员屏蔽了组件创建过程中的大量细节,极大的降低了使用门槛,大幅度提高了开发效率。

  5组件化:Spring 实现了使用简单的组件配置组合成一个复杂的应用。在 Spring 中可以使用 XML 和 Java 注解组合这些对象。这使得我们可以基于一个个功能明确、边界清晰的组件有条不紊的搭建超大型复杂应用系统。

  6声明式:很多以前需要编写代码才能实现的功能,现在只需要声明需求即可由框架代为实现。

  7一站式:在 IOC 和 AOP 的基础上可以整合各种企业应用的开源框架和优秀的第三方类库。而且 Spring 旗下的项目已经覆盖了广泛领域,很多方面的功能性需求可以在 Spring Framework 的基础上全部使用 Spring 来实现。

  

  1.2 Spring Framework五大功能模块

  

  2.IOC容器的概念

  2.1 生活中的普通容器

  

  

  

  

  普通容器只能用来存储,没有更多功能。

  

  2.2 程序中的普通容器

  变量,数组,集合

  

  普通容器只能用来存储,没有更多功能。

  

  2.3 生活中的复杂容器

  

  政府管理我们的一生,生老病死都和政府有关。

  

  4. IOC容器在Spring中的实现

  Spring 的 IOC 容器就是 IOC 思想的一个落地的产品实现。IOC 容器中管理的组件也叫做 bean。在创建 bean 之前,首先需要创建 IOC 容器。Spring 提供了 IOC 容器的两种实现方式:

  ①BeanFactory

  这是 IOC 容器的基本实现,是 Spring 内部使用的接口。面向 Spring 本身,不提供给开发人员使用。

  

  ②ApplicationContex --我们自己使用

  BeanFactory 的子接口,提供了更多高级特性。面向 Spring 的使用者,几乎所有场合都使用 ApplicationContext 而不是底层的 BeanFactory。

  

  所以,以后在 Spring 环境下看到一个类或接口的名称中包含 ApplicationContext,那基本就可以断定,这个类或接口与 IOC 容器有关。

  

  ③ApplicationContext的主要实现类

  

  

  

  

  5. 基于XML管理bean

  5.1 实验01[重要]创建bean,并根据id获取

  5.1.1 目标

  由 Spring 的 IOC 容器创建类的对象,并根据bean的id属性值来获取bean对象。

  5.1.2 思路

  

  

  

  

  自定义一个类IEmpService= EmpServiceImpl,这个类在Spring的配置文件中进行配置,然后用ClassPathXMLApplicationContext这个类去读取配置文件,然后根据配置文件中配置的东西,将这个对象建立出来。

  

  5.1.3 创建maven项目,导入依赖

  完整的pom文件

  
projectxmlns="http://maven.apache.org/POM/4.0.0"

  xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"

  xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"

   modelVersion 4.0.0 /modelVersion

   groupId com.hy /groupId

   artifactId mybatis001 /artifactId

   version 0.0.1 /version

   packaging war /packaging

  

   properties

   project.build.sourceEncoding UTF-8 /project.build.sourceEncoding

   project.reporting.outputEncoding UTF-8 /project.reporting.outputEncoding

   maven.compiler.encoding UTF-8 /maven.compiler.encoding

   /properties

  

   dependencies

   !-- Spring core--

   dependency

   groupId org.springframework /groupId

   artifactId spring-context /artifactId

   version 5.2.20.RELEASE /version

   /dependency

  

   !-- https://mvnrepository.com/artifact/org.projectlombok/lombok --

   dependency

   groupId org.projectlombok /groupId

   artifactId lombok /artifactId

   version 1.18.20 /version

   /dependency

  

   !-- Spring-Test --

   dependency

   groupId org.springframework /groupId

   artifactId spring-test /artifactId

   version 5.2.20.RELEASE /version

   /dependency

   !--junit的核心jar包 --

   dependency

   groupId junit /groupId

   artifactId junit /artifactId

   version 4.12 /version

   scope test /scope

   /dependency

   /dependencies

  

   build

   plugins

   !-- 指定jdk,防止update project --

   plugin

   groupId org.apache.maven.plugins /groupId

   artifactId maven-compiler-plugin /artifactId

   configuration

   source 1.8 /source

   target 1.8 /target

   !-- 项目编码 --

   encoding UTF-8 /encoding

   /configuration

   /plugin

   /plugins

   /build

   /project

  
publicclassEmpServiceImpl implementsEmpService {

  @Override

  publicvoidreg() {

  System.out.println("用户注册");

  }

  }

  

  
 

  1)先导入spring的相关环境,其实就是导入pom文件jar包的坐标gav。

  2)创建spring的配置文件applicationContext.xml

  

  

  
?xmlversion="1.0"encoding="UTF-8"?

   beansxmlns="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"

  

   !-- 实验01 [重要] 创建bean --

   !-- bean标签,通过配置bean标签告诉IOC容器需要创建是哪个类的对象

  id属性:当前bean的唯一标识,设定了这个唯一标识后,在其他地方就可以用来引用这个bean对象

  class属性:指定这个bean要配置的类的全类名 --

   beanid="empService" /

   /beans

  
publicclassTestService {

  // ApplicationContext就是我们的IOC容器

  privateApplicationContext iocContainer= newClassPathXmlApplicationContext("applicationContext.xml");

  

  //实验01 [重要] 创建bean

  //从IOC容器中获取已配置的bean

  @Test

  publicvoidtestEmpService() {

  //通过bean的id属性的值得,获取这个bean对象

  EmpService empService= (EmpService)iocContainer.getBean("empService");

  empService.reg();

  }

  }

  
 

  注意:Spring底层默认通过反射技术调用类的无参构造方法来创建类的对象。

  5.1.6 无参构造方法的重要性,写一个有参的构造方法,测试

  
报错:如果没有无参构造方法,则会抛出下面的异常:

  

  

  

  

  所以对一个JavaBean来说,无参构造器和属性的getXxx()、setXxx()方法是必须存在的,特别是在框架中。

  

  5.1.7 用IOC容器创建对象和自己建区别

  

  

  被放入到容器当中,相当于加入了会员俱乐部

  

  在Spring环境下能够享受到的所有福利,都必须通过 IOC 容器附加到组件类上,所以随着我们在 Spring 中学习的功能越来越多,IOC 容器创建的组件类的对象就会比自己 new 的对象其功能强大的越来越多。

  

  5.2 实验02[重要]创建bean,并根据class类型来获取

  5.2.1 指定类型的 bean 唯一(只有一个)

  

  5.2.2 测试

  
importorg.junit.Test;

  importorg.springframework.context.ApplicationContext;

  importorg.springframework.context.support.ClassPathXmlApplicationContext;

  

  importcom.hy.service.EmpService;

  importcom.hy.service.impl.EmpServiceImpl;

  

  publicclassTestService2 {

  // ApplicationContext就是我们的IOC容器

  privateApplicationContext iocContainer= newClassPathXmlApplicationContext("applicationContext.xml");

  

  //实验02 [重要] 创建bean,并且根据类型获取bean对象

  //从IOC容器中根据类型获取已配置的bean对象

  @Test

  publicvoidtestEmpService() {

  // EmpService empService = iocContainer.getBean(EmpService.class); //正确

  EmpService empService= iocContainer.getBean(EmpServiceImpl.class); //正确

  empService.reg(); //用户注册

  }

  }

  

  
 

  5.2.3 问题:如果配置了两个同样类型的bean对象,则报错,根据类型获取报错。

  
?xmlversion="1.0"encoding="UTF-8"?

   beansxmlns="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"

  

   !-- 实验01 [重要] 创建bean --

   !-- bean标签,通过配置bean标签告诉IOC容器需要创建是哪个类的对象

  id属性:当前bean的唯一标识,设定了这个唯一标识后,在其他地方就可以用来引用这个bean对象

  class属性:指定这个bean要配置的类的全类名 --

   beanid="empService" /

  

   beanid="empService2" /

   /beans

  

  
importorg.junit.Test;

  importorg.springframework.context.ApplicationContext;

  importorg.springframework.context.support.ClassPathXmlApplicationContext;

  

  importcom.hy.service.EmpService;

  importcom.hy.service.impl.EmpServiceImpl;

  

  publicclassTestService2 {

  // ApplicationContext就是我们的IOC容器

  privateApplicationContext iocContainer= newClassPathXmlApplicationContext("applicationContext.xml");

  

  //实验02 [重要] 创建bean,并且根据类型获取bean对象

  //如果容器中配置了同样类型的两个bean对象,

  //从IOC容器中根据类型获取已配置的bean对象时,则报错。

  @Test

  publicvoidtestEmpService() {

  // EmpService empService = iocContainer.getBean(EmpService.class); //失败

  EmpService empService= iocContainer.getBean(EmpServiceImpl.class); //失败

  }

  }

  

  
 

  

  

  

  

  

  

  5.2.4 总结

  如果一个类实现了某个接口,根据接口类型可以获取 bean 吗?

  可以,前提是bean唯一,如下的配置就不行。

  
 

  如果一个接口有多个实现类,这些实现类都配置了 bean标签,根据接口类型可以获取 bean 吗?

  不行,因为bean不唯一

  

  根据类型来获取bean时,在满足bean唯一性的前提下,其实只是看:『对象 instanceof指定的类型』的返回结果,只要返回的是true就可以认定为和类型匹配,能够获取到。

  5.3 实验03[重要] 给bean的属性赋值:通过setXxx方法注入

  5.3.1 首先,给EmpServiceImpl添加一个属性,提供setXxx和getXxx方法

  
publicclassEmpServiceImpl implementsEmpService {

  privateString empStatus;

  

  

  publicString getEmpStatus() {

  returnempStatus;

  }

  

  

  publicvoidsetEmpStatus(String empStatus) {

  this.empStatus= empStatus;

  }

  @Override

  publicvoidreg() {

  System.out.println("用户注册");

  }

  }

  
!-- 实验03 [重要] 创建bean,同时给bean的属性赋值:通过setXxx方法注入 --

   beanid="empService3"

   !-- property标签:通过类的setXxx()方法给该类的对象设置属性值

  name属性:指定属性名(其实这个名字是setXxx和getXx方法定义的,和属性名无关)

  value属性:指定属性值

  --

   propertyname="empStatus"value="积极"/

   /bean

  
importorg.junit.Test;

  importorg.springframework.context.ApplicationContext;

  importorg.springframework.context.support.ClassPathXmlApplicationContext;

  

  importcom.hy.service.impl.EmpServiceImpl;

  

  publicclassTestService3 {

  privateApplicationContext iocContainer= newClassPathXmlApplicationContext("applicationContext.xml");

  

  //实验03 [重要]

  @Test

  publicvoidtestEmpService() {

  //通过bean的id属性的值得,获取这个bean对象

  EmpServiceImpl empService3= (EmpServiceImpl)iocContainer.getBean("empService3");

  System.out.println(empService3.getEmpStatus()); //积极

  }

  }

  
!-- 实验四 [重要]给bean的属性赋值:引用外部已声明的bean --

   bean id="empMapper"

   property name="name" value="范冰冰"/

   /bean

  

   bean id="empService"

   !-- 使用外部已经配置好的empMapper这个bean来给当前组件的happyMachine属性赋值 --

   !-- 引用另外一个bean不能再使用value属性,而要使用ref属性指定bean的id --

   !--

  如果错把ref属性写成了value属性,会抛出异常:

  Caused by: java.lang.IllegalStateException: Cannot convert value of type java.lang.String to required type com. for property empMapper : no matching editors or conversion strategy found

  意思是不能把String类型转换成我们要的EmpMapper类型

  说明我们使用value属性时,Spring只把这个属性看做一个普通的字符串,不会认为这是一个bean的id,更不会根据它去找到bean来赋值

  --

   property name="empMapper" ref=" empMapper "/

   /bean

  
!-- 内部bean PersonServiceImpl里有一个Dept属性 --

   beanid="personService"

   propertyname="dept"

   !-- 在property标签内部再配置一个bean,这就是内部bean

  内部bean可以直接用来给property指定的属性赋值

  --

   !--

  内部bean,仅仅被外部bean使用,不能让其他的bean使用,所以内部bean不需要id

  --

   bean /

   /property

   /bean

  
5.6 引入外部属性文件[重要]

   !-- 实验六 [重要]给bean的属性赋值:引入外部属性文件 --

   !-- 使用context名称空间下的property-placeholder标签引入外部属性文件(本质上就是记录外部属性文件的位置) --

   !-- location属性:指定外部属性文件的路径。classpath:表示这个路径从类路径根目录开始 --

   context:property-placeholder location="classpath:jdbc.properties"/

  

   bean id="druidDataSource"

   !-- 通过${key}的形式引用外部属性文件中的数据 --

   property name="url" value="${jdbc.url}"/

   property name="driverClassName" value="${jdbc.driver}"/

   property name="username" value="${jdbc.user}"/

   property name="password" value="${jdbc.password}"/

   /bean

  5.7 给bean的属性赋值:级联属性赋值(不重要,不怎么使用)

  使用内部bean,给级联属性赋值。

  

   !-- 实验七 给bean的属性赋值:级联属性赋值 --

   bean id="happyComponent7"

  

   !-- 使用内部bean的形式将happyMachine属性初始化 --

   property name="happyMachine"

   !-- 相当于创建了HappyMachine的空对象赋值给happyMachine属性 --

   bean /

   /property

  

   !--

  如果happyMachine属性没有被初始化,那么直接访问级联属性会抛出异常:

  Caused by: org.springframework.beans.NullValueInNestedPathException: Invalid property happyMachine of bean class [com.atguigu.ioc.component.HappyComponent]: Value of nested property happyMachine is null

  --

   property name="happyMachine.machineName" value="aaa"/

   /bean

  5.8 通过构造方法,给bean属性赋值[不重要]

   !-- 实验八 给bean的属性赋值:构造器注入 --

   bean id="happyTeam"

   !-- 调用类的构造器创建对象并同时传入参数值 --

   !-- constructor-arg标签:给构造器对应位置传入参数 --

   !-- index属性:指定当前参数在参数列表中的索引位置 --

   !-- name属性:指定当前参数的参数名 --

   !-- value属性:指定参数值 --

   constructor-arg index="0" name="teamName" value="gaysTeam"/

   constructor-arg index="1" name="memberCount" value="10" /

   constructor-arg index="2" name="memberSalary" value="555.55"/

   /bean

  5.9 给bean的集合属性赋值[不重要]

  private List String nameList; //标签不需要完全一致,一样可以赋值成功

  private Map String,String nameMap;

  

   !-- 实验十一 给bean的属性赋值:集合属性 --

   bean id="classes"

   property name="nameList"

   !-- list标签:准备一组集合类型的数据,给集合属性赋值 --

   !-- list

   value member01 /value

   value member02 /value

   value member03 /value

   /list --

  

   !-- 使用set标签也能实现相同效果,只是附带了去重功能 --

   !-- set

   value member01 /value

   value member02 /value

   value member02 /value

   /set --

  

   !-- array也同样兼容 --

   array

   value member01 /value

   value member02 /value

   value member02 /value

   /array

  

   /property

  

   property name="nameMap"

   !-- 给Map类型的属性赋值 --

   !-- map

   entry key="财务部" value="张三"/

   entry key="行政部" value="李四"/

   entry key="销售部" value="王五"/

   /map --

  

   !-- 也可以使用props标签 --

   props

   prop key="财务部" 张三2 /prop

   prop key="行政部" 李四2 /prop

   prop key="销售部" 王五2 /prop

   /props

   /property

   /bean

  5.10 自动装配 与 手动装配[了解]

  所谓自动装配就是一个组件需要其他组件时,由 IOC 容器负责找到那个需要的组件,并装配进去。

  
!-- 使用bean标签的autowire属性设置自动装配效果 --

   !-- byType表示根据类型进行装配,此时如果类型匹配的bean不止一个,那么会抛NoUniqueBeanDefinitionException --

   !-- byName表示根据bean的id进行匹配。而bean的id是根据需要装配组件的属性的属性名来确定的 --

   bean id="happyController"

  

  autowire="byName"

   !-- 手动装配:在property标签中使用ref属性明确指定要装配的bean --

   !-- property name="happyService" ref="happyService"/ --

   /bean

  
public void testExperiment12() {

  HappyController happyController = iocContainer.getBean(HappyController.class);

  

  HappyService happyService = happyController.getHappyService();

  

  System.out.println("happyService = " + happyService);

  }

  5.11 bean的声明周期以及初始化

  在无参构造方法打一个断点。

  bean标签scope配置singleton时,bean的创建是在IOC容器初始化的时候,创建的,而不是在getBean的时候创建的。

  bean标签scope配置prototype时,bean的创建实在调用getBean方法的时候创建的。

  

  IOC容器关闭之前销毁bean对象。

  

  

  

  

  

  

  

  这个接口总没有close方法在子接口中。

  

  

  

  

  6,基于注解管理bean

  和 XML 配置文件一样,注解本身并不能执行,注解本身仅仅只是做一个标记,具体的功能是框架检测到注解标记的位置,然后针对这个位置按照注解标记的功能来执行具体操作。

  本质上:所有一切的操作都是Java代码来完成的,XML和注解只是告诉框架中的Java代码如何执行。

  

  

  

  

  1)首先根据所需要的组件类型到IOC容器中查找

  能够找到唯一的bean:直接执行装配

  如果完全找不到匹配这个类型的bean:装配失败

  和所需类型匹配的bean不止一个

  没有@Qualifier注解:根据@Autowired标记位置成员变量的变量名作为bean的id进行匹配

  能够找到:执行装配

  找不到:装配失败

  使用@Qualifier注解:根据@Qualifier注解中指定的名称作为bean的id进行匹配

  能够找到:执行装配

  找不到:装配失败

  以上就是一,Spring的简介和安装,深入理解IOC容器及测试(springioc容器)的详细内容,想要了解更多 一,Spring的简介和安装,深入理解IOC容器及测试的内容,请持续关注盛行IT软件开发工作室。

郑重声明:本文由网友发布,不代表盛行IT的观点,版权归原作者所有,仅为传播更多信息之目的,如有侵权请联系,我们将第一时间修改或删除,多谢。

留言与评论(共有 条评论)
   
验证码: