服务器之家:专注于服务器技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - Java教程 - 一个@Component注解引发的大坑

一个@Component注解引发的大坑

2021-12-23 13:19老汪丶 Java教程

这篇文章主要介绍了一个@Component注解引发的大坑,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教

一个@Component注解引发的大坑

首先,我们这个是用springboot架构来实现的业务

这是项目包结构和配置文件结构

一个@Component注解引发的大坑

一个@Component注解引发的大坑

这是定时需要执行的任务

一个@Component注解引发的大坑

这是我执行PromoCodeCMCJob这个定时器的报错信息

一个@Component注解引发的大坑

问题是这样的

加了两个MQ之后才报错的这个信息,当我执行启动PromoCodeCMCJob定时任务的时候就报错,报错信息如上,这个时候我两个MQ任务的定时任务都加了@Component和 @Autowired private MQProduceFactory gpRepayProducer 注入,此时就报错了,

刚开始的时候我只是 @Autowired private MQProduceFactory gpRepayProducer 注释掉然后再启动PromoCodeCMCJob这个定时任务,什么,居然没问题,这个时候完全能执启动,所以此时我就断定是这个MQProduceFactory注入原因引起的,但是我有MQ队列,这个是MQ的生产者,

注释掉就不能忘MQ生产消息了,所以注释掉这个是不行的,然后我就想其它办法去解决,像重新分包,想了各种解决办法都没有解决,始终没有想到去掉@Component这个注解,这个注解翻译是组件的意思,

是把当前类注入到IOC容器中,直到我研究了半天左右的时候,一直没有想到点上,浪费了很多时间,快下班了的时候让我想到了,既然被在IOC容器中了,那我不把他注入进去不就完了,这是我就试着把@Component注解去掉,然后在配置文件中加入bean,变成了这样

一个@Component注解引发的大坑

配置文件中不加这个bean的话启动定时任务也是会报错的,这样的话我就把需要的MQProduceFactory这个放入到bean里面了,而没有在IOC了,最后就完美的解决了,有MQ三个定时任务都能运行了!

思考

@Component注解是IOC容器层面的,是个很大的容器,能包容很多,而bean只是一个工厂,IOC容器能包含着bean工厂,IOC容器中可以有很多bean工厂,

但是bean里面只有一个,一般都是通过bean标签的id来标识的,所以在项目中没有最完美的方案,只有更好的方案,只有把架构,方案都用对了,才能更好去解决业务需求的问题。

 

对spring @component注解的理解

  • 1、@controller 控制器(注入服务)
  • 2、@service 服务(注入dao)
  • 3、@repository dao(实现dao访问)
  • 4、@component (把普通pojo实例化到spring容器中,相当于配置文件中的<bean id="" class=""/>)

Spring 2.5 中除了提供 @Component 注释外,还定义了几个拥有特殊语义的注释,它们分别是:@Repository、@Service 和 @Controller。在目前的 Spring 版本中,这 3 个注释和 @Component 是等效的,但是从注释类的命名上,很容易看出这 3 个注释分别和持久层、业务层和控制层(Web 层)相对应。虽然目前这 3 个注释和 @Component 相比没有什么新意,但 Spring 将在以后的版本中为它们添加特殊的功能。所以,如果 Web 应用程序采用了经典的三层分层结构的话,最好在持久层、业务层和控制层分别采用 @Repository、@Service 和 @Controller 对分层中的类进行注释,而用 @Component 对那些比较中立的类进行注释。

在 一个稍大的项目中,通常会有上百个组件,如果这些组件采用xml的bean定义来配置,显然会增加配置文件的体积,查找以及维护起来也不太方便。 Spring2.5为我们引入了组件自动扫描机制,他可以在类路径底下寻找标注了 @Component,@Service,@Controller,@Repository注解的类,并把这些类纳入进spring容器中管理。它的作用 和在xml文件中使用bean节点配置组件时一样的。

今天在做项目时把一个既不是业务控制(Controller)、又不是业务逻辑(Service)、也不是数据访问(Dao)的类在不理解用注解@component的情况下注解成为Controller使用,结果肯定是有问题了,所以请教以后发现只用@component注解就完全可以解决了,所以在网上查阅了@component注解,特此记录一下,方便日后查阅。

1、@controller 控制器(注入服务)

  • 用于标注控制层,相当于struts中的action层

2、@service 服务(注入dao)

  • 用于标注服务层,主要用来进行业务的逻辑处理

3、@repository(实现dao访问)

  • 用于标注数据访问层,也可以说用于标注数据访问组件,即DAO组件.

4、@component (把普通pojo实例化到spring容器中,相当于配置文件中的 <bean id="" class=""/>)

  • 泛指各种组件,就是说当我们的类不属于各种归类的时候(不属于@Controller、@Services等的时候),我们就可以使用@Component来标注这个类。

说明:

<context:component-scan base-package=”com.*”> 

上面的这个例子是引入Component组件的例子,其中base-package表示为需要扫描的所有子包。

共同点:被@controller 、@service、@repository 、@component 注解的类,都会把这些类纳入进spring容器中进行管理

@Component注解的使用

@Component取代 或者 @Component(“id”) 取代配置文件applicationContext.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"
     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.fly.spring.annotation"></context:component-scan>
</beans>

注解的类

package com.fly.spring.annotation; 
import org.springframework.stereotype.Component; 
@Component("person")
public class Person { 
  public void getString(){
      System.out.println("*********hello everyOne**********");
  }
}

测试的类

package com.fly.spring.annotation; 
import org.junit.Test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext; 
public class TestDemo {

  @Test
  public void demo(){
      String xmlpath = "com/fly/spring/annotation/applicationContext.xml";
      ApplicationContext context = new ClassPathXmlApplicationContext(xmlpath);
      Person bean = (Person) context.getBean("person");
      bean.getString();
  }
}

以上为个人经验,希望能给大家一个参考,也希望大家多多支持服务器之家。

原文链接:https://blog.csdn.net/wry11687603/article/details/90676644

延伸 · 阅读

精彩推荐