【Spring】Bean的作用域和生命周期

这篇具有很好参考价值的文章主要介绍了【Spring】Bean的作用域和生命周期。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

目录

一、引入案例来探讨Bean的作用域

二、Bean的作用域

2.1、Bean的6种作用域

2.2、设置Bean的作用域

三、Spring的执行流程

 四、Bean的声明周期

1、生命周期演示


一、引入案例来探讨Bean的作用域

首先我们创建一个User类,定义一个用户信息,在定义一个Users类,使用方法注解将user存入Spring中,然后两个用户A对这个公共的Bean获取到之后,在自己的类中对Bean进行了修改,我们预期的结果是公共的Bean可以在各自的类中被修改,但是不能影响到其他类。

package com.java.demo.model;

public class User {
    private int id;
    private String name;

    public int getId() {
        return id;
    }

    public void setId(int id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    @Override
    public String toString() {
        return "User{" +
                "id=" + id +
                ", name='" + name + '\'' +
                '}';
    }

}
package com.java.demo.model;


import org.springframework.context.annotation.Bean;
import org.springframework.stereotype.Component;
/*
* 公共对象
* */

@Component
public class Users {
  @Bean("user")
  public User getUser(){
      User user = new User();
      user.setId(123);
      user.setName("李逵");
      return user;
  }
}

 A想着在自己的类中对Bean对象进行修改,不会影响到其他的类。

package com.java.demo.controller;

import com.java.demo.model.User;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;

/*
* 用户控制器
* 作者:A
* */
@Controller
public class UserController2 {

    @Autowired
    private User user;
    public void doMethod(){
        User user2 = user;
        System.out.println("User -> 修改之前 : user "+user);
        user2.setId(111);
        user2.setName("黑旋风");
        System.out.println("User -> 修改之后 : user "+user);
    }
}

 B没有做任何修改,只是在自己的类中获取了之后,打印了这个Bean对象

package com.java.demo.controller;

import com.java.demo.model.User;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;

/*
* 用户控制器
* 作者:B
* */
@Controller
public class UserController3 {
    @Autowired
    private User user;
    public void doMethod(){
        System.out.println("UserController3: user->"+user);
    }
}

【Spring】Bean的作用域和生命周期,Spring,spring,java,mysql

 可以看到A用户在自己的类中修改了Bean对象,也影响到了B获取的Bean对象的结果。这个问题的原因在于Bean对象在默认情况下是单例模式,也就是A和B用户使用的都是同一个对象。为了达到我们预期的效果,下面我们来了解一下Bean的作用域。 


二、Bean的作用域

我们在学习Java基础知识的时候,了解到的作用域是说源代码中定义的变量的可用范围。但是在Spring中Bean的作用域试着Bean在Spring整个框架中的某种行为模式,就比如我们刚刚的代码,它其中的Bean对象的作用域就是单例作用域。

2.1、Bean的6种作用域

Spring容器在初始化一个Bean的实例时,同时会指定该实例的作用域。

1️⃣singleton(单例作用域):是Spring中,Bean默认的作用域,若一个Bean的作用域是单例的,那么每个IoC容器只会有一个Bean对象,所有对这个Bean的依赖和获取这个Bean的代码,拿到的都是同一个Bean对象,这个Bean对象是全局共享的。其他人修改了这个值之后,那么另一个人读取到的就是被修改的值。单例模式的Bean不是线程安全的,可以将Bean里面的属性设置为ThreadLocal(本地线程),就会是线程安全的

需要注意的是这里的单例和我们之前说到的设计模式中的单例是不同的。设计模式中说到的单例模式指的是某个类在进程中只有唯一的一个实例;而Spring中的单例指的是在一个Spring容器中,只会缓存某个类的一个Bean对象,所有通过这个容器获取Bean的方式,拿到的都是同一个Bean对象。但是在不同的Spring容器中,每一个Spring容器都会存在某个类的唯一的一个Bean对象。也就是说这里的单例是限定在一个Spring容器中,而不是整个应用程序中。

【Spring】Bean的作用域和生命周期,Spring,spring,java,mysql

2️⃣prototype(原型作用域)也可以理解为多例作用域。若一个Bean的作用域是prototype,那么Spring容器并不会缓存创建的Bean,程序中对这个Bean的每一次获取,容器都会重新实例化一个Bean对象。这也就意味着容器不会帮我们做对象销毁的工作。

3️⃣request(请求作用域)它将Bean的使用范围限定在一个http请求中,对于每个请求,都会单独创建一个Bean,一次的请求和响应共享一个Bean。请求结束,Bean也会随之销毁,使用request作用域一般不存在线程安全问题。

4️⃣session(会话作用域):它将Bean的使用范围限定在了一次http会话中,对于每一个会话,Spring容器都会创建一个单独的Bean,若session被销毁,则Bean也随之销毁。

5️⃣application(全局作用域):在整个Web应用期间,创建一个Bean实例。适合存储全局的配置数据等。

6️⃣websocket(HTTP WebSocket 作用域):在⼀个HTTP WebSocket的⽣命周期中,定义⼀个Bean实例。WebSocket 的每次会话中,保存了⼀个 Map 结构的头信息,将⽤来包裹客户端消息头。第⼀次初始化后,直到 WebSocket 结束都是同⼀个Bean。


2.2、设置Bean的作用域

我们可以使用注解@Scope来声明Bean的作用域,@Scope注解既可以修饰方法也可以修饰类,它的声明的方式有两种。

  1. 直接设置值:@Scope("prototype")
  2. 使用枚举设置:@Scope(ConfigurableBeanFactory.SCOPE_PROTOTYPE) 

 下面我们将开始的案例设置有单例作用域修改为原型作用域。

package com.java.demo.model;


import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Scope;
import org.springframework.stereotype.Component;
/*
* 公共对象
* */

@Component
public class Users {
  @Bean("user")
  @Scope("prototype")
  public User getUser(){
      User user = new User();
      user.setId(123);
      user.setName("李逵");
      return user;
  }
}

【Spring】Bean的作用域和生命周期,Spring,spring,java,mysql


三、Spring的执行流程

Bean 执⾏流程(Spring 大体执⾏流程):启动 Spring 容器 -> 实例化 Bean(分配内存空间,从⽆到 有) -> Bean 注册到 Spring 中(存操作) -> 将 Bean 装配到需要的类中(取操作)。

【Spring】Bean的作用域和生命周期,Spring,spring,java,mysql

 

 四、Bean的声明周期

 生命周期指的是一个对象从创建到销毁的整个生命过程,所以这里的Bean的生命周期指的就是Bean从创建到使用再到销毁的过程。

1️⃣实例化Bean:给Bean分配内存空间,(相当于毛坯房

2️⃣设置属性:当前类创建Bean对象时,依赖其他的Bean对象,这个时候使用(属性注入、Setter注入,构造方法注入)的方式引入依赖的Bean对象,赋值给当前类的属性。(相当于购买装修的基本材料)

3️⃣Bean的初始化:这里相当于装修房子

  • 执行各种通知:相当于通知各个装修师傅来施工
  • 初始化的前置方法:相当于师傅到达现场之后,和业主商量装修的方案
  • 初始化方法:这里初始化的方式有两种一种是使用xml的方式,一种使用注解@PostConstruct的方式;相当于师傅开始经行装修的工作
  • 初始化的后置方法:相当于房子装修完毕之后的清理工作。

4️⃣使用Bean:相当于房子可以入住了

5️⃣销毁Bean:相当于拆或者买了房子

❓❓❓为什么设置属性在初始化之前进行?


❗❗❗通过下面的代码来理解,BeanLifeComponent类中有一个Users类型的变量。如果BeanLifeComponent类中的方法调用users对象的方法时,users对象还没有被引入(也就是说还没有引入这个依赖),那么执行时程序一定会报错。所以就需要先引入这个依赖(Bean对象)。先执行属性设置,有助于当前类创建对象,因为当前类中的方法可能使用了这个属性,属性就必须先赋值完成。

public class BeanLifeComponent implements BeanNameAware {
    @Autowired
    private Users users;
    @Override
    public void setBeanName(String s) {
        //通知方法
        System.out.println("执行了 BeanNameAware ->"+s);
    }
    @PostConstruct
    public void doPostConstruct(){
        //使用注解的初始化方式
        System.out.println("执行了@PostConstruct");
    }
}

1、生命周期演示

下面我们通过这个例子来说明

import org.springframework.beans.factory.BeanNameAware;

import javax.annotation.PostConstruct;
import javax.annotation.PreDestroy;

public class BeanLifeComponent implements BeanNameAware {
    @Override
    public void setBeanName(String s) {
        //通知方法
        System.out.println("执行了 BeanNameAware ->"+s);
    }
    @PostConstruct
    public void doPostConstruct(){
        //使用注解的初始化方式
        System.out.println("执行了@PostConstruct");
    }

    public void myInit(){
        //使用xml的初始化方式
        System.out.println("执行了myInit");
    }

    public void sayHi(){
        System.out.println("Bean");
    }
    @PreDestroy
    public void preDestroy(){
        //销毁方法
        System.out.println("执行了preDestroy方法");
    }
}
import org.springframework.context.support.ClassPathXmlApplicationContext;

public class BeanLifeTest {
    public static void main(String[] args) {
        ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("spring-config.xml");
        BeanLifeComponent component = context.getBean("myBean", BeanLifeComponent.class);
        component.sayHi();
        context.close();//这里会调用preDestroy方法进行对象销毁
    }
}

配置信息中Bean的init-method属性表示初始化方法

<?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:content="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 https://www.springframework.org/schema/context/spring-context.xsd">
    <content:component-scan base-package="com.java"></content:component-scan>
   
    <bean id="myBean" class="BeanLifeComponent" init-method="myInit"></bean>

</beans>

【Spring】Bean的作用域和生命周期,Spring,spring,java,mysql文章来源地址https://www.toymoban.com/news/detail-635026.html

到了这里,关于【Spring】Bean的作用域和生命周期的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • Spring系列4 -- Bean的作用域和生命周期

    目录 1. 案例 2. 作用域定义 2.1 Bean的6种作用域 2.2 设置作用域 3. Sring的执行流程 4. Bean的生命周期 思考: 为什么不是先进行初始化然后再进行设置属性呢?         假设现在有⼀个公共的 Bean,提供给 A ⽤户和 B ⽤户使⽤,然⽽在使⽤的途中 A ⽤户却“悄悄”地修改了公共

    2024年02月15日
    浏览(33)
  • 5、Spring之bean的作用域和生命周期

    5.1.1.1、配置bean 注意:当bean不配置scope属性时,默认是singleton(单例) 5.1.1.2、测试 由控制台日志可知,此时ioc获取到的两个bean本质上是同一个对象 5.1.2.1、配置bean 5.1.2.2、测试 由控制台日志可知,此时ioc获取到的两个bean本质上是不同的对象 如果是在WebApplicationContext环境下

    2024年02月14日
    浏览(46)
  • 【Spring学习】Bean对象的作用域和生命周期,了解了这些你就真正熟悉spring框架了.

    前言: 大家好,我是 良辰丫 ,我们已经学会了Spring的存取,今天我们将一起来学习Bean对象的作用域和生命周期.💌💌💌 🧑个人主页:良辰针不戳 📖所属专栏:javaEE进阶篇之框架学习 🍎励志语句:生活也许会让我们遍体鳞伤,但最终这些伤口会成为我们一辈子的财富。 💦期

    2024年02月07日
    浏览(63)
  • @Autowired和@Resource注解之间的关系区别,Bean的作用域和生命周期,Spring的执行流程

    目录 一. @Autowired 和 @Resource 注解 二. Bean的作用域  1. singleton(单例模式) 2. prototype(原型模式)(多例模式) 3. 请求作用域:request 4. 会话作用域:session 三. Spring 的执行流程 四. Bean 的生命周期   1. 实例化  2. 设置属性  3. Bean 初始化      3.1 执行各种各种 Aware 通知;     

    2024年02月04日
    浏览(64)
  • Bean 作用域和生命周期

    Spring 容器是用来存储和读取 Bean 的 , 因此 Bean 是 Spring 中最核心的操作资源. 编写代码过程中 , bean 对象如果有多个属性 , 创建 Getter , Setter, 构造方法 等方法 , 会产生大量冗长的代码. 那么为了使代码更加简洁 , 我们可以使用 Lombok 框架 , 只需要一行注释 , 就可以避免大量冗长

    2024年02月05日
    浏览(79)
  • Bean作用域和生命周期

    hi,今天为大家带啦Bean的作用域和生命周期的相关知识 Bean的作用域和我们之前学过的不一样,我们之前学的作用域是一个范围,而现在指的是 Bean在Spring框架中的某种行为模式,也就是一个动作. 这样干巴巴的说看我可能无法理解,我们来举个例子 创建一个公共类的一个公共对象

    2024年02月15日
    浏览(53)
  • Bean的作用域和生命周期

    目录 1.作⽤域定义 1.1Bean的6个作用域 1.singleton:单例作用域 2.prototype:多例作用域 3.request:请求作用域 4.session:会话作用域 5.application:全局作用域 6.websocket:HTTP WebSocket作用域 单例作⽤域(singleton) VS 全局作⽤域(application) 1.2设置作用域 1.直接设置值@Scope(\\\"potptype\\\") 2.用枚举设置:@Scop

    2024年02月02日
    浏览(89)
  • 【JavaEE进阶】Bean 作用域和生命周期

    注意在此例子中需要用到lombok lombok是什么? Lombok 是一个 Java 库,它通过注解的方式来简化 Java 代码的编写。它提供了一组注解,让我们可以通过在代码中添加这些注解来自动生成样板式的代码,如 getter、setter、构造函数、toString 等。 使用 Lombok 可以有效地减少冗余的样板代

    2024年02月12日
    浏览(48)
  • 【Spring】Bean的作用域与生命周期详情:请简述Spring的执行流程并分析Bean的生命周期?

     我们都知道,Spring框架为开发人员提供了很多便捷,这使得开发人员能够更加专注于应用程序的核心业务逻辑,而不需要花费大量时间和精力在技术细节上。作为一个包含众多工具方法的IoC容器,存取JavaBean是其极为重要的一个环节。本文就对Spring中的Bean的作用域和生命周

    2024年02月12日
    浏览(48)
  • Spring Bean作用域与生命周期

    目录 Bean的作用域: Bean有六大行为模式 1、singleton:单例模式(默认) 2、prototype: 原型模式(多例模式) 3、request: 请求作用域(Spring MVC) 4、session: 会话作用域(Spring MVC) 5、application: 全局作用域(Spring MVC) 6、websocket: HTTP WebSocket 作用域(Spring WebSocket) applicationContext和singleton的区别  Bea

    2024年02月02日
    浏览(42)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包