Spring5之IOC操作Bean管理(基于xml和注解)(三)

简介: Spring5之IOC操作Bean管理(基于xml和注解)

4、FactoryBean

Spring有两种类型的bean,一种是普通的bean,另一种是工厂bean(FactoryBean),工厂模式的意义就是为了不暴露对象创建的过程。

普通bean:在配置文件中定义bean类型就是返回类型

df5effc2ce79e4eb2d0207f36ff8a7a9_watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBA5YWo5p2R56ys5LqM5biF,size_20,color_FFFFFF,t_70,g_se,x_16.png工厂bean:在配置文件中定义的bean类型可以和返回类型不一样

98fdd113f765c52e5fa2df9cf5141746_watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBA5YWo5p2R56ys5LqM5biF,size_20,color_FFFFFF,t_70,g_se,x_16.png

需要注意的是,上面还提了一个BeanFactory,两者的区别还是要搞清楚的,BeanFactory是IOC容器的基本实现,是Spring内部的使用接口。


创建一个Mybean类,实现FactoryBean接口,重写FactoryBean中的三个方法,可以看出源码中的三个方法:


getObject():返回需要注册的对象 ,如果为单例,该实例会放到Spring容器中单实例缓存池中

getObjectType():返回对象的类型

isSingleton():判断是否是单例 ,非单例时每次创建都会返回一个新的bean

package org.springframework.beans.factory;
import org.springframework.lang.Nullable;
public interface FactoryBean<T> {
    String OBJECT_TYPE_ATTRIBUTE = "factoryBeanObjectType";
    @Nullable
    T getObject() throws Exception;
    @Nullable
    Class<?> getObjectType();
    default boolean isSingleton() {
        return true;
    }
}
package com.factorybean;
import com.Course;
import org.springframework.beans.factory.FactoryBean;
public class MyBean implements FactoryBean<Course> {
    //定义返回bean
    @Override
    public Course getObject() throws Exception {
        Course course=new Course();
        course.setCourseName("老滑头");
        return course;
    }
    @Override
    public Class<?> getObjectType() {
        return null;
    }
    @Override
    public boolean isSingleton() {
        return FactoryBean.super.isSingleton();
    }
}

5、bean的作用域

(1)在spring里面,默认的情况下,bean是单实例,当用getBean获取对象两次时,输出的引用对象地址一样

(2) 如何设置单实例还是多实例

在spring配置文件bean标签里面有属性(scope)用于设置单实例还是多实例。

scope属性值:

第一个值为默认值,singleton,表示单例对象,第二个值,prototype,表示多实例对象。

singleton和prototype的区别:


第一 singleton单实例,prototype多实例


第二 设置scope值是singleton的时候,在加载spring配置文件时就会创建单实例对象(饿汉式)。设置scope值是prototype的时候,不是在加载spring配置文件的时候创建对象,而是在调用getBean方法时才会创建多实例对象。可以看出当用getBean获取对象两次时,输出的引用对象地址不一样。

df3f09a1b24bc548e5dc83aa108f005d_watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBA5YWo5p2R56ys5LqM5biF,size_20,color_FFFFFF,t_70,g_se,x_16.png

6、bean生命周期

生命周期,即从对象的创建到对象销毁的过程

bean的生命周期(面试常问):


(1)通过构造器创建bean实例(无参构造)


(2)为bean的属性设置值和对其他bean的引用(调用set方法)


(3)把bean实例传递给bean后置处理,执行postProcessBeforeInitialization方法


(4)调用bean的初始化方法(需要进行配置)


(5)把bean实例传递给bean后置处理器,执行postProcessAfterInitialization方法


(6)bean可以使用了(对象获取到了)


(7)当容器关闭的时候,调用bean的销毁方法(需要自己配置销毁的方法)

package com.bean;
public class Orders {
    private String oName;
    //无参构造方法
    public Orders() {
        System.out.println("第一步:执行无参构造创建bean实例");
    }
    public void setoName(String oName) {
        this.oName = oName;
        System.out.println("第二步:调用set方法设置对象的属性值");
    }
    //创建执行的初始化方法
    public void initMethod() {
        System.out.println("第三步:执行初始化的方法");
    }
    //创建执行的销毁方法
    public void destroyMethod() {
        System.out.println("第五步:执行销毁的方法");
    }
}
@Test
    public void test4(){
        ClassPathXmlApplicationContext context=new ClassPathXmlApplicationContext("bean4.xml");
        Orders orders = context.getBean("orders", Orders.class);
        System.out.println("第四步:获取创建bean实例对象");
        System.out.println(orders);
        //手动让bean实例销毁
        context.close();
    }
package com.bean;
import org.springframework.beans.BeansException;
import org.springframework.beans.factory.config.BeanPostProcessor;
public class MyBeanPost implements BeanPostProcessor {
    @Override
    public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
        System.out.println("在初始化之前执行的方法");
        return bean;
    }
    @Override
    public  Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
        System.out.println("在初始化之后执行的方法");
        return bean;
    }
}
<?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="orders" class="com.bean.Orders" init-method="initMethod" destroy-method="destroyMethod">
        <property name="oName" value="手机"/>
    </bean>
    <!--配置后置处理器-->
    <bean id="myBeanPost" class="com.bean.MyBeanPost"/>
</beans>

66c4016c7ea95895f1595aac29b1ef81_watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBA5YWo5p2R56ys5LqM5biF,size_20,color_FFFFFF,t_70,g_se,x_16.png

7、xml自动装配

根据指定的装配规则(属性名称或者属性类型),Spring自动将匹配的属性值进行注入。bean便签属性autowire,配置自动装配,autowire属性常用的两个值:byName根据属性名称注入,注入值bean的id和类属性名称一样,如果不一样,属性注入失败,得出的结果为null

663e74c2ab7d411ccb21cca2e044d63b_watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBA5YWo5p2R56ys5LqM5biF,size_20,color_FFFFFF,t_70,g_se,x_16.png

e4db300d919b8e3767a437f9623d46be_watermark,type_d3F5LXplbmhlaQ,shadow_50,text_Q1NETiBA5YWo5p2R56ys5LqM5biF,size_20,color_FFFFFF,t_70,g_se,x_16.png

byType根据属性类型注入(class的类型),如果是多个属性,采取就近原则,选第一个,后面的autowire注解也会谈到。

<?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="employees" class="com.autowire.Employees" autowire="byType">
     </bean>
    <bean id="department" class="com.autowire.Department">
    </bean>
</beans>

8、外部属性文件

外部属性文件在Mybatis中经常会用到,Spring也可以通过引入外部属性文件配置数据库连接池。把外部properties属性文件引入到配置文件中,这里需要引入context名称空间:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:context="http://www.springframework.org/schema/context"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
       http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd">

六、IOC操作Bean管理(基于注解的方式

使用注解进行spring配置,注解作用在类上面,方法上面,属性上面,使用注解的目的:简化xml配置,springboot就是简化spring的配置,可以实现完全注解开发(基于注解的方式在实际应用中使用的较多)。

1、Spring针对Bean管理创建对象提供注解


(1)@Component:相当于配置文件中的,泛指各种组件,就是说当我们的类不属于各种归类的时候(不属于@Controller、@Services等的时候),我们就可以使用@Component来标注这个类

(2)@Service:service层

(3)@Controller:web层

(4)@Repository:dao层


*上面四个注解功能是一样的,都可以创建bean实例


2、基于注解方式实现对象创建


第一步 开启组件扫描

<context:component-scan base-package="com"/>

如果扫描多个包,多个包使用","隔开 ,并且扫描包的上层目录。use-default-filters="false" 表示现在不使用默认的filter,自己配置filter。

context:include-filter,设置扫描哪些内容:

<context:component-scan base-package="com.service" use-default-filters="false">
        <context:include-filter type="annotation" expression="org.springframework.stereotype.Controller"/>
    </context:component-scan>

context:exclude-filter:设置哪些内容不进行扫描context:exclude-filter:设置哪些内容不进行扫描

<context:component-scan base-package="com.service">
        <context:exclude-filter type="annotation" expression="org.springframework.stereotype.Controller"/>
    </context:component-scan>

3、基于注解方式进行属性的注入

(1)@AutoWired:根据属性类型进行自动装配


 第一步:创建service和dao对象,在service和dao类上添加创建对象注解,在注解里面value属性值可以不写,默认是类名称,首字母小写,value值和bean的id等价。

package com.dao;
import org.springframework.stereotype.Repository;
@Repository(value ="userDaoImpl")
public class UserDaoImpl implements UserDao{
    @Override
    public void add() {
        System.out.println("dao add..........");
    }
}
package com.service;
import com.dao.UserDao;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Service;
@Service
public class UserService {
    private UserDao userDao;
    @Value(value = "老王")
    private String name;
    public void add() {
        System.out.println("service add........"+name);
        userDao.add();
    }
}

第二步:在service里面注入dao对象,在service类里面添加dao类型的属性,在属性上使用注解。

(2)@Qualifier:根据属性名称进行注入。这个@Qualifier注解的使用要和上面的@AutoWired一起使用。

    @Autowired //根据类型进行注入
    @Qualifier(value = "userDaoImpl11")//根据名称进行注入

(3)@Resource:可以根据类型注入,也可以根据名称注入。根据导包来看,import javax.annotation.Resource;是 java自带的注解,不是框架里有的。

@Resource(name = "userDaoImpl11")//根据名称进行注入

(4)@Value:注入普通类型属性

 @Value(value = "老王")
 private String name;

4、完全注解开发

创建一个配置类,替代xml配置文件。@Configuration作为配置类注解,可以替代xml配置文件,

组件扫描也可以用注解@ComponentScan(basePackages = {"com"})实现。

package com.config;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;
@Configuration      //作为配置类,替代xml配置文件
@ComponentScan(basePackages = {"com"})
public class SpringConfig {
}
相关文章
|
4天前
|
设计模式 XML Java
【23种设计模式·全精解析 | 自定义Spring框架篇】Spring核心源码分析+自定义Spring的IOC功能,依赖注入功能
本文详细介绍了Spring框架的核心功能,并通过手写自定义Spring框架的方式,深入理解了Spring的IOC(控制反转)和DI(依赖注入)功能,并且学会实际运用设计模式到真实开发中。
【23种设计模式·全精解析 | 自定义Spring框架篇】Spring核心源码分析+自定义Spring的IOC功能,依赖注入功能
|
3天前
|
存储 Java 应用服务中间件
【Spring】IoC和DI,控制反转,Bean对象的获取方式
IoC,DI,控制反转容器,Bean的基本常识,类注解@Controller,获取Bean对象的常用三种方式
|
8天前
|
XML Java 数据格式
Spring容器Bean之XML配置方式
通过对以上内容的掌握,开发人员可以灵活地使用Spring的XML配置方式来管理应用程序的Bean,提高代码的模块化和可维护性。
41 6
|
24天前
|
存储 缓存 Java
Spring面试必问:手写Spring IoC 循环依赖底层源码剖析
在Spring框架中,IoC(Inversion of Control,控制反转)是一个核心概念,它允许容器管理对象的生命周期和依赖关系。然而,在实际应用中,我们可能会遇到对象间的循环依赖问题。本文将深入探讨Spring如何解决IoC中的循环依赖问题,并通过手写源码的方式,让你对其底层原理有一个全新的认识。
44 2
|
1月前
|
XML 缓存 Java
搞透 IOC、Spring IOC ,看这篇就够了!
本文详细解析了Spring框架的核心内容——IOC(控制反转)及其依赖注入(DI)的实现原理,帮助读者理解如何通过IOC实现组件解耦,提高程序的灵活性和可维护性。关注【mikechen的互联网架构】,10年+BAT架构经验倾囊相授。
|
1月前
|
安全 Java 测试技术
Java开发必读,谈谈对Spring IOC与AOP的理解
Spring的IOC和AOP机制通过依赖注入和横切关注点的分离,大大提高了代码的模块化和可维护性。IOC使得对象的创建和管理变得灵活可控,降低了对象之间的耦合度;AOP则通过动态代理机制实现了横切关注点的集中管理,减少了重复代码。理解和掌握这两个核心概念,是高效使用Spring框架的关键。希望本文对你深入理解Spring的IOC和AOP有所帮助。
44 0
|
2月前
|
Java Spring 容器
Spring IOC、AOP与事务管理底层原理及源码解析
【10月更文挑战第1天】Spring框架以其强大的控制反转(IOC)和面向切面编程(AOP)功能,成为Java企业级开发中的首选框架。本文将深入探讨Spring IOC和AOP的底层原理,并通过源码解析来揭示其实现机制。同时,我们还将探讨Spring事务管理的核心原理,并给出相应的源码示例。
149 9
|
2月前
|
存储 开发框架 Java
什么是Spring?什么是IOC?什么是DI?IOC和DI的关系? —— 零基础可无压力学习,带源码
文章详细介绍了Spring、IOC、DI的概念和关系,解释了控制反转(IOC)和依赖注入(DI)的原理,并提供了IOC的代码示例,阐述了Spring框架作为IOC容器的应用。
41 0
什么是Spring?什么是IOC?什么是DI?IOC和DI的关系? —— 零基础可无压力学习,带源码
|
2月前
|
XML Java 数据格式
Spring IOC容器的深度解析及实战应用
【10月更文挑战第14天】在软件工程中,随着系统规模的扩大,对象间的依赖关系变得越来越复杂,这导致了系统的高耦合度,增加了开发和维护的难度。为解决这一问题,Michael Mattson在1996年提出了IOC(Inversion of Control,控制反转)理论,旨在降低对象间的耦合度,提高系统的灵活性和可维护性。Spring框架正是基于这一理论,通过IOC容器实现了对象间的依赖注入和生命周期管理。
80 0
|
2月前
|
XML Java 数据格式
Spring的IOC和AOP
Spring的IOC和AOP
58 0