com.alibaba.dubbo.config.annotation.Service和org.springframework.stereotype.Service两种@Service的区别

本文详细解析了Spring框架中的@Service、@Controller、@Repository、@Component注解及Dubbo框架中的@Service注解的用途与配置。比较了Spring与Dubbo在服务提供者与消费者上的不同配置方式。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

org.springframework.stereotype.Service

属于Spring
作用:只是用来表示此类是业务层组件

spring配置文件:

<!-- 配置包扫描器 -->
<context:component-scan base-package="com.zm.service"/>

常用的作用在类上的注解,还有以下几种:

    @Controller
    用于标注控制层组件(如Struts的action,springMVC的controller)
    @Service
    用于标注业务层组件,主要进行业务的处理(通常定义在servic层)
    @Repository
    用于标注数据访问层组件
    @Component
    用于标注泛指组件,当我们不知道某类属于哪类时(不属于@Controller、@Services等时),就可以标注为@Componet

以上所有的的作用在类上的注解,归根到底,都是讲注解的类纳入到spring容器进行管理,之所以名字不同,只是表示组件的名称不同,就如世上所有的人都有名字(注解名),但他们都是人(标识为组件,放到spring容器)

 

com.alibaba.dubbo.config.annotation.Service

属于dubbo

作用:是dubbo用来声明provider的注解(即声明该类是服务的提供者的注解)

spring配置文件:

<!-- 配置包扫描器,alibaba的@Service注解 -->
 <dubbo:annotation package="com.zm.service" />

<!-- 提供方应用信息,用于计算依赖关系 -->
<dubbo:application name="e3-manager" />
<dubbo:registry protocol="zookeeper" address="127.0.0.1:2181" />
<!-- 用dubbo协议在20880端口暴露服务 -->
<dubbo:protocol name="dubbo" port="20880" />
 

服务的提供者

如果你的程序是分布式程序,且使用的dubbo实现的RPC,可以有两种配置:

1.使用spring的@Service

必须在spring配置文件中非注解的方式显示暴露一下服务接口,存在的弊端:如果有好多服务提供者,要在spring配置文件中写好多dubbo:service,配置如下:

<!-- 配置包扫描器,spring的@Service注解 -->
<context:component-scan base-package="com.zm.service"/>

<!-- 提供方应用信息,用于计算依赖关系 -->
<dubbo:application name="e3-manager" />
<dubbo:registry protocol="zookeeper" address="127.0.0.1:2181" />
<!-- 用dubbo协议在20880端口暴露服务 -->
<dubbo:protocol name="dubbo" port="20880" />
<!-- 声明需要暴露的服务接口 -->
<dubbo:service interface=com.zm.service.ItemService" ref="itemServiceImpl" timeout="600000"/>

 

第二种:使用alibaba的dubbo的@Service
此时就不需要在spring配置文件中显示声明要暴露的服务接口了,spring配置文件中省去好多dubbo:service,配置如下:

<!-- 配置包扫描器,alibaba的@Service注解 -->
<dubbo:annotation package="com.zm.service" />

<!-- 提供方应用信息,用于计算依赖关系 -->
<dubbo:application name="e3-manager" />
<dubbo:registry protocol="zookeeper" address="127.0.0.1:2181" />
<!-- 用dubbo协议在20880端口暴露服务 -->
<dubbo:protocol name="dubbo" port="20880" />
 

 

 

服务的消费者

 

dubbo注解

第一种:使用@Reference引用服务
spring配置文件如下:

<!-- 配置包扫描器 -->
<dubbo:annotation package="com.zm.controller" />

<!-- 引用dubbo服务 -->
<dubbo:application name="e3-manager-web"/>
<dubbo:registry protocol="zookeeper" address="127.0.0.1:2181"/>   

Controller类上面使用@Reference表明我要引用的服务

<!-- 也是alibaba的注解-->
import com.alibaba.dubbo.config.annotation.Reference;

@Reference
private ItemService itemService;
 

第二种:使用autowired注解

spring配置文件:

<!-- spring的@service -->
<context:component-scan base-package="com.zm.controller" />

<!-- 引用dubbo服务 -->
<dubbo:application name="e3-manager-web"/>
<dubbo:registry protocol="zookeeper" address="192.168.20.191:2181"/>    
<dubbo:reference interface="com.zm.service.ItemService" id="itemService" />

Controller类上面使用@Autowired表明我要引用的服务

import org.springframework.beans.factory.annotation.Autowired;

@Autowired
private ItemService itemService;

这样写不推荐,因为有一下两点

1、当你配置的引用的服务没有提供者是,编译阶段就会出错,因为@Autowired默认的去找bean,没有找到对应的bean,就去你引用的dubbo服务中去找,但是这是没有提供者,即没有这个真正的类,所以编译就报错了;而用alibaba的@Service只是一个引用,它能找到这个引用,但是如果没有提供者,也会报错ERROR,显示No Provider没有提供者,但是能编译过去项目能正常运行
2、如果引用的dubbo服务非常多的话,spring配置文件要写好多的dubbo:reference

 

最后,谢谢老哥的文章:https://blog.youkuaiyun.com/wrs120/article/details/80468174

<think>好的,我现在需要帮用户解决一个Java的异常问题:java.lang.IllegalArgumentException: @Service interfaceClass() or interfaceName() or interface class must be present!。首先,我得理解这个异常的具体含义出现的原因。 用户可能是在使用Spring框架的@Service注解时遇到了问题。根据异常信息,问题出在@Service注解的interfaceClass()、interfaceName()或者接口类没有正确指定。通常,这个异常在使用Spring Cloud的Feign客户端或者Dubbo等需要指定服务接口的框架时出现,因为这些框架要求明确声明服务接口。 接下来,我需要回忆一下@Service注解在不同框架中的用法。比如,在Dubbo中,@Service注解是用来暴露服务的,需要指定interfaceClass属性来声明服务接口。而在Spring Cloud Feign中,可能也有类似的要求。如果用户在使用这些框架时,忘记在@Service注解中指定接口类,就会触发这个异常。 可能的错误场景包括:用户在注解中没有提供interfaceClass或interfaceName属性,或者提供的类名不正确,或者服务类没有实现指定的接口。例如,用户可能写了一个服务类,但是没有正确实现接口,或者在注解中没有指定接口。 接下来,我需要考虑如何引导用户解决这个问题。首先,检查@Service注解的使用是否正确,确认是否有指定interfaceClass或interfaceName。然后,确认接口是否存在并且路径正确。此外,还要确保服务类确实实现了该接口,避免拼写错误或包路径错误。 可能的解决方案步骤包括: 1. 检查@Service注解是否正确添加了interfaceClass或interfaceName属性。 2. 确保指定的接口类存在,并且包路径正确。 3. 验证服务类是否实现了该接口。 4. 如果是Dubbo或Feign,查看相关文档确认注解的正确用法。 同时,我需要举一个具体的例子,说明正确的用法是什么样的。比如,在Dubbo中,正确的注解应该是@Service(interfaceClass = UserService.class),这样框架才能正确识别服务接口。 另外,需要提醒用户检查依赖版本是否兼容,有时候版本不匹配也可能导致注解属性不被识别。但根据异常信息,更可能的原因是注解参数缺失,所以优先检查注解的写法。 最后,总结解决方法,强调检查注解参数、接口是否存在以及实现是否正确。这样用户就可以逐步排查问题,找到导致异常的具体原因。</think>当你在使用Spring框架(特别是结合Dubbo或Feign等组件)时遇到`java.lang.IllegalArgumentException: @Service interfaceClass() or interfaceName() or interface class must be present!`异常,通常是因为**未正确声明服务接口**。以下是逐步分析与解决方法: --- ### 1️⃣ **异常原因** - **场景**:在使用`@Service`注解(非Spring原生注解,常见于Dubbo或自定义注解)时,**未指定服务接口**。 - **触发条件**:框架要求通过`interfaceClass`或`interfaceName`属性显式声明服务接口,但实际代码中未提供。 - **示例错误代码**: ```java @Service // 缺少接口声明 public class UserServiceImpl implements UserService { // ... } ``` --- ### 2️⃣ **解决方案** #### 步骤1:检查注解来源 - **Dubbo**:确认使用的是`org.apache.dubbo.config.annotation.Service`注解,而非Spring的`@Service`。 - **Feign/其他**:如果是自定义服务注解,需检查文档是否要求声明接口。 #### 步骤2:显式指定接口 在`@Service`注解中**明确声明接口**: ```java // Dubbo示例 import org.apache.dubbo.config.annotation.Service; @Service(interfaceClass = UserService.class) // 指定interfaceClass public class UserServiceImpl implements UserService { // ... } ``` 或通过`interfaceName`(需全限定类名): ```java @Service(interfaceName = "com.example.UserService") public class UserServiceImpl implements UserService { // ... } ``` #### 步骤3:验证接口实现 - 确保`UserService`是一个**接口**,且`UserServiceImpl`类确实实现了它。 - 检查接口的包路径是否与声明的一致(避免拼写错误)。 #### 步骤4:检查依赖版本 - **Dubbo版本兼容性**:确认`dubbo-spring-boot-starter`版本与Spring Boot版本匹配。 - **注解冲突**:避免同时引入Spring的`@Service`Dubbo的`@Service`导致混淆。 --- ### 3️⃣ **常见错误场景** - **错误1**:误用Spring的`@Service`注解(`org.springframework.stereotype.Service`),而非Dubbo的注解。 - **错误2**:接口类名拼写错误或未编译到class文件。 - **错误3**:未实现声明的接口。 --- ### 4️⃣ **代码修正示例** ```java // 正确导入Dubbo@Service注解 import org.apache.dubbo.config.annotation.Service; // 显式声明interfaceClass @Service(interfaceClass = UserService.class) public class UserServiceImpl implements UserService { @Override public String getUser() { return "User"; } } ``` --- ### 5️⃣ **总结** 此异常的核心问题是**服务接口未正确声明**。通过以下步骤排查: 1. 确认使用正确的`@Service`注解(Dubbo或其他框架)。 2. 添加`interfaceClass`或`interfaceName`属性。 3. 验证接口存在性、实现关系路径正确性。 若问题仍未解决,建议检查框架文档或调试框架源码(查看`@Service`注解的处理逻辑)。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值