猿创征文 | Spring框架【管理对象(IOC详解)】

举报
爱吃豆的土豆 发表于 2022/09/25 03:56:08 2022/09/25
【摘要】 目录 1,管理对象(IOC详解) 1.1 :什么是IOC 1.2:Bean创建 1.2.1:Bean相关注解 1.3:依赖注入(DI) 1.3.1:什么是DI 1.3.2:按照名称注入 1.3.3:按照类型注入 1.3.4:注入简单数据:@Value 1.3.5:properies数据注入 1.4:@Bean注入第...

目录

1,管理对象(IOC详解)

1.1 :什么是IOC

1.2:Bean创建

1.2.1:Bean相关注解

1.3:依赖注入(DI)

1.3.1:什么是DI

1.3.2:按照名称注入

1.3.3:按照类型注入

1.3.4:注入简单数据:@Value

1.3.5:properies数据注入

1.4:@Bean注入第三方类

1.4.1:按照类型

1.4.2:按照名称

1.4.3:参数类型:引入数据

1.4.4:参数类型:简单数据

1.5:Bean作用域

1.5.1:概述

1.5.2:单例

1.5.3:多例

1.5.4:常量

1.6:生命周期

1.6.1:什么是生命周期

1.6.2:生命周期详解

1.6.3:方式一:详解-初始化&销毁

1.6.4:方式二:第三方@Bean

1.6.5:生命周期函数有什么用吗?


1,管理对象(IOC详解)

  • Spring框架是企业使用最多的框架,没有之一。

  • Spring是一站式框架,也就是Spring可以整合其他框架。

    • Spring IoC:对象工厂及依赖注入

    • Spring AOP:面向切面编程技术,Spring事务管理的基础。

    • Spring Transaction management:Spring事务管理。

    • Spring Web MVC:后面单独学习。

 

 

1.1 :什么是IOC

  • IoC 是 Inversion of Control 的缩写,即“控制反转”。

  • 控制反转:将创建对象的权利,由自己(new)反转给spring。

  • 图解1:未使用IoC

  • 图解2:使用IoC  

  • IoC作用:

  • 统一管理对象

  • 解决对象之间的耦合

    • 之前使用,类之间存在耦合

  • 解决程序耦合

  • servlet类中,只用了service接口,表示web层和service层解耦。

  • service实现类中,只用了dao的接口,表示service层和dao层解耦。

好处: 可以实现解耦, 让类和类之间的耦合度降低,  将对象的创建权交给Spring管理 

1.2:Bean创建

  • 在spring 容器中管理的对象,统称为bean。例如:UserDao、UserService 等

1.2.1:Bean相关注解

注解 描述
@Component 将修饰的资源交予spring管理。 value属性:为资源命名(唯一标识)
@Controller 衍生注解,与@Component作用和属性相同。特用于修饰==表示层==的资源。
@Service 衍生注解,与@Component作用和属性相同。特用于修饰==业务逻辑层==的资源。
@Repository 衍生注解,与@Component作用和属性相同。特用于修饰==数据访问层==的资源。

1.3:依赖注入(DI)

1.3.1:什么是DI

依赖注入(Dependency Injection,DI)Spring 容器在创建被调用者的实例时,会自动将调用者需要的对象实例注入给调用者。

注解 描述 修饰位置
@Resource(name=”…”) 按照指定名称注入对象 字段、setter方法
@ Resource 按照类型注入对象 字段、setter方法
@Value 注入简单值 字段、setter方法、参数
@PropertySource 加载properties配置文件

1.3.2:按照名称注入


  
  1. public class 类名{
  2. @Resource(name="名称")
  3. private 类型 变量;
  4. }

字段注入


  
  1. @Resource(name = "studentService4")
  2. private StudentService studentService;

 setter方法注入


  
  1. private StudentService studentService;
  2. @Resource(name = "studentService4")
  3. public void setStudentService(StudentService studentService) {
  4. this.studentService = studentService;
  5. }

1.3.3:按照类型注入


  
  1. public class 类名{
  2. @Resource
  3. private 类型 变量;
  4. }

1.3.4:注入简单数据:@Value

  • 简单数据:基本数据类型、字符串等。

  • 需要:定义User对象,给User对象注入数据

@Value 可以给成员变量注入、也可以给属性注入(getter/setter)
  • 步骤

    • 步骤1:目标类,User,进行普通数据注入

    • 步骤2:配置类

    • 步骤3:测试类

步骤1:目标类,User,进行普通数据注入


  
  1. @Component
  2. public class User {
  3. @Value("jack")
  4. private String username;
  5. @Value("18")
  6. private Integer age;
  7. //....
  8. }

步骤2:配置类


  
  1. @Configuration
  2. @ComponentScan(basePackages = "com.czxy.demo05_di_value.domain")
  3. public class Demo05Configuration {
  4. }

 步骤3:测试类


  
  1. @RunWith(SpringRunner.class)
  2. @ContextConfiguration(classes = Demo05Configuration.class)
  3. public class TestDemo05 {
  4. @Resource
  5. private User user;
  6. @Test
  7. public void testDemo5() {
  8. System.out.println(user);
  9. }
  10. }

1.3.5:properies数据注入

  • 需求:读取数据库配置信息

  • 步骤:

    • 步骤1:编写demo07.properties文件

    • 步骤2:编写配置类,读取properties内容。@Value修饰setter

    • 步骤3:测试类

编写properties文件,key=value

#key=value
jdbc.driver=com.mysql.cj.jdbc.Driver
jdbc.url=jdbc:mysql://localhost:3306/db2
jdbc.username=root
jdbc.password=1234 

使用@PropertySource("classpath:properties文件")加载properties文件,使用@Value("${key}")进行注入

 

 测试类

 

1.4:@Bean注入第三方类

  • 在实际开发中,有很多第三方提供类(jar包里),需要在spring中使用。

  • Spring提供@Bean注解,整合第三方类。

注解 描述
@Bean 将第三方对象,添加到spring容器中,方法名为默认名。
@Bean(name = "") 按照指定名称,将第三方对象,添加到spring容器中。

 

1.4.1:按照类型

  • 需要:假设UserDao是第三方(不能添加注解),需要使用UserDao

  • 步骤:

    • 步骤1:模拟类

    • 步骤2:配置类

    • 步骤3:测试类

  • 模拟类

  • 配置类

     

     配置类

  • 
        
    1. @RunWith(SpringRunner.class)
    2. @ContextConfiguration(classes = Demo08Configuration.class)
    3. public class TestDemo08 {
    4. @Resource
    5. private UserDao userDao;
    6. @Test
    7. public void testDemo07() {
    8. // UserDao userDao = new UserDao();
    9. userDao.selectAll();
    10. }
    11. }

1.4.2:按照名称

  • 需求:定义一个UserDao接口,编写2个实现类A、B,分别按照名称进行注入

  • 步骤:

    1. 模拟数据类

      • 接口

      • 实现类A

      • 实现类B

    2. 配置类,创建2个实现类,并进行不同的命名

    3. 测试类,依次注入不同命名的实现类

  • 实现

    • 配置类

    • 
            
      1. @Configuration
      2. public class Demo09Configuration {
      3. @Bean(name="userDaoA")
      4. public UserDao createUserDaoA() {
      5. return new UserDaoImplA();
      6. }
      7. @Bean(name="userDaoB")
      8. public UserDao createUserDaoB() {
      9. return new UserDaoImplB();
      10. }
      11. }

      测试类

    • 
            
      1. @RunWith(SpringRunner.class)
      2. @ContextConfiguration(classes = Demo09Configuration.class)
      3. public class TestDemo09 {
      4. @Resource(name="userDaoA")
      5. private UserDao userDaoA;
      6. @Resource(name="userDaoB")
      7. private UserDao userDaoB;
      8. @Test
      9. public void testDemo07() {
      10. userDaoA.selectAll();
      11. userDaoB.selectAll();
      12. }
      13. }

1.4.3:参数类型:引入数据

  • 需求:service、dao 都是第三方

  • @Bean 修饰的方法,如果有参数,将自动注入。


  
  1. @Bean
  2. public 返回值 方法名(参数类型 参数名) { //主动注入参数对象
  3. }
  • 步骤:

    • 模拟类

      • UserDao

      • UserService

    • 配置类

      • 编写方法,createUserDao

      • 编写方法,createUserService( UserDao userDao )

    • 测试类

  • 实现

    • 模拟类:dao实现类没有注解


  
  1. public interface StudentDao {
  2. public void selectAll();
  3. }

 


  
  1. public class StudentDaoImpl implements StudentDao {
  2. @Override
  3. public void selectAll() {
  4. System.out.println("demo10 student dao ");
  5. }
  6. }

模拟类:service,没有注解  


  
  1. public interface StudentService {
  2. public void selectAll();
  3. }

  
  1. public class StudentServiceImpl implements StudentService {
  2. private StudentDao studentDao;
  3. public void setStudentDao(StudentDao studentDao) {
  4. this.studentDao = studentDao;
  5. }
  6. @Override
  7. public void selectAll() {
  8. System.out.println("demo10 student service");
  9. studentDao.selectAll();
  10. }
  11. }

 配置类


  
  1. @Configuration
  2. @ComponentScan(basePackages = {"com.czxy.demo10_bean_param"})
  3. public class Demo10Configuration {
  4. @Bean
  5. public StudentDao studentDao() {
  6. return new StudentDaoImpl();
  7. }
  8. @Bean
  9. public StudentService studentService10(StudentDao studentDao) {
  10. StudentServiceImpl studentService = new StudentServiceImpl();
  11. studentService.setStudentDao(studentDao);
  12. return studentService;
  13. }
  14. }

 测试类


  
  1. @RunWith(SpringRunner.class)
  2. @ContextConfiguration(classes = Demo10Configuration.class)
  3. public class TestDemo10 {
  4. @Resource(name = "studentService10")
  5. private StudentService studentService;
  6. @Test
  7. public void testDemo() {
  8. studentService.selectAll();
  9. }
  10. }

 

1.4.4:参数类型:简单数据

properties配置文件

配置类  

 

测试类  

 

1.5:Bean作用域

1.5.1:概述

  • bean作用域:一个对象的使用范围。

  • 通过@Scope可以设置Bean的作用域

注解 取值
@Scope singleton 默认值,单例的。整个spring容器只有一个
prototype 多例的。每获得一次创建一份
  • 需求:编写UserDao,获得对象,注入2次。

1.5.2:单例

dao,确定作用域方式


  
  1. @Repository
  2. @Scope("singleton")
  3. public class UserDao {
  4. }

配置类  

 测试类,注入2次,打印结果一样的。

 

1.5.3:多例

修改单例代码

 

1.5.4:常量

 

1.6:生命周期

1.6.1:什么是生命周期

  • 生命周期:指Spring创建Bean到销毁Bean的整个过程。

  • spring bean 完整生命周期参数

  • 在实际开发中,最常用的是bean初始化销毁

1.6.2:生命周期详解

  • 完整示意图

    实例:(⑧、⑬为XML配置内容)

  •  

    步骤1:创建后处理bean

  • 
        
    1. @Component
    2. public class MyBeanPostProcessor implements BeanPostProcessor {
    3. @Override
    4. public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
    5. if("dog".equalsIgnoreCase(beanName)) {
    6. System.out.println("5. BeanPostProcessor#before --> " + beanName);
    7. }
    8. return bean;
    9. }
    10. @Override
    11. public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
    12. if("dog".equalsIgnoreCase(beanName)) {
    13. System.out.println("9. BeanPostProcessor#after --> " + beanName);
    14. }
    15. return bean;
    16. }
    17. }

     步骤2:编写目标类Dog

  • 
        
    1. @Component
    2. //@Scope("prototype")
    3. public class Dog implements BeanNameAware, BeanFactoryAware, InitializingBean, DisposableBean {
    4. public Dog() {
    5. System.out.println("1. 初始化");
    6. }
    7. @Value("旺财")
    8. public void setName(String name) {
    9. System.out.println("2. properties --> " + name);
    10. }
    11. @Override
    12. public void setBeanName(String s) {
    13. System.out.println("3. BeanNameAware#setBeanName --> " + s);
    14. }
    15. @Override
    16. public void setBeanFactory(BeanFactory beanFactory) throws BeansException {
    17. System.out.println("4. BeanFactoryAware#beanFactory ");
    18. }
    19. @PostConstruct //初始化
    20. public void init() {
    21. System.out.println("6. 小狗 出生了");
    22. }
    23. @Override
    24. public void afterPropertiesSet() throws Exception {
    25. System.out.println("7. InitializingBean#afterPropertiesSet");
    26. }
    27. public void eat() {
    28. System.out.println("10. 正在吃...");
    29. }
    30. @PreDestroy //销毁
    31. public void DogDestroy() {
    32. System.out.println("11. 小狗 挂了");
    33. }
    34. @Override
    35. public void destroy() throws Exception {
    36. System.out.println("12. DisposableBean#destroy");
    37. }
    38. }

    步骤3:配置类

  • 
        
    1. @Configuration
    2. @ComponentScan(basePackages = {"com.czxy.demo13_lifecycle.domain","com.czxy.demo13_lifecycle.processor"})
    3. public class Demo13Configuration {
    4. }

    步骤4:测试类

  • 
        
    1. @RunWith(SpringRunner.class)
    2. @ContextConfiguration(classes = Demo13Configuration.class)
    3. public class TestDemo13 {
    4. @Resource
    5. private Dog dog;
    6. @Test
    7. public void testDemo13() {
    8. dog.eat();
    9. }
    10. }

1.6.3:方式一:详解-初始化&销毁

  • 需求:

    • 编写目标类Dog,并执行eat方法打印正在吃...

    • 在eat()前后分别执行初始化小狗 出生了、销毁小狗 挂了

  • 目标类:需要完成初始化、销毁功能的类

    • @PostConstruct 用于修饰==初始化==方法。

    • @PreDestroy 用于修饰==销毁==方法。

 

配置类:  

 测试类:

 

1.6.4:方式二:第三方@Bean

  • 需求:

    • 使用@Bean配置目标类Dog的初始化和销毁

  • 目录类(假设Dog由第三方jar提供,没有源码,不允许使用注解@Component

 

 配置类,使用@Bean注册第三方对象,通过 initMethod 和 destroyMethod 两个属性设置初始化和销毁

测试类

 

 

1.6.5:生命周期函数有什么用吗?

释放资源:


  
  1. public class 类名 {
  2. @Bean(destroyMethod="close")
  3. public DataSource datasource() {
  4. return new DruidDataSource();
  5. }
  6. }

文章来源: qianxu.blog.csdn.net,作者:爱吃豆的土豆,版权归原作者所有,如需转载,请联系作者。

原文链接:qianxu.blog.csdn.net/article/details/126726547

【版权声明】本文为华为云社区用户转载文章,如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。