首页 > 编程知识 正文

springmvc常用5种注解,自定义注解实现aop

时间:2023-05-04 07:45:09 阅读:176712 作者:4488

摘要:spring是我们web开发中不可缺少的框架,基于传统的xml方法来配置bean实在太繁琐了。 spring2.5及更高版本注释的出现可以大大简化我们的配置。 本文来源于华为云社区《如何高效提升Java开发效率—Spring注解开发全套教程!》,作者:灰猴子。

为了避免在33558www.Sina.com/IOC中逐个声明类对象,首先根据每类的功能,在Spring中首先规定了基于组件的注释,大致分为以下四种。

一、使用注解标识组件

也可以标识由Spring IOC容器管理的组件,并将其理解为除数据库层、业务逻辑层和控制层组件以外的组件使用的注释。

普通组件:@Component

标识由Spring IOC容器管理的持久性层组件。 通常用于标记数据库层

持久化层组件:@Respository

标识由Spring IOC容器管理的业务逻辑层组件。

业务逻辑层组件:@Service

标识由Spring IOC容器管理的表示层控制器组件。

表述层控制器组件:@Controller

在33558www.Sina.com/Dao层部件中添加@Respository注释

/** *数据库层注释* */@ repositorypublicclassbookdao { publicvoidsavebook () system.out.println ) bookDao中的书已保存}从这四个注释中,可以首先对所有组件逐一进行分类。 为什么使用注释进行分类,归根到底是为了能够简单迅速地知道哪个类具有什么类型的功能,而且在方便之后通过这四个注释将组件添加到IOC容器中

同时这几个注解后面也可以添加一些参数,比如比较常用的一个是注解的括号中加value属性,来表示这个组件在容器中的ID,如果不设置value值时,默认的ID是类名的全称(第一个字母小写)。

3358www.Sina.com/@Respository注释用于一个表示层控制器组件不会导致错误,因此称为@respository、@Service、@Controller 便于将组件放入容器中。

3358 www.Sina.com/http://www.Sina.com /现在对所有组件进行了详细分类,这样是否意味着将所有组件添加到了IOC容器中? 如果是这样的话,我们真的实现了低码时代.

3358www.Sina.com/其中Spring在IOC上提供包扫描功能。 从这个名字可以看出,如下实例,

数据包扫描的具体操作如下。

由于包扫描取决于上下文命名空间,因此必须将此命名空间添加到IOC中。 添加命名空间的方法有两种。 一种方法是将以下代码添加到IOC头文件中。

xmlns 3360上下文=' http://www.spring framework.org/schema/context '但是,这很难记住,不建议使用。

另一种方法是单击屏幕正下方的Namespaces,从中找到并检查上下文。

容器中数据包扫描的代码如下:

扫描context : component-scan base-package=' com.spring '/context 3360 component-scan 3358 ww.Sina.com /上的代码

所以你也可以把这四个注解理解为是Spring分发给不同功能组件的一个“身份证”,只有拥有这四种“身份证”,那么这个组件就可以被加入到IOC容器中。但是,这样扫描的范围可能有点大。 那么,能再缩小一下进行数据包扫描的范围吗? 当然可以。

在这里有一点需要注意:事实上Spring并没有能力识别一个组件到底是不是它所标记的类型,例如:

上下文:

component-scan base-package="com.atguigu.component" resource-pattern="autowire/*.class"/> (1)扫描包含特定组件

如果我们仅仅是想要扫描包含特定特征的组件,那么我们可以如下方法:

<context:include-filter>子节点表示要包含的目标类

但是需要注意的是:由于context:component-scan默认是将所有的类全部都添加进去,所以在此基础上再添加是没有用的,需要在context:component-scan中加入属性参数use-default-filters,use-default-filters="true" 表示默认将所有的类都添加进去,false表示将所有的类都不添加进去,

如下代码表示仅仅扫描包含include-filter中所指特征的组件,其中的type用来表示使用何种类型的扫描表达式,expression后面跟表达式。

<context:component-scan base-package="com.spring" use-default-filters="false"> <context:include-filter type="annotation" expression="org.springframework.stereotype.Repository"/></context:component-scan> (2)扫描排除特定组件

尽然有扫描包含特定组件,那么就有扫描排除特定组件,

<context:exclude-filter>子节点表示要排除在外的目标类。

以下代码表示扫描除以下特征外的其他组件。

<context:component-scan base-package="com.spring"> <context:exclude-filter type="assignable" expression="com.spring.service.BookService"/> </context:component-scan>

同时component-scan下可以拥有若干个include-filter和exclude-filter子节点,来表示可以包含多个哪种特征的组件或排除具有哪种特征的组件。

关于上面说到的type中填写的过滤表达式类型及作用如下表:

最常用的上面两个,以下三个几乎不用:

type="aspectj" aspectj表达式

type="custom" 定义一个TypeFile,自己写一个类定义使用哪一个

type="regex" 利用正则表达式

注意有bug:有些小伙伴们在进行注解开发的时候注解和扫描都写的很完美,可就是不起作用,原因可能是缺少相应特有的一个jar包,在这里需要导入额外的一个aop包

spring-aop-4.0.0.RELEASE.jar

到这里,将组件添加到容器中的操作就算是完成了,

在我们将组件添加成功之后呢,我们可以在组件图标的右上角看到一个小S的图标,这个时候就表示这个组件已经成功的加入到了容器中,

3、实现注解的三步骤

总结一下实现注解的三步骤:

添加context依赖
context:component-scan为类添加相应的注解 导入aop包
spring-aop-4.0.0.RELEASE.jar三、组件自动装配

但是这样就结束了嘛?就这么轻松了嘛?之前学习的bean的作用域与生命周期这些都没用了嘛?当然不是!!!更重要的是组件还没获取呢!!!

那么接下来就来和大家讲一下使用注解开发的高端操作,让你知道使用注解是多么的香!!!

我们平常在使用类的时候,难免会在类中调用其他自定义的类对吧,就比如说,Controller组件中往往需要用到Service组件的实例,Service组件中往往需要用到Repository组件的实例。

那么如果我们对这些需要实例化的组件一个一个的在类中进行实例化,是不是就显得太麻烦了呢?哎,聪明机智的程序员们怎么会没有想到这一点呢!所以组件的自动装配就出现了,​

在spring中我们可以通过注解的形式来对组件进行自动的装配,那么到底如何对组件进行装配的呢?

其实是这样的,在IOC中指定要扫描的包时,<context:component-scan> 元素会自动注册一个bean的后置处理器:AutowiredAnnotationBeanPostProcessor的实例。该后置处理器可以自动装配标记了@Autowired、@Resource或@Inject注解的属性。

而上面的@Autowired、@Resource或@Inject这三个注解,就是我们在进行组件的自动装配时最常用的注解,

下面我和大家介绍一下这三种注解的具体使用。

1、@autowired注解

@autowired注解能够根据类型实现自动装配。无论是构造器、普通字段(即使是非public)、还是一切具有参数的方法都可以应用@Autowired注解

默认情况下,所有使用@Autowired注解的属性都需要被设置。当Spring找不到匹配的bean装配属性时,会抛出异常。

(1)@autowired装配原理

接下里我来和大家详细的讲一下@autowired注解的装配原理:

1、使用自动装配时,首先会根据类型去容器中查找相应的组件,这就类似于

getBean("bookService.class"),

2、如果没有找到就抛异常,如果找到一个就赋值

3、如果找到多个,那么也是有一定的装配依据的,并不是随便找一个进行装配。

首先根据属性名作为ID进行继续寻找,找到对应属性名的组件就进行装配,没有找到就报错,报错的原因是:使用变量名作为id进行匹配时候,没有找到对应的属性名

(2)@Qualifier指定装配ID

对于这种报错其实还有一种解决:就是使用@Qualifier("bookService")指定查找ID,找到就装配,找不到报错,指定查找ID的代码示例如下:

// 添加注解表示自动装填@Autowired@Qualifier("bookdao")private BookDao bookDao; (3)required—装配报错解决

那么要是每次找不到就报错,这样程序不是就崩了吗?对于这样的情况应该怎么办呢?其实还有一种解决办法,解决找不到报错:使用required参数,

@Autowired(required=false) required=false表示如果实在找不到,就装配null

反正装配的依据就是,按照多种规则查找合适的装配对象,直到查找成功,实在不成功就返回null。

(4)特殊属性的自动装配

上面是使用@Autowired注解的基本原理与步骤,我们直到spring的注解开发是十分强大的,下面我们再来说几个特殊的属性的装配。

@Autowired注解可以应用在数组类型的属性上,此时Spring将会把所有匹配的bean进行自动装配。

@Autowired注解也可以应用在集合属性上,此时Spring读取该集合的类型信息,然后自动装配所有与之兼容的bean。

@Autowired注解用在java.util.Map上时,若该Map的键值为String,那么 Spring将自动装配与值类型兼容的bean作为值,并以bean的id值作为键。

这样一来,@Autowired注解的自动装配是不是就显得十分的强大了,以后妈妈再也不用担心我new对象了!!!​

2、@Resource注解

@Resource注解要求提供一个bean名称的属性,若该属性为空,则自动采用标注处的变量或方法名作为bean的名称。

3、@Inject注解

@Inject和@Autowired注解一样也是按类型注入匹配的bean,但没有reqired属性。

以上就是进行自动装配时使用的三个注解,在这里再总结一下,

@autoWried是spring自带的,更强大一些,能够实现required=false

@Resource也是java自带的,扩展性更强,所以如果切换成另一个容器框架,@Resource还是可以用的,而@Inject和@Autowired注解一样也是按类型注入匹配的bean,但没有reqired属性。其实在日常开发中,我们最常用到的、功能最强大的注解还是@Autowired注解。所以记住这一个基本上就可以了。

然后再总结一下使用注解的好处,主要就是节省了new对象时麻烦,直接使用一个@Autowired注解,spring就可以自动的为该属性赋值,一般来说将组件加入到IOC的注解和@Autowired是结合使用的。

四、注解使用的小细节

其实在使用注解进行开发时还有一些小细节需要注意,我在这里给大家总结一下。

1、整合多个配置文件

当我们开发时的项目过大的时候,在一个配置文件写如配置有时候就不能满足我们的需求,所以Spring允许通过<import>将多个配置文件引入到一个文件中,进行配置文件的集成。这样在启动Spring容器时,仅需要指定这个合并好的配置文件就可以。import元素的resource属性支持Spring的标准的路径资源,

如下示例,我们有springmvc.xml和spring.xml两个配置文件,现在我们想要将spring.xml引入到springmvc.xml中,方法是:在springmvc.xml中写入下面代码:

<import resource="spring.xml"/> 2、路径书写问题

对于spring中常用地址书写,有时候需要使用classpath,而有时候又需要其他,针对不同的路径书写,有不一样的含义和使用:具体看下表:

3、获取组件时的问题

对于使用注解方法添加到容器中的组件,我们在IOC容器中是看不到的,那么获取它的时候应该如何获取呢?

我们上面也说了,在注解中不指定id的前提下,spring是会自动的为每一个组件设置一个第一个字母小写的组件的全称作为ID,(如Book类的ID默认是book)。在容器中获取组件的方法和以往一样,但是如果是单实例的话,一般建议以类为参数进行获取。如:

Book book = (Book)ioc.getBean(Book.class); 五、写在最后

以上就是Spring注解开发的全部知识点了,是不是觉得使用注解开发比原生代码简洁多了,注解也是SSM框架乃至之后开发会经常用到的东西,

点击关注,第一时间了解华为云新鲜技术~

版权声明:该文观点仅代表作者本人。处理文章:请发送邮件至 三1五14八八95#扣扣.com 举报,一经查实,本站将立刻删除。