目录

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:按照名称注入

public class 类名{
    @Resource(name="名称")
    private 类型 变量;
}

字段注入

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

 setter方法注入

    private StudentService studentService;
    @Resource(name = "studentService4")
    public void setStudentService(StudentService studentService) {
        this.studentService = studentService;
    }

1.3.3:按照类型注入

public class 类名{
    @Resource
    private 类型 变量;
}

1.3.4:注入简单数据:@Value

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

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

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

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

    • 步骤2:配置类

    • 步骤3:测试类

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

@Component
public class User {
    @Value("jack")
    private String username;
    @Value("18")
    private Integer age;
	//....
}

步骤2:配置类

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

 步骤3:测试类

@RunWith(SpringRunner.class)
@ContextConfiguration(classes = Demo05Configuration.class)
public class TestDemo05 {
    @Resource
    private User user;
    @Test
    public void testDemo5() {
        System.out.println(user);
    }
}

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:测试类

  • 模拟类

  • 配置类

     

     配置类

  • @RunWith(SpringRunner.class)
    @ContextConfiguration(classes = Demo08Configuration.class)
    public class TestDemo08 {
        @Resource
        private UserDao userDao;
        @Test
        public void testDemo07() {
    //        UserDao userDao = new UserDao();
            userDao.selectAll();
        }
    }

1.4.2:按照名称

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

  • 步骤:

    1. 模拟数据类

      • 接口

      • 实现类A

      • 实现类B

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

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

  • 实现

    • 配置类

    • @Configuration
      public class Demo09Configuration {
          @Bean(name="userDaoA")
          public UserDao createUserDaoA() {
              return new UserDaoImplA();
          }
          @Bean(name="userDaoB")
          public UserDao createUserDaoB() {
              return new UserDaoImplB();
          }
      }

      测试类

    • @RunWith(SpringRunner.class)
      @ContextConfiguration(classes = Demo09Configuration.class)
      public class TestDemo09 {
          @Resource(name="userDaoA")
          private UserDao userDaoA;
          @Resource(name="userDaoB")
          private UserDao userDaoB;
          @Test
          public void testDemo07() {
              userDaoA.selectAll();
              userDaoB.selectAll();
          }
      }

1.4.3:参数类型:引入数据

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

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

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

    • 模拟类

      • UserDao

      • UserService

    • 配置类

      • 编写方法,createUserDao

      • 编写方法,createUserService( UserDao userDao )

    • 测试类

  • 实现

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

public interface StudentDao {
    public void selectAll();
}
public class StudentDaoImpl implements StudentDao {
    @Override
    public void selectAll() {
        System.out.println("demo10 student dao ");
    }
}

模拟类:service,没有注解  

public interface StudentService {
    public void selectAll();
}
public class StudentServiceImpl implements StudentService {
    private StudentDao studentDao;
    public void setStudentDao(StudentDao studentDao) {
        this.studentDao = studentDao;
    }
    @Override
    public void selectAll() {
        System.out.println("demo10 student service");
        studentDao.selectAll();
    }
}

 配置类

@Configuration
@ComponentScan(basePackages = {"com.czxy.demo10_bean_param"})
public class Demo10Configuration {
    @Bean
    public StudentDao studentDao() {
        return new StudentDaoImpl();
    }
    @Bean
    public StudentService studentService10(StudentDao studentDao) {
        StudentServiceImpl studentService = new StudentServiceImpl();
        studentService.setStudentDao(studentDao);
        return studentService;
    }
}

 测试类

@RunWith(SpringRunner.class)
@ContextConfiguration(classes = Demo10Configuration.class)
public class TestDemo10 {
    @Resource(name = "studentService10")
    private StudentService studentService;
    @Test
    public void testDemo() {
        studentService.selectAll();
    }
}

1.4.4:参数类型:简单数据

properties配置文件

配置类  

 

测试类  

1.5:Bean作用域

1.5.1:概述

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

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

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

1.5.2:单例

dao,确定作用域方式

@Repository
@Scope("singleton")
public class UserDao {
}

配置类  

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

1.5.3:多例

修改单例代码

1.5.4:常量

1.6:生命周期

1.6.1:什么是生命周期

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

  • spring bean 完整生命周期参数

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

1.6.2:生命周期详解

  • 完整示意图

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

  •  

    步骤1:创建后处理bean

  • @Component
    public class MyBeanPostProcessor implements BeanPostProcessor {
        @Override
        public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
            if("dog".equalsIgnoreCase(beanName)) {
                System.out.println("5. BeanPostProcessor#before --> " + beanName);
            }
            return bean;
        }
        @Override
        public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
            if("dog".equalsIgnoreCase(beanName)) {
                System.out.println("9. BeanPostProcessor#after --> " + beanName);
            }
            return bean;
        }
    }
    

     步骤2:编写目标类Dog

  • @Component
    //@Scope("prototype")
    public class Dog implements BeanNameAware, BeanFactoryAware, InitializingBean, DisposableBean {
        public Dog() {
            System.out.println("1. 初始化");
        }
        @Value("旺财")
        public void setName(String name) {
            System.out.println("2. properties --> " + name);
        }
        @Override
        public void setBeanName(String s) {
            System.out.println("3. BeanNameAware#setBeanName --> " + s);
        }
        @Override
        public void setBeanFactory(BeanFactory beanFactory) throws BeansException {
            System.out.println("4. BeanFactoryAware#beanFactory ");
        }
        @PostConstruct  //初始化
        public void init() {
            System.out.println("6. 小狗 出生了");
        }
        @Override
        public void afterPropertiesSet() throws Exception {
            System.out.println("7. InitializingBean#afterPropertiesSet");
        }
        public void eat() {
            System.out.println("10. 正在吃...");
        }
        @PreDestroy     //销毁
        public void DogDestroy() {
            System.out.println("11. 小狗 挂了");
        }
        @Override
        public void destroy() throws Exception {
            System.out.println("12. DisposableBean#destroy");
        }
    }

    步骤3:配置类

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

    步骤4:测试类

  • @RunWith(SpringRunner.class)
    @ContextConfiguration(classes = Demo13Configuration.class)
    public class TestDemo13 {
        @Resource
        private Dog dog;
        @Test
        public void testDemo13() {
            dog.eat();
        }
    }

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

  • 需求:

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

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

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

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

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

 

配置类:  

 测试类:

1.6.4:方式二:第三方@Bean

  • 需求:

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

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

 

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

测试类

 

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

释放资源:

public class 类名 {
    @Bean(destroyMethod="close")
    public DataSource datasource() {
        return new DruidDataSource();
    }
}
声明:本站所有文章,如无特殊说明或标注,均为本站原创发布。任何个人或组织,在未征得本站同意时,禁止复制、盗用、采集、发布本站内容到任何网站、书籍等各类媒体平台。如若本站内容侵犯了原著者的合法权益,可联系我们进行处理。