【Spring篇】Spring注解式开发

本文根据哔哩哔哩课程内容结合自己自学所得,用于自己复习,如有错误欢迎指正;


我在想用一句话激励我自己努力学习,却想不出来什么惊为天人、精妙绝伦的句子,脑子里全是上课老师想说却没想起的四个字  “ 唯手熟尔 ”,是啊!不过唯手熟尔!


目录

Spring注解式开发

自定义注解

自定义注解的使用

通过反射机制读取类上的注解

需求,给你一个包名,包中有很多对象,然后将该包下实现了@component注解的类存到一个map集合中,其中注解ID为key,实例化后的对象为value值;

Spring声明Bean注解

Spring注解使用

选择实例化Bean

负责注入的注解

@value注解

@autowired注解

使用@autowired注解的方式解决注入非简单类型的数据,默认是根据类型装配的;注意点有下面两点:

如何解决上面的问题

@Resource注解

@Resource VS @autowired

spring.xml也变成注解类


Spring注解式开发

首先回顾注解相关知识点,

  1. 什么是注解?

  2. 注解的定义格式?

  3. 注解属性值的定义?

  4. 如何使用注解?

自定义注解

自定义注解的使用

通过反射机制读取类上的注解

需求,给你一个包名,包中有很多对象,然后将该包下实现了@component注解的类存到一个map集合中,其中注解ID为key,实例化后的对象为value值;

Spring声明Bean注解

这里插入一个知识点,其实注解就是元数据,第一次接触元数据是在JDBC中,所谓元注解就是修饰数据的数据,这里的注解和xml配置信息也是元注解,他们都是为了修饰数据;

在Spring中声明容器管理的Bean的注解有四个,

  1. @component

  2. @service

  3. @controller

  4. @repository

    其中只有component注解为原始注解,但是为了提高可读性,又给他起了下面三个别名;源码如下:

     @Target(value = {ElementType.TYPE})//target是为了指定注解能够出现的地方;
     @Retention(value = RetentionPolicy.RUNTIME)
     public @interface Component {
         String value();
     }
     ​
     @Target({ElementType.TYPE})
     @Retention(RetentionPolicy.RUNTIME)
     @Documented
     @Component
     public @interface Controller {
         @AliasFor(
             annotation = Component.class
         )
         String value() default "";
     }
     ​
     ​
     @Target({ElementType.TYPE})
     @Retention(RetentionPolicy.RUNTIME)
     @Documented
     @Component
     public @interface Service {
         @AliasFor(
             annotation = Component.class
         )
         String value() default "";
     }
     ​
     @Target({ElementType.TYPE})
     @Retention(RetentionPolicy.RUNTIME)
     @Documented
     @Component
     public @interface Repository {
         @AliasFor(
             annotation = Component.class
         )
         String value() default "";
     }

Spring注解使用

使用注解的前期步骤:

  1. 加入aop依赖(引入Spring-context即可,它里面包含了Spring-aop)

  2. 在配置文件中添加context命名空间

     <?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:context="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 http://www.springframework.org/schema/context/spring-context.xsd">
     ​
     </beans>
  3. 在配置文件中指定扫描的包

     <context:component-scan base-package="com.powernode.spring6.bean"/>
     //如果出现多个包的情况,可以声明他们的共同父类,或者通过逗号的方式,同时声明多个包
  4. 在Bean类上使用注解

     在Bean类上使用注解的几个小细节;
     1,在注解的使用中,如果添加的是value属性,其中value可以省略,只需要写对应的值即可;
     2,如果我们在使用注解时,不指定value属性的值时,Spring会给我自动创建一个属性值(采用默认方式),即Bean类的名字首字母小写

选择实例化Bean

有时候我们有这样的需求,将某个包下添加了@component注解的Bean单独实例化,其他的Bean都不实例化,如何解决;

Spring为我们提供了两种方式,use-default-filter属性值为false或者true

 <context:component-scan base-package="com.powernode.spring6.bean3" use-default-filters="false">
        <context:include-filter type="annotation" expression="org.springframework.stereotype.Controller"/>
    </context:component-scan>

添加use-default-filter属性,true时表示采用Spring默认规则,实例化被四个注解标注的Bean;

false表示,不在使用默认的实例化规则,全部都不实例化,如果想要实例化某个注解,就需要配置子标签context:include-filter,并在其中声明需要实例化的注解;

负责注入的注解

前面说了实例化Bean的注解@component,现在说替代set注入或者构造函数注入的注解;

@value注解

对于简单类型,都可以使用value注解,但是这种注解目前是硬编码;耦合度太高,我也不知道后面会咋弄;直接在添加了@component注解的类属性上添加@value注解;

@Component
 public class User {
     @Value(value = "zhangsan")
     private String name;
     @Value("20")
     private int age;
 }}

这种注入方式不依赖get\set方法,而且@value注解可以放在属性上、set方法上、构造方法上、或者构造方法的形参上;

@autowired注解

对于简单类型我们使用@value注解,那么对于复杂类型我们就使用@autowired注解;这种方式是自动装配,在之前我们接触过基于xml文件的自动装配,他是通过给Bean标签中添加autowire属性开启自动装配功能,然后他的属性值就会开始自动装配,底层调用的是set方法,是通过方法名来自动装配的,演示案例:

  1.     
     <bean id="userService" class="com.powernode.spring6.service.UserService" autowire="byName"/>
         
         <bean id="aaa" class="com.powernode.spring6.dao.UserDao"/>
     public class UserService {
         // 这里没修改
         private UserDao aaa;
     ​
         /*public void setAaa(UserDao aaa) {
             this.aaa = aaa;
         }*/
     ​
         // set方法名变化了,就无法自动装配了
         public void setDao(UserDao aaa){
             this.aaa = aaa;
         }
     ​
         public void save(){
             aaa.insert();
         }
     }

    使用autowire属性为byName这种自动装配时必须注意set方法名,还有一种是根据类型自动装配,但是此时需要注意的是符合条件的实现类必须只有一个;

  2. 无论是byName还是byType,在装配的时候都是基于set方法的。而采用注解方式时就不依赖set方法了;

使用@autowired注解的方式解决注入非简单类型的数据,默认是根据类型装配的;注意点有下面两点:
  • 第一处:该注解可以标注在哪里?

    • 构造方法上

      • 方法上

      • 形参上

      • 属性上

      • 注解上

  • <span style="color:red">第二处:该注解有一个required属性,默认值是true,表示在注入的时候要求被注入的Bean必须是存在的,如果不存在则报错。如果required属性设置为false,表示注入的Bean存在或者不存在都没关系,存在的话就注入,不存在的话,也不报错。</span>

由于@autowired注解采用的是默认的方式也就是byType类型,所以实习类必须只有一个,否则会报错;

如何解决上面的问题

配合@Qualifier注解同时使用,在@Qualifier注解中指明Bean标签的名称即可;

   @Autowired
     @Qualifier("userDaoForOracle") // 这个是bean的名字。
     public void setUserDao(UserDao userDao) {
         this.userDao = userDao;
     }

总结:

  • @Autowired注解可以出现在:属性上、构造方法上、构造方法的参数上、setter方法上。

  • 当带参数的构造方法只有一个,@Autowired注解可以省略。多个的话就会报错

  • @Autowired注解默认根据类型注入。如果要根据名称注入的话,需要配合@Qualifier注解一起使用。

@Resource注解

这个注解使用的最广泛,最受欢迎,同样他的作用也是完成非简单类型的注入,

@Resource VS @autowired

@Resource注解也可以完成非简单类型注入。那它和@Autowired注解有什么区别?

  • @Resource注解是JDK扩展包中的,也就是说属于JDK的一部分。所以该注解是标准注解,更加具有通用性。(JSR-250标准中制定的注解类型。JSR是Java规范提案。)

  • @Autowired注解是Spring框架自己的。

  • @Resource注解默认根据名称装配byName,未指定name时,使用属性名作为name。通过name找不到的话会自动启动通过类型byType装配。

  • @Autowired注解默认根据类型装配byType,如果想根据名称装配,需要配合@Qualifier注解一起用。

  • @Resource注解用在属性上、setter方法上。

  • @Autowired注解用在属性上、setter方法上、构造方法上、构造方法参数上。

@Resource注解属于JDK扩展包,所以不在JDK当中,需要额外引入以下依赖:【如果是JDK8的话不需要额外引入依赖。高于JDK11或低于JDK8需要引入以下依赖。

<dependency>
   <groupId>jakarta.annotation</groupId>
   <artifactId>jakarta.annotation-api</artifactId>
   <version>2.1.1</version>
 </dependency>

spring.xml也变成注解类

所谓的全注解开发就是不再使用spring配置文件了。写一个配置类来代替配置文件。

<?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:context="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 http://www.springframework.org/schema/context/spring-context.xsd">
     <context:component-scan base-package="com.powernode.spring6.dao,com.powernode.spring6.service"/>
 </beans>

二者比较

 @Configuration
 @ComponentScan({"com.powernode.spring6.dao", "com.powernode.spring6.service"})
 public class Spring6Configuration {
 }

但是相应的测试程序就需要改变了:

 @Test
 public void testNoXml(){
     ApplicationContext applicationContext = new AnnotationConfigApplicationContext(Spring6Configuration.class);
     UserService userService = applicationContext.getBean("userService", UserService.class);
     userService.save();
 }

ClassPathXmlApplicationContext 类换成了 AnnotationConfigApplicationContext


他日若遂凌云志,敢笑黄巢不丈夫,加油!!!

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值