Spring<01>Spring IOC&DI

SpringIoC&DI

1. Spring概述

1.1 什么是Spring(理解)

​ Spring是分层的JavaEE/SE应用全栈轻量级开源框架,以Ioc(Inverse Of Control:控制反转)和AOP(Aspect Oriented Programming:面向切面编程)为内核。

​ Spring提供了展示层Spring MVC 、持久层SpringJDBCTemplate以及业务层事务管理等众多企业级应用技术,还能整合开源世界众多第三方框架和类库,逐渐成为使用最多的JavaEE企业开应用开源框架。

1.2 Spring的发展历程(了解)

​ Rod Johnson(Spring之父)

​ 2017年9月发布了Spring的最新版本Spring5.0通用版(GA)

1.3 Spring的优势(理解)

​ 方便解耦,简化开发

​ AOP编程的支持

​ 声明式事务的支持

​ 方便程序测试

​ 方便继承各种优秀的框架

​ 降低JavaEE API的使用难度

​ Java源码是经典学习范例

1.4 Spring的体系结构(了解)

​ 此处应该有图片

2. Spring快速入门

2.1 Spring开发程序的步骤

​ ①导入spring开发的基本包坐标

​ ②编写dao接口和实现类

​ ③编写Spring核心配置文件(applicationContext.xml)

​ ④在Spring核心配置文件中配置UserDaoImpl

​ ⑤使用Spring的API获取Bean实例(UserDaoImpl的对象)

2.2 导入Spring开发的基本包坐标

<properties>
    <spring.version>5.0.5.RELEASE</spring.version>
</properties>
<!--导入Spring-context包,context依赖于core、beans、expression-->
<dependencies>
    <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-context</artifactId>
        <version>${spring.version}</version>
    </dependency>
    <!-- 需要单元测试,所以导入junit的坐标-->
    <dependency>
      <groupId>junit</groupId>
      <artifactId>junit</artifactId>
      <version>4.12</version>
    </dependency>
</dependencies>

2.3 创建UserDao和UserDaoImpl类

public interface UserDao {
	public void save();
}
public class UserDaoImpl implements UserDao {
	@Override
	public void save() {
		System.out.println("UserDao save method running....");
	}
}

2.4 创建核心配置文件并配置bean

在类路径(resources)下创建核心配置文件applicationContext.xml。

配置文件类型为spring config,名字任意,建议按照惯例写成applicationContext.xml。

在applicationContext.xml文件中配置Bean,关联UserDaoImpl类。

<?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">
    
	<bean id="userDao" class="com.itheima.dao.impl.UserDaoImpl"></bean>
</beans>

2.5 *通过Spring的API获取bean实例并调用

*在test类里面编写Junit单元测试(开发环境使用业务层代码完成dao层代码调用)

@Test
public void userDaoTest() {
    // 加载配置文件,创建Spring容器对象
    ApplicationContext ac = new ClassPathXmlApplicationContext("applicationContext.xml");
    UserDao userDao = (UserDao) ac.getBean("userDao");
    userDao.saveUser(new User());
}

控制台输出如下结果:

UserDao save method running....

2.6 *总结对比(★理解★)

​ 上面的内容就是通过Spring的Ioc思想,将UserDaoImpl的创建权由程序编写者交给了Spring容器,当然最终拿到对象并且使用对象的人肯定还是程序员,但是对象的创建者已经发生了改变。

​ 学习到这里,有同学可能会有疑问,这样做原来很简单的一句代码:

UserDao userdao = new UserDaoImpl();
userDao.saveUser(new User());

​ 竟然变成了复杂的代码+配置

<bean id="userDao" class="com.itheima.dao.impl.UserDaoImpl"></bean>
ApplicationContext ac = new ClassPathXmlApplicationContext("applicationContext.xml");
UserDao userDao = (UserDao) ac.getBean("userDao");
userDao.saveUser(new User());

​ 这样出力不讨好为了什么?这里首先我们要明白,使用Spring框架的目的是什么?

​ 1)使用Spring的IOC容器,将对象创建、对象之间的依赖关系管理(马上学)交给Spring,降低组件之间的耦合性,让我们更专注于应用逻辑。

​ 2)还可以提供诸如展示层、持久层、事务管理和兼容诸多第三方开源框架等等优势。

​ 所以,使用框架最终目的并不是为了让我们少写代码,是要简化Java开发,很直观的一点就是解耦合(大家先记住,之后会有体验),在另外一个层面更好的完成想要的功能。

​ 当然,在某些时候,也会适当减少我们的代码量(在讲完DI之后会有体现)。

3. Spring配置文件的配置

3.1 *Bean标签的基本配置 - id&class

​ *Bean标签作用:明确 交给Spring创建的对象 所属的类

​ 默认情况下,Spring容器通过反射调用上述类的无参构造创建该类对象,如果没有无参构造将创建不成功。

​ id:Bean所示类在Spring容器中的唯一标识

​ class:配置Bean实例所属类的全限定名称

3.2 Bean标签范围的配置 - scope

Bean标签有属性scope,用于表示bean所示对象的作用范围,取值如下

取值范围说明
singleton默认值,单例的。整个容器只有一个bean所示类的对象。
prototype多例的。整个Spring容器中可以有多个bean所示类的对象。
requestWeb项目中,Spring创建一个Bean所示类的对象,存入request域中
sessionWeb项目中,Spring创建一个Bean所示类的对象,存入session域中
global sessionWeb项目中,在Portlet环境。如果没有Portlet环境,该配置等同于session
  1. 当scope属性的取值为singleton时

Bean的实例化个数:1个

Bean的初始化时机:当Spring核心配置文件被加载时,实例化该Bean

Bean的生命周期:

​ 对象的创建:当应用加载,容器创建时,对象就被创建了

​ 对象的运行:只要容器在,对象就一直存在

​ 对象的销毁:当应用卸载时,容器被销毁了,对象也一起被销毁

2)当scope属性的取值为prototype时

Bean的实例化个数:多个

Bean的初始化时机:当调用getBean()方法时,实例化Bean

Bean的生命周期:

​ 对象的创建:当调用getBean()方法时,实例化Bean

​ 对象的运行:只要对象一直在使用,对象就会一直存在

​ 对象的销毁:当对象长时间不被使用时,被Java的垃圾回收器回收

3.3 Bean生命周期配置指定方法

​ init-method:指定Bean所示类初始化方法的名称

​ destroy-method:指定Bean所示类销毁方法的名称(需要在明确有容器销毁时才会被调用,具体容器实现类对象的close对象被调用时会调用该指定方法)

3.4 Bean实例化的三种方式

三种创建方式下,Spring提供的获取Bean对象的方法相同;仅仅是配置的方式以及配置方式背后的实现不同。获取bean的代码如下:

ApplicationContext ac = new ClassPathXmlApplicationContext("applicationContext.xml");
UserDao userDao = (UserDao) ac.getBean("userDao");
userDao.saveUser(new User());

1)使用Bean所示类的无参构造实例化(重点)

​ Bean实例化最常用的方式,Spring核心配置文件配置如下:

	<bean id="userDao" class="com.itheima.dao.impl.UserDaoImpl"></bean>

2)使用静态工厂方法实例化(应用场景:eg:connection的获取/解决系统遗留问题 )

​ Spring核心配置文件配置如下:

	<bean id="userDao" class="com.itheima.factory.StaticFactory" factory-method="getUserDao"/>

​ 需要编写StaticFactory工厂类,并提供static方法返回Bean所示类对象

package com.itheima.factory;
……
public class StaticFactory{
    public satic UserDao getUserDao(){
        return new UserDaoImpl();
    }
}

3)使用实例工厂方法实例化(解决系统遗留问题 )

​ 因为需要使用工厂实例(对象)创建Bean所示类的对象,所以需要为工厂创建一个Bean标签,Spring核心配置文件配置如下:

	<bean id="dynamicFactory" class="com.itheima.factory.DynamicFactory"></bean>
	<bean id="userDao" factory-bean="dynamicFactory" factory-method="getUserDao"></bean>

​ 需要编写DynamicFactory工厂类,并提供非static方法返回Bean所示类对象

package com.itheima.factory;
……
public class DynamicFactory{
    public UserDao getUserDao(){
        return new UserDaoImpl();
    }
}

​ *实例工厂方法变种(实现FactoryBean接口实例化,成MyBatis框架使用,后面详讲)

​ Spring核心配置文件配置如下:

<!-- 4.实现FactoryBean接口实例化 :实例工厂的变种,如集成MyBatis框架使用-->
<bean id="newDynamicFactory" class="com.itheima.factory.NewDynamicFactory"></bean>

​ 编写一个类,实现org.springframework.beans.factory.FactoryBean接口并重写其方法

package com.itheima.factory;

import org.springframework.beans.factory.FactoryBean;

public class NewDynamicFactory implements FactoryBean<UserDao> {
    
   @Override
   public UserDao getObject() throws Exception {
       UserDao userDao = new userDaoImpl();
       return cat3;
   }

   @Override
   public Class<?> getObjectType() {
       
       return UserDao.class;
   }
	//从FactoryBean 中可以看出默认实单例模式
	//可以修改为多例的,覆盖原来的方法
	//public boolean isSingleton() {
    //      return false;
    //}
}

3.5 Bean的手动注入(★非Spring原生提供支持★)

在web应用开发过程中,除了dao层对象,我们还需要创建service层对象,具体步骤如下:

①在Spring核心配置文件中配置UserServiceImpl、UserDaoImpl的Bean,将两者的创建权交由程序编写者交给Spring容器。

<bean id="userService" class="com.itheima.service.impl.UserServiceImpl"/>
<bean id="userDao" class="com.itheima.dao.impl.UserDaoImpl"/>

②编写UserServiceImpl、UserDaoImpl类,实现相应接口的方法。在持久层中要操作数据库等持久化内容。

public interface UserService {
    public void save() {};
}
public class UserServiceImpl implements UserService {
	@Override
	public void save() {
		//此处代码略,详见第③歩
        //这里需要有一个userDao的对象,以便调用dao层操作数据库
        //这种情况就是UserService依赖UserDao
        
	}
}
public interface UserDao {
    public void save() {};
}

public class UserDaoImpl implements UserDao{
	@Override
    public void save(){
        //这里假装正在操作数据库等持久化内容
        System.out.println("saving.....");
    }
}

③因为要在UserServiceImpl中使用UserDaoImpl对象,所以在UserServiceImpl类的save()方法内部创建UserDaoImpl对象(当然这里对象的创建者是Spring容器),然后调用该对象的save方法。(重点

public class UserServiceImpl implements UserService {
	@Override
	public void save() {
    	System.out.println("-------下面是重点代码-------");
		ApplicationContext applicationContext = new
			ClassPathXmlApplicationContext("applicationContext.xml");
		UserDao userDao = (UserDao) applicationContext.getBean("userDao");
		userDao.save();
        System.out.println("-------上面是重点代码-------");
	}
}

④测试:l可以在maven的Test模块里面添加测试代码(或者在模拟的ctroller类中使用main方法测试)

ApplicationContext apc = new ClassPathXmlApplicationContext("applicationContext.xml");
UserService userService = (UserService) apc.getBean("userService");
userService.save();

3.6 Bean依赖注入概念

在编写程序时,通过SPring的控制反转,我们把对象创建的工作交给了(反转给了)SPring容器,没有了硬编码。但是依赖关系并不会消除,Ioc只是降低了他们的依赖关系,eg:业务层仍需要调用Service层对象的方法。

既然service层、dao层的对象创建我们交给了Spring容器,那两层对象之间的关系是不是也可以交给Spring容器来管理呢?答案是肯定的。

简而言之,我们坐等框架(Spring容器)把持久层对象传入业务层,我们不需手动创建,直接使用即可。这种情况叫做依赖注入(Dependency Injection),在我们的案例里面就是:把持久层的对象注入到业务层。

依赖注入(Dependency Injection):它是 Spring 框架核心 IOC 的具体实现。

3.7 Bean依赖注入的三种方式

1)set方法注入:

​ 在UserServiceImpl中添加userDao属性和setUserDao方法

public calss UserServiceImpl implements UserService{
 	private UserDao userDao;
    public void setUserDao(UserDao userDao){
        this.userDao = userDao;
    }
    public void save(){
        userDao.save();
    }
}

​ 在Spring配置文件(容器)中为userService配置property子标签(注入属性)

<bean id="userDao" class="com.itheima.dao.impl.UserDaoImpl"></bean>
<bean id="userService" class="com.itheima.service.impl.UserServiceImpl">
	<property name="userDao" ref="userDao"></property>
</bean>

​ **中有两个UserDao,但是代表的含义不一样,第一个是指userServiceImpl类中的属性名是UserDao,第二个userDao指的是xml配置文件中id=userDao的bean。

2)set方法:p命名空间注入

​ 为了方便使用set方法进行注入,对xml编写格式进行了简化,其本质也是set方法注入。步骤如下:

​ 首先,引入P命名空间:

xmlns:p="http://www.springframwwork.org/schema/p"

​ 修改注入方式:

<bean id="userDao" class="com.itheima.dao.impl.UserDaoImpl"></bean>
<bean id="userService" class="com.itheima.service.impl.UserServiceImpl" p:userDao-ref="userDao"></bean>

3)构造方法注入

​ 在UserServiceImpl中添加有参构造,参数为userDao(为了健壮性,同时添加无参构造):

public class UserServiceImpl implements UserService{
    private UserDao userDao;
    //删除set方法,添加有参构造和无参构造
    public UserServiceImpl(UserDao userDao){
        this.userDao = userDao;
    }
    
    //因为很多场景下默认使用无参构造,为了程序的健壮性,添加无参构造
    public UserServiceImpl(){}
    
    //调用数据持久层方法操作数据库
    public void save(){
        userDao.save();
    }
}

​ 在Spring配置文件(容器)中为UserService配置constructor子标签(注入属性)

<bean id="userDao" class="com.itheima.userDao.impl.UserDaoImpl"></bean>
<bean id="userService" class="com.itheima.userServie.impl.UserServiceImpl">
    <!-- name的值是构造方法的形参名称,ref的值是要赋值给形参的值,也就是容器中的Bean:UserDao-->
	<constructor-arg name="userDao" ref="userDao"></constructor-arg>
</bean>

3.8 Bean依赖注入的数据类型

上面的操作过程中,注入的都是引用数据类型,通过ref属性关联被注入的数据。

除了引用数据类型之外,普通数据类型、集合都可以在容器中进行注入。

总而言之,可注入的类型分三种:

​ 普通数据类型(基本数据类型 + 字符串)

​ 引用数据类型(不含集合、字符串的引用数据类型)

​ 集合数据类型(集合数据类型)

下面将以set方法注入方式为例,演示基本数据类型、集合数据类型的注入。

1)基本数据类型注入:

在UserDaoImpl中添加普通数据类型的成员变量和set方法

public class UserDaoImpl implements UserDao{
 	private String company;
    private int age;
    public void setCompany(String company){
        this.company = company;
    }
    public void setAge(int age){
        this.age = age;
    }
    public void save(){
        System.out.println(company + "===" + age);
        System.out.println("userDao method running.....")
    }	    
}

​ 在Spring配置文件(容器)中,为UserDao配置property子标签(注入属性)

<bean id="userDao" class="com.ithiema.dao.impl.UserDaoImpl">
	<property name="company" value="传智播客"></property>
    <property name="age" value="18"></property>
</bean>

​ 2)集合数据类型(List)注入

​ 在UserDaoImpl中添加集合数据类型(List)的成员变量和set方法

public Class UserDaoImpl{
    private List<String> strList;
	public void setStrList(List<String> strList) {
	this.strList = strList;
        
	public void save() {
        System.out.println(strList);
        System.out.println("UserDao save method running....");
    }   
}

​ 在Spring配置文件中为userDao设置子标签(注入属性)

<bean id="userDao" class="com.itheima.dao.impl.UserDaoImpl">
	<property name="strList">
        <list>
            <value>aaa</value>
            <value>bbb</value>
            <value>ccc</value>
        </list>
    </property>
</bean>

​ 3)集合数据类型(List)注入

​ 在UserDaoImpl中添加集合数据类型(List)的成员变量和set方法

public Class UserDaoImpl{
    private List<User> userList;
	public void setUserList(List<User> userList) {
	this.userList = userList;
        
	public void save() {
        System.out.println(userList);
        System.out.println("UserDao save method running....");
    }   
}

​ 在Spring配置文件中为userDao设置子标签(注入属性)

<bean id="user1" class="com.itheima.domain.User">
    <property name="name" value="张三"></property>
    <property name="id" value="12"></property>
</bean>
<bean id="user2" class="com.itheima.domain.User">
    <property name="name" value="李四"></property>
    <property name="id" value="13"></property>
</bean>
<bean id="userDao" class="com.itheima.dao.impl.UserDaoImpl">
    <property name="userList">
        <list>
            <ref bean="user1"></ref>
            <ref bean="user2"></ref>
            <bean id="user3" class="com.itheima.domain.User">
                <property name="name" value="王五"></property>
                <property name="id" value="15"></property>
            </bean>
        </list>
    </property>
</bean>

4)集合数据类型(Map)注入

​ 在UserDaoImpl中添加集合数据类型(Map<String, User>)的成员变量和set方法

public Class UserDaoImpl{
    private Map<String, User> userMap;
	public void setUserMap(Map<String, User> userMap) {
	this.UserMap = userMap;
        
	public void save() {
        System.out.println(userMap);
        System.out.println("UserDao save method running....");
    }   
}

​ 在Spring配置文件中为userDao设置子标签(注入属性)

<bean id="user1" class="com.itheima.domain.User">
    <property name="name" value="张三"></property>
    <property name="id" value="12"></property>
</bean>
<bean id="user2" class="com.itheima.domain.User">
    <property name="name" value="李四"></property>
    <property name="id" value="13"></property>
</bean>
<bean id="userDao" class="com.itheima.dao.impl.UserDaoImpl">
    <property name="userMap">
        <map>
            <entry key="zhangsan" value-ref="user1"></entry>
            <entry key="lisi" value-ref="user2"></entry>
        </map>
    </property>
</bean>

3)集合数据类型(Properties)注入

​ 在UserDaoImpl中添加结合数据类型(Properties)的成员变量和set方法

public class UserDaoImpl implements UserDao {
    private Properties pops;
    public void setPops(Properties pops) {
   		this.pops = pops;
    }
    
	public void save() {	
		System.out.println(pops);
		System.out.println("UserDao save method running....");
	}
}

​ 在Spring配置文件中为userDao设置子标签(注入属性)

<bean id="userDao" class="com.itheima.dao.impl.UserDaoImpl">
    <property name="pops">
        <props>
            <prop key="p1">aaa</prop>
            <prop key="p2">bbb</prop>
            <prop key="p3">ccc</prop>
        </props>
    </property>
</bean>

3.9 Spring引入其他配置文件

​ 在实际开发中,Spring配置文件内容会非常多,这就导致Spring配置繁杂而且容易出问题。

​ 所以可以把Spring的配置文件按照业务需要、软件分层进行拆分,然后在主配置文件中引入其他配置文件。

<import resource="applicationContext-xxx.xml">

3.10 Spring配置文件总结

<bean>标签
    id属性:在容器中Bean实例的唯一标识,不允许重复(可以没有)
    class属性:要实例化的Bean的全限定名
    scope属性:Bean的作用范围,常用是Singleton(默认)和prototype
    <property>标签:属性注入
        name属性:属性名称(通过截取set方法获得)
        value属性:注入的普通属性值
        ref属性:注入的容器中的bean对象的值
        <list>标签
        	<value> 注入的普通属性值
        	<bean> 在list标签内部直接注入一个bean对象
        	<ref> 通过该标签的bean属性注入一个已有的bean对象
        <map>标签
        	<entry>	一对键值对,key/key-erf为键,value/value-ref为值
        <props>标签
        	<prop> key属性为key,标签内容为值
 
        	
    <constructor-arg>标签: 构造注入
    	name属性:构造方法的形参
    	ref属性:要注入的容器中的bean对象的值
    	value属性:要注入的普通属性值
<import>标签:导入其他的Spring的分文件

4 Spring常用API

4.1 ApplicationContext继承体系

​ ApplicationContext:接口,代表应用上下文(Spring容器),可以通过其实现类的对象获得Spring容器中的Bean对象。

4.2 ApplicationContext的实现类

​ 1)ClassPathXmlApplicationContext

​ 从类的根路径下加载xml配置文件,推荐使用这种(相对于第二种来说)

​ 2)FileSystemXmlApplicationContext

​ 从指定磁盘路径加载xml配置文件,配置文件可以在磁盘任意位置

​ 3)AnnotationConfigApplicationContext

​ 创建使用注解方式配置的Spring容器对象,用来解析识别注解。

4.3 getBean()方法使用

​ getBean()方法存在重载:

public Object getBean(String name) throws BeansException {
	assertBeanFactoryActive();
	return getBeanFactory().getBean(name);
}
public <T> T getBean(Class<T> requiredType) throws BeansException {
	assertBeanFactoryActive();
	return getBeanFactory().getBean(requiredType);
}	

​ 其中,当参数的数据类型是字符串时,表示根据Bean的id从容器中获得Bean实例,返回是Object,需要强转称对象时间的类型。
​ 当参数的数据类型是Class类型时,表示根据数据类型从容器中匹配Bean实例,不需要墙砖。但是当容器中相同类型的Bean有多个时,此方法会报错。

ApplicationContext applicationContext = new
	ClassPathXmlApplicationContext("applicationContext.xml");
UserService userService1 = (UserService) applicationContext.getBean("userService");
UserService userService2 = applicationContext.getBean(UserService.class);

​ getBean()方法存在重载:

public Object getBean(String name) throws BeansException {
	assertBeanFactoryActive();
	return getBeanFactory().getBean(name);
}
public <T> T getBean(Class<T> requiredType) throws BeansException {
	assertBeanFactoryActive();
	return getBeanFactory().getBean(requiredType);
}	

​ 其中,当参数的数据类型是字符串时,表示根据Bean的id从容器中获得Bean实例,返回是Object,需要强转称对象时间的类型。
​ 当参数的数据类型是Class类型时,表示根据数据类型从容器中匹配Bean实例,不需要墙砖。但是当容器中相同类型的Bean有多个时,此方法会报错。

ApplicationContext applicationContext = new
	ClassPathXmlApplicationContext("applicationContext.xml");
UserService userService1 = (UserService) applicationContext.getBean("userService");
UserService userService2 = applicationContext.getBean(UserService.class);
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值