【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@Componentpublic @interface Controller {@AliasFor(annotation = Component.class)String value() default "";}​​@Target({ElementType.TYPE})@Retention(RetentionPolicy.RUNTIME)@Documented@Componentpublic @interface Service {@AliasFor(annotation = Component.class)String value() default "";}​@Target({ElementType.TYPE})@Retention(RetentionPolicy.RUNTIME)@Documented@Componentpublic @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.xsdhttp://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注解;

@Componentpublic 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.xsdhttp://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 {}

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

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

ClassPathXmlApplicationContext 类换成了 AnnotationConfigApplicationContext


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

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/209976.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

分类预测 | Matlab实现基于PSO-PNN粒子群算法优化概率神经网络的数据分类预测

分类预测 | Matlab实现基于PSO-PNN粒子群算法优化概率神经网络的数据分类预测 目录 分类预测 | Matlab实现基于PSO-PNN粒子群算法优化概率神经网络的数据分类预测分类效果基本描述程序设计参考资料 分类效果 基本描述 1.Matlab实现基于PSO-PNN粒子群算法优化概率神经网络的数据…

Spring-jdbcTemplate-配置数据库连接池

1、jdbc.properties jdbc.drivercom.mysql.cj.jdbc.Driver jdbc.urljdbc:mysql:///studb jdbc.userroot jdbc.pwd123456 2、beans.xml <?xml version"1.0" encoding"UTF-8"?> <beans xmlns"http://www.springframework.org/schema/beans&…

YOLOv8使用自己训练的模型,将检测图片进行可视化:效果超过YOLOv5模型,丰富改进模型的检测展示

💡更多改进内容📚 芒果专栏 💡🚀🚀🚀内含改进源代码 按步骤操作运行改进后的代码即可💡更方便的统计更多实验数据,方便写作 YOLOv8使用自己训练的模型,将检测图片进行可视化:效果超过YOLOv5模型,丰富改进模型的检测展示 文章目录 核心代码改进新增代码部分修…

【深度学习】Transformer简介

近年来&#xff0c;Transformer模型在自然语言处理&#xff08;NLP&#xff09;领域中横扫千军&#xff0c;以BERT、GPT为代表的模型屡屡屠榜&#xff0c;目前已经成为了该领域的标准模型。同时&#xff0c;在计算机视觉等领域中&#xff0c;Transformer模型也逐渐得到了重视&a…

Springboot将多个图片导出成zip压缩包

Springboot将多个图片导出成zip压缩包 将多个图片导出成zip压缩包 /*** 判断时间差是否超过6小时* param startTime 开始时间* param endTime 结束时间* return*/public static boolean isWithin6Hours(String startTime, String endTime) {// 定义日期时间格式DateTimeFormatt…

Python教程73:Pandas中一维数组Series学习

创建一维数据类型Series dataNone 要转化为Series的数据(也可用dict直接设置行索引) 若是标量则必须设置索引,该值会重复,来匹配索引的长度 indexNone 设置行索引 dtypeNone 设置数据类型(使用numpy数据类型) nameNone 设置Series的name属性 copyFalse 不复制 (当data为ndarray…

每日一题 1410. HTML 实体解析器(中等,模拟)

模拟&#xff0c;没什么好说的 class Solution:def entityParser(self, text: str) -> str:entityMap {&quot;: ",&apos;: "",>: >,<: <,&frasl;: /,&amp;: &,}i 0n len(text)res []while i < n:isEntity Falseif …

【从入门到起飞】JavaSE—多线程(3)(生命周期,线程安全问题,同步方法)

&#x1f38a;专栏【JavaSE】 &#x1f354;喜欢的诗句&#xff1a;路漫漫其修远兮&#xff0c;吾将上下而求索。 &#x1f386;音乐分享【如愿】 &#x1f384;欢迎并且感谢大家指出小吉的问题&#x1f970; 文章目录 &#x1f354;生命周期&#x1f384;线程的安全问题&#…

普通平衡树

题意&#xff1a;略&#xff0c;题中较清晰。 用二叉查找树来存储数据&#xff0c;为了增加效率&#xff0c;尽量使左子树和右子树的深度差不超过一&#xff0c;这样可以时间控制在logn&#xff0c;效率比较高。 右旋和左旋&#xff0c;目的是为了维护二叉树的操作&#xff0…

【计算思维】蓝桥杯STEMA 科技素养考试真题及解析 6

1、明明买了一个扫地机器人&#xff0c;可以通过以下指令控制机器人运动: F:向前走 10 个单位长度 L:原地左转 90 度 R:原地右转 90 度 机器人初始方向向右&#xff0c;需要按顺序执行以下那条指令&#xff0c;才能打扫完下图中的道路 A、F-L-F-R-F-F-R-F-L-F B、F-R-F-L-F-F…

innoDB的缓冲池(Buffer Pool)的工作原理

数据存在磁盘了&#xff0c;总不能次次和磁盘交互吧&#xff0c;所以innoDB有一个缓冲池&#xff08;Buffer Pool&#xff09;&#xff0c;有了缓冲池后&#xff0c;读写就优先在缓冲池了。读先在缓冲池读&#xff0c;没有再去磁盘加载进缓冲池&#xff1b;写也是先写缓冲池&am…

CSGO搬砖干货,全网最详细教学!

CSGO游戏搬砖全套操作流程及注意事项&#xff08;第一课&#xff09; 在电竞游戏中&#xff0c;CSGO&#xff08;Counter-Strike: Global Offensive&#xff09;被广大玩家誉为经典之作。然而&#xff0c;除了在游戏中展现个人实力和团队合作外&#xff0c;有些玩家还将CSGO作为…