Spring中Bean的作用域和Spring生命周期

从前面的文章中我们看出Spring是用来存储和读取Bean的,因此Spring中Bean是最核心的资源,所以我们将对Bean进行深入的理解。

Bean的作用域

现在有一个公共的Bean,提供给了两个用户去使用,但是在使用过程中,用户一修改了公共的Bean数据,当用户2在使用时,却发现产生了逻辑上的错误。

我们的预期结果是,公共的Bean可以在各自的类中去修改,但是不影响其他的类。

案例

公共的Bean:


@Controller
public class Users {
    @Bean
    public User getUser() {
        User user = new User();
        user.setId(1);
        user.setName("张三");
        return user;
    }
}

可以看到此时的名称是张三。

用户1在使用时,进行了修改操作:

@Controller
public class UserController {
    @Autowired
    private User user1;

    public User getUser1() {
        User user = user1;
        System.out.println("修改之前name="+ user1.getName());
        user1.setName("李四");
        return user1;
    }
}

用户2去使用公共的Bean:

@Controller
public class UserController2 {
    @Autowired
    private User user2;

    public User getUser2() {
        User user = user2;
        return user;
    }
}

然后我们此时打印用户1和用户2公共Bean的值:

public class start {
    public static void main(String[] args) {
        ApplicationContext context = new ClassPathXmlApplicationContext("spring-config.xml");
        UserController userController = context.getBean("userController", UserController.class);
        System.out.println(userController.getUser1().toString());

        UserController2 userController2 = context.getBean("userController2", UserController2.class);
        System.out.println(userController2.getUser2().toString());
    }
}

 我们发现用户2在使用Bean的时候竟然使用的是用户1修改之后的Bean。

原因分析

出现这种情况的原因就是因为Bean默认是在单例状态下工作的,也就是说所有人的使用都是同一个Bean,这是因为单例模式在很大程度上提高了性能,所以在Spring中Bean的作用域就是默认的单例模式(singleton)。

作用域定义

限定程序中变量的可见范围叫做作用域,而Bean的作用域是指在Spring整个框架中的某种行为模式,比如singleton单例作用域,就是表示Bean在整个Spring中只有一份,是全局共享的,当其他人修改了这个Bean之后,那么另一个人读取到的就是被修改过的值。

Bean作用域类型

  1. 单例作用域   singleton
  2. 原型作用域   prototype
  3. 请求作用域   request
  4. 会话作用域   session
  5. 全局作用域   application
  6. HTTPWebSocket作用域  Websocket

注意:在上面的这6中作用域中,只有前面的两个是适用于Spring普通项目,其他的4中适用于Spring MVC项目。

单例作用域

单例作用域就是默认的作用域,我们上面写的代码里面的公共的Bean的作用域就是单例作用域,所以会出现上述的问题。

原型作用域

对于该作用域下的Bean,每次进行获取Bean时都会创建新的实例。

我们可以修改上面的代码,将作用域类型修改为原型模式之后,就会发现上面的问题就解决了。

 使用@Scope("prototype")进行修改作用域

@Controller
public class Users {
    @Bean
    @Scope("prototype")
    public User getUser() {
        User user = new User();
        user.setId(1);
        user.setName("张三");
        return user;
    }
}

 问题解决。

其余4个作用域只适用于SpringMVC项目,这里不做过多赘述。

Spring执行流程

启动容器    

根据配置文件Bean进行初始化

 在加载配置文件的过程中,将会扫描com.java.domo这个包下面的Spring注解,也就是5大类注解。只有在这个包下面的类,并且使用了5大类注解的类,才会注册到Spring容器中。

注册Bean到容器中

上述类都是添加了5大类注解的,所以会将5这些类都注册到容器中。

装配Bean属性

如果Bean对象需要其他Bean对象作为属性,可以使用@Autowired或者@Resource

 Bean的执行流程

启动Spring容器——>实例化Bean(分配内存空间,从无到有)——>Bean注册到Spring中(存操作)——>将Bean装配到需要的类中(取操作)。

Bean生命周期

所谓的生命周期就是一个对象从诞生到销毁的全部过程,我们把这个过程就叫做一个对象的生命周期。
Bean的生命周期分为5大部分:

  1. 实例化Bean (为Bean分配内存空间)
  2. 设置属性   (Bean的注入和装配)
  3. Bean初始化   
  4. 使用Bean    
  5. 销毁Bean

其中第3步也就是Bean初始化里面实现各种Aware的通知方法。

  • 执行各种通知
  • 执行初始化前置方法
  • 初始化方法
  • 初始化后置方法

猜你喜欢

转载自blog.csdn.net/qq_63525426/article/details/131863933