被各种注解搞晕了?那快来看看Spring Bean注解详解!

前言

本篇博客中,我们将会讨论用于声明不同类型 Beans 的几种最常用的 Spring Bean 注解。

众所周知,Spring 容器中有许多配置 Bean 的方法,我们既可以通过 XML 配置,也可以在配置类中使用 @Bean 注解来声明 Beans。此外,我们还可以使用 org.springframework.stereotype 包中的一个注解来对类进行标记,然后把其余工作交给组件扫描即可。

组件扫描

在 Spring 中,一旦我们启用了组件扫描, Spring 就会自动扫描包中的 Bean。

通过使用 @ComponentScan ,Spring 就会自动去扫描那些带有注释配置的类,我们可以使用 basePackages 或者 value 参数(两者是一样的,value 只不过是 basePackages 的另一种称呼)来直接指定我们所要扫描的包的名称,然后 Spring 就会去扫描我们指定包下所有带有 @Component 注解的类,然后将其自动注册为一个 Bean。

@Configuration
// 以下两者之一即可
@ComponentScan(basePackages = "com.cunyu.annotions")
// @ComponentScan(value = "com.cunyu.annotions")
class PetFactoryConfig{
    ……
}

此外,我们还可以使用 basePackageClasses 参数来指向基础包中的类。

@Configuration
@ComponentScan(basePackageClasses = PetFactoryConfig.class)
class PetFactoryConfig{
    ……
}

basePackages 和 basePackageClasses 两个参数都是数组类型的,所以在传参时我们可以为他们提供多个包。

而如果没有为 @ComponentScan 指定参数,那么 Spring 就只会扫描和 @ComponentScan 注释的类位于同一个包的带有 @Component 注解的其他类,然后将它们自动创建为一个 Bean。

@ComponentScan 充分利用了 Java 8 中的重复注解特性,因此我们能够用它来多次标记一个类:

@Configuration
@ComponentScan(basePackages = "com.cunyu.annotions")
@ComponentScan(basePackageClasses = PetFactoryConfig.class)
class PetFactoryConfig{
    ……
}

除开上面的方式来标记一个类外,我们还可以使用 @CompentScans 来将多个 @ComponentScan 包含起来,用于指定多个 @ComponentScan 配置。

@Configuration
@ComponentScans({
    @ComponentScan(basePackages = "com.cunyu.annotions"),
    @ComponentScan(basePackageClasses = PetFactoryConfig.class)
})
class PetFactoryConfig{
    ……
}

除开使用注解的方式来实现组件扫描之外,我们还可以通过配置 XML 来进行,只需要在我们的配置文件中如下内容即可:

<?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";
       xmlns:c="http://www.springframework.org/schema/c";
       xmlns:p="http://www.springframework.org/schema/p";
       xsi:schemaLocation="http://www.springframework.org/schema/beanshttp://www.springframework.org/schema/beans/spring-beans.xsd
                           http://www.springframework.org/schema/contexthttp://www.springframework.org/schema/context/spring-context.xsd">;

    <context:component-scan base-package="com.cunyu.dao" />
    <context:component-scan base-package="com.cunyu.service" />
    <context:component-scan base-package="com.cunyu.controller" />
</beans>

context:component-scan 元素将实现同注解 @ComponentScan 一样的效果,即扫描 com.cunyu 包下所有带有 @Component 注解的类,并将它们注册创建为 Bean。

@Component

上一小节中我们讲了 @ComponentScan 会自动扫描组件过程中会去扫描那些带有 @Component 注解的类,并将其注册创建为 Bean,比如下面的例子:

@Component
public class Cat{
    ……
}

其中 Cat 类中带有 @Component 注解,当 Spring 自动扫描时,就会去扫描 Cat 这个类,并创建一个名为 cat 的 Bean 实例。

注意:默认情况下,使用 @ComponentScan 去扫描 @Component 注解的类,生成对应类的 Bean 实例时,Bean 实例具有与类名相同的名称,但不同的是 Bean 实例的首字母是小写,而一般类名首字母是大写。

@Component 是任意 Spring 管理组建的通用构造型,当组件不好归类时,一般使用该注解,又可以分为如下几个常用元注解:

  • @Repository:位于持久层,能将数据库操作跑出的原生异常转换为 Spring 持久层异常,用于标注数据访问组件,即 DAO 组件;
  • @Service:位于业务逻辑层,只是标注该类位于业务层逻辑;
  • @Configuration:用于定义配置类,可替换 XML 配置文件,被注解的类内部包含一个或多个被 @Bean 注解的方法,这些方法将会被 AnnotationConfigApplicaitonContext 或者 AnnotaionConfigWebApplicationContext 类扫描,并用于构建 Bean 定义,初始化 Spring 容器;
  • @Controller:属于 Spring MVC 的注解,进行前端请求的处理、转发、重定向;用于标注控制层组件;

它们是针对不同使用场景而采取的带有特定功能化的注解组件,其实质功能其实和 @Component 一样。因此,如果一个类被 @Component 注解了,那么就可以根据这个类的实际功能,利用 @Repository、@Service … 等代替,而且代替后的注解会具备更多的功能。

@Repository

DAO(Data Access Object,数据访问对象,为某种类型的数据库或其他持久性机制提供一个抽象接口的对象) 或者 Repository 类通常代表应用程序中的数据访问层,我们一般倾向于使用 @Repository 注解

@Repository
public class PetRepository{
    ……
}

通过使用 @Repository 注解,它将启用自动持久化异常转换。此时,当我们使用一些持久化框架,比如 Hibernate、MyBatis……时,当带有 @Repository 注解的类在抛出本地异常时,就会自动将其转换为 Spring 中的 DataAccessException 的子类。

而要启动异常转换,我们就需要自己去声明我们 PersistenceExceptionTranslationPostProcessor 的实例,声明的方式可以分为注解性和 XML 配置型。但是一般而言,Spring 都会自动帮我们去完成这个过程,所以我们就不再需要自己去手动声明了。

  • 注解型
@Bean
public PersistenceExceptionTranslationPostProcessor exceptionTranslation(){
    return new PersistenceExceptionTranslationPostProcessor();
}

  • XML 配置型
<bean class="org.springframework.dao.annotion.PersistenceExceptionTranslationPostProcessor" />

@Service

对于应用程序中的业务逻辑,一般都是位于服务层,因此我们使用 @Service 注解一个类,来指明该类属于服务层。表示定义一个 Bean,自动根据所标注的组件实例化一个首字母为小写的 Bean。实例如下:

@Service
public class PetService{
    ……
}

上述代码中 PetService 类被标注为一个 Bean,其名称为 petServie。

@Configuration

一般用于配置类,而且还可以包含用 @Bean 所注解的 Bean 定义方法,实例如下:

@Configuration
public class PetFactoryConfig{
    @Bean
    public Dog dog(){
        return new Dog();
    }
}

既然说了使用 @Confgiuration 可以和 XML 配置文件互换,那么以上的配置类等价于如下配置:

<beans>
    <bean id="dog" class = "com.cunyu.dao.Dog"/>
</beans>

注意:使用 @Configuration 注解时,一般需要遵循如下原则:

  1. @Configuration 注解的类不可以是 final 类型;
  2. @Configuration 注解的类不可以是匿名类;
  3. 嵌套的 @Configuration 必须是静态类;

@Controller

@Controller 注解是一个类级别的注解,当把它用在类上时,表示该类在 Spring MVC 中充当控制器,该类将被 Spring 自动扫描,一般我们在该类中加入 @RequestMapping("…"),就可以直接使用浏览器来访问对应界面进行逻辑处理了。实例如下:

@Controller
public class PetController{
    @RequestMapping(value = "/hello", method = RequestMethod.GET)
    public String sayHello(){
        return "你好";
    }
    ……
}

此时,当我们去浏览器中访问 localhost:8080/hello 时,页面中就会显示 你好 这条信息。

注意,@Controller 注解的类下,我们又发现了 RequestMapping 注解,该注解主要 6 个属性,分别介绍如下:

  • value :用于指定浏览器指定的地址;
  • method,指定请求的 method 类型,一般有 GET、POST、PUT、DELETE,而现在我们一般使用对应的注解 @GetMapping、@PostMapping、@PutMapping、@DeleteMapping;
  • consumes:用于指定处理请求的提交内容类别(Content-Type),如 application/json、test/html;
  • produces:指定返回的内容类别;
  • params:指定 request 中必须包含某些引用数值时,才能让该方法处理;
  • headers:指定 request 中必须包含某些指定 header 值,才能让该方法处理请求;

原型注解及 AOP

当我们使用 Spring 原型注解时,可以十分容易地创建一个指向所有具有特定构造型的类的切入点。

比如我们如果想要衡量一个方法在 DAO 层中的执行时间,就可以充分利用 @Repository 注解的特点。

@Aspect
@Component
public class PerformanceAspect {
    @Pointcut("within(@org.springframework.stereotype.Repository *)")
    public void repositoryClassMethods() {};

    @Around("repositoryClassMethods()")
    public Object measureMethodExecutionTime(ProceedingJoinPoint joinPoint) 
      throws Throwable {
        long start = System.nanoTime();
        Object returnValue = joinPoint.proceed();
        long end = System.nanoTime();
        String methodName = joinPoint.getSignature().getName();
        System.out.println(
          "Execution of " + methodName + " took " + 
          TimeUnit.NANOSECONDS.toMillis(end - start) + " ms");
        return returnValue;
    }
}

以上实例中,我们创建了一个切入点,该切入点会去匹配带有 @Repository 注解的类中的所有方法。然后我们用 @Around 通知来定位切入点,并确定被拦截的方法调用的执行时间。通过使用这个方式,我们就可以轻松地给每个应用程序添加日志记录、性能管理、审计以及其他行为。

总结

好了,以上就主要介绍了 Spring 中的原型注释,并了解了它们各自所代表的意义。此外,还学习了如何使用组件扫描功能,从而告知 Spring 容器在何处能找到带有注解的类。

如果你有更多的见解,欢迎评论留言,一起交流呀!

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 218,036评论 6 506
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,046评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 164,411评论 0 354
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,622评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,661评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,521评论 1 304
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,288评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,200评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,644评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,837评论 3 336
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,953评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,673评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,281评论 3 329
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,889评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,011评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,119评论 3 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,901评论 2 355

推荐阅读更多精彩内容