当前位置: 首页 > 工具软件 > Sprint.js > 使用案例 >

Sprint-Day02

连德义
2023-12-01

什么是注解驱动

注解启动时使用注解的形式替代xml配置,将繁杂的spring配置文件从工程中彻底消除掉,简化书写

注解驱动的弊端

  • 为了达成注解驱动的目的,可能会将原先很简单的书写,变的更加复杂
  • XML中配置第三方开发的资源是很方便的,但使用注解驱动无法在第三方开发的资源中进行编辑,因此会增大开发工作量
  • 注解的配置需要基于源代码

常用注解

启动注解功能

  • 启动注解扫描,加载类中配置的注解项
<context:component-scan base-package="packageName"/>

说明:

在进行包扫描时,会对配置的包及其子包中所有文件进行扫描

  • 扫描过程是以文件夹递归迭代的形式进行的

  • 扫描过程仅读取合法的java文件

  • 扫描时仅读取spring可识别的注解

  • 扫描结束后会将可识别的有效注解转化为spring对应的资源加入IoC容器

注意

  • 从加载效率上来说注解优于XML配置文件
  • 无论是注解格式还是XML配置格式,最终都是将资源加载到IoC容器中,差别仅仅是数据读取方式不同

bean的定义

使用到的注解

  • @Conponent
  • @Contrller
  • @Service
  • @Repository

使用的位置:类定义的上方

作用:设置该类为Spring管理的bean

@Component(value="beanId")
public class ClassName{}

说明:@Controller、@Service 、@Repository是@Component的衍生注解,功能同@Component

其内属性:value(默认) :定义bean的访问id

//@Component(value = "userService")//value属性指定bean的名称,同时value可省略
//@Component("userService")//@Component是bean的通用注解,不代表其它标识意义。弊端:通过该注解无法确定当前类是属于三层架构的具体那一层
//@Component//如果不指定bean的名称,那么默认为类名且首字母小写:userServiceImpl
//@Controller//作用等同于@Component,标签语义:大表当前bean是web层的bean对象
//@Service("userService")//作用等同于@Component,标签语义:大表当前bean是服务层的bean对象
@Repository("userService")//作用等同于@Component,标签语义:大表当前bean是dao层的bean对象

配置注解扫描:

 <!--开启spring的注解扫描:一切在指定目录下符合spring注解规范的资源都统统被加入ioc容器-->
 <context:component-scan base-package="com.itheima"/>

bean的作用域

名称:@Scope
类型:类注解
位置:类定义上方
作用:设置该类作为bean对应的属性
值:默认为singleton
singleton–>单例
prototype–>多例

@Scope("指定作用域")
public class ClassName{}

bean的生命周期

- 名称:@PostConstruct(指定初始化方法)、@PreDestroy(指定销毁方法)

- 类型:方法注解

- 位置:方法定义上方

- 作用:设置该类作为bean对应的生命周期方法

@PostConstruct
public void init() { 
	  System.out.println("init..."); 
}
@PreDestroy
public void destroyM(){
     System.out.println("destroyM run....");
}

bean的非引用类型属性注入

- 名称:@Value

- 类型:属性注解、方法注解

- 位置:属性定义上方(最常用),方法定义上方

- 作用:设置对应属性的值或对方法进行传参

@Value("${jdbc.username}")
private String username;

- 说明:

  • value值仅支持非引用类型数据(基本类型+String),可对作用的属性直接赋值;
  • value值支持读取properties文件中的属性值,通过类属性将properties中数据传入类中;
  • value值支持SpEL;
  • @value注解如果添加在属性上方,可以省略set方法(set方法的目的是为属性赋值);
  • 获取字段对象:Feild—>setAccessible (true)允许外部访问—》暴力反射

- 相关属性

  • value(默认):定义对应的属性值或参数值
    @Value("${jdbc.username}")//加载外部的properties资源,并赋值
    private String userName;

bean的引用类型属性注入

- 名称:@Autowired、@Qualifier

- 类型:属性注解、方法注解

- 位置:属性定义上方,方法定义上方

- 作用:设置对应属性的对象或对方法进行引用类型传参

- 前提:注入的对象,必须在IOC容器中存在

- 特点:优先根据类型装配,如果存在多个相同类型的bean,则按照属性名称装配

@Autowired(required = true)
@Qualifier("userDao")
private UserDao userDao;

//说明:
1. @Autowired
     原理:
         注入的引用类型对象(来自IOC容器)
         如果IOC容器中只有一个此类型的对象,直接注入
         如果IOC容器中有多个此类型的对象,按照IOC容器中对象的名称注入,如果名称不匹配报错
     属性: 了解,一般不写
         required: 
             true: 此对象必须注入成功,若不成功则报错. 默认值
             false: 可以注入不成功,此对象为null
2. @Qualifier
		1). 作用:在自动按照类型注入的基础之上,再按照bean的id注入.
		2). 注意:此注解不能独立使用,它必须和Autowired一起使用             

注入的原理

    @Test
    public void method02() throws Exception {
        Class<?> clazz = Class.forName("com.itheima.service.impl.UserServiceImpl");
        Object obj = clazz.newInstance();
        //获得类中所有的属性,并遍历
        Field[] fields = clazz.getDeclaredFields();
        for (Field field : fields) {
            //判断属性上是否存在Value注解
            boolean result = field.isAnnotationPresent(Value.class);
            if(result){
                //如果存在Value注解,就获得此注解对象的值,并设置给属性
                Value v = field.getAnnotation(Value.class);
                String value = v.value();
                //设置私有字段允许访问,默认被关键字private修饰,所以就没必要添加setter方法了
                field.setAccessible(true);
                //obj.url = "jdbc:mysql://localhost:3306/db1"
                field.set(obj,value);
            }
        }
        System.out.println(obj);
    }

加载properties文件

- 名称:@PropertySource

- 类型:类注解

- 位置:类定义上方

- 作用:加载properties文件中的属性值

- 前提:注解作用的类必须被bean注解修饰

@PropertySource(value = "classpath:filename.properties")
@Component//其它注解也可
public class ClassName {
    @Value("${propertiesAttributeName}")
    private String attributeName;
}

加载第三方资源

工厂模式: 框架中的类(第三方资源)是字节码,不是源码,无法使用注解注入,所以用工厂模式进行配置这个bean

- 名称:@Bean

- 类型:方法注解

- 位置:方法定义上方

- 作用:设置该方法的返回值作为spring管理的bean

- 注意事项:

1.如果使用@Bean为对作用的bean取名称,那么默认的bean的名称就是

方法名称

2.构建bean对象的方法所属的类必须实例化为一个bean被IOC容器管理,说白了就是类之上要打IOC相关的注解
@Bean
public DruidDataSource createDataSource() {    return ……;    }

- 说明:

  • 因为第三方bean无法在其源码上进行修改,使用@Bean解决第三方bean的引入问题
  • 该注解用于替代XML配置中的静态工厂与实例工厂创建bean,不区分方法是否为静态或非静态
  • @Bean所在的类必须被spring扫描加载,否则该注解无法生效

- 相关属性

  • value(默认):定义bean的访问id
    /**
     *
//     * @param userDao
     * @return
     * @Bean注解作用的方法返回的对象自动被ioc容器关系
     *      注意点:
     *          1.当前方法所属类必须被ioc容器管理;
     *          2.如果没有给bean取名字,默认就是方法名称;
     *          3.如果方法有入参,则会默认根据入参的类型取ioc容器中获取对应的bean
     *          然后作为入参,传入该方法!
     *          如果方法注入的类型的bean有多个,则推荐结合@Qualifier一块适合用
     */
    @Bean("datasource")
    public DruidDatasource myDruidDatasource(@Qualifier("userDao1") UserDao userDao,@Value("${jdbc.username}") String userName){
//    public DruidDatasource myDruidDatasource(){
        System.out.println(userDao);
        DruidDatasource datasource = new DruidDatasource();
        datasource.setJdbcUserName("root");
        datasource.setJdbcUrl("jdbc:mysql:///mysql");
        return datasource;
    }

纯注解格式

- 名称:@Configuration、@ComponentScan

- 类型:类注解

- 位置:类定义上方

- 作用:设置当前类为spring核心配置加载类

@Configuration
@ComponentScan("scanPackageName")
public class SpringConfigClassName{
}

说明:

  • 核心配合类用于替换spring核心配置文件,此类可以设置空的,不设置变量与属性

  • bean扫描工作使用注解@ComponentScan替代

AnnotationConfigApplicationContext

  • 加载纯注解格式上下文对象,需要使用AnnotationConfigApplicationContext
@Configuration//该注解同样也是@Component的衍生注解,代表的语义:配置,用于取代xml功能
@ComponentScan("com.itheima")//开启注解扫描,作用等价于:<context:component-scan base-package="com.itheima"/>
public class SpringConfig {
}

第三方bean配置与管理

- 名称:@Import

- 类型:类注解

- 位置:类定义上方

- 作用:导入第三方bean作为spring控制的资源

@Configuration
@Import(OtherClassName.class)
public class ClassName {
}

说明:

  • @Import注解在同一个类上,仅允许添加一次,如果需要导入多个,使用数组的形式进行设定

  • 在被导入的类中可以继续使用@Import导入其他资源(了解)

  • @Bean所在的类可以使用导入的形式进入spring容器,无需声明为bean

@Configuration//该注解同样也是@Component的衍生注解,代表的语义:配置,用于取代xml功能
@ComponentScan("com.itheima")//开启注解扫描,作用等价于:<context:component-scan base-package="com.itheima"/>
//@Import(PasswordService.class)
@Import({PasswordService.class,SpringConfig2.class})//批量导入ioc容器,bean的名称就是类的全限定名称
public class SpringConfig {
}
 类似资料: