Spring中接口的bean是如何注入,@service 为什么不能用在interface上

1、
Spring中接口的bean是如何注入的
大家都知道@Service注入的是实现类serviceImpl,那使用时怎么能获取到接口,而且还能调用到实现类的方法。

其中只注入了实现类serviceImpl的bean,接口只是用来接收的。这里就要说到@Autowired/@Resource的注入原理:@Autowired是Spring的注解,Autowired默认先按byType,如果发现找到多个bean,则再按照byName方式比对,如果还有多个bean,则报错Caused by: org.springframework.beans.factory.NoUniqueBeanDefinitionException;@Resource是JDK1.6支持的注解,默认按照byName进行装配。如果没有指定name属性,当注解写在字段上时,默认取字段名按名称查找,当注解写在setter方法上默认取属性名,当找不到与名称匹配的bean时才按照类型匹配。

再来说Controller获取实例的过程:使用@Autowired,程序在spring的容器中查找类型时TestService的bean,刚好找到有且只有一个此类型的bean,即TestServiceImpl,所以就把testServiceImpl自动装配到了Controller的实例TestService中。

问:如果一个接口有多个实现类时,通过注解获取实例时怎么知道应该获取的是哪一个实现类serviceImpl呢?

@Autowired需要结合@Qualifier,注:实现类的开头小写类名TestServiceImpl2->testServiceImpl2
@Autowired
@Qualifier(“testServiceImpl2”)
private TestService testService;
@Resource也可以使用(name=“testServiceImpl2”),如果不显示的指定name值,就会自动把实例变量的名称作为name的值
@Resource
private TestService testServiceImpl2;

问:为什么非要调用接口来多此一举,而不直接调用实现类serviceImpl的bean来得简单明了呢?

直接使用serviceImpl的bean是可以的,这样加一层接口的原因:1.AOP程序的思想,给别人调用的接口,调用这只想知道方法和功能,而对于这个方法内部逻辑实现并不关心。2.降低各个模块间的关联,实现松耦合、程序分成、最主要的体现了继承只能单继承,接口却可以多实现

2、
今天遇到了一个错误导致服务器没起来

错误描述大概是:控制层需要注入的userService对象,spring容器中没有 ,检查后发现,是@Service注解没有加,一开始我想着既然是需要userService接口对象,那我在这个接口上加一个@Service注解,运行后还是报错。

查了一下资料,突然醒悟,接口看作是一个抽象类,抽象类不可以实例化,而@Service 中包含了@Component注解,也就意味着@Service注解的作用就是将Spring中的Bean实例化,相互矛盾。

最终解决方案:将@Service加在接口实现类上

### MapStruct Mapper Spring ComponentModel BeanNotFound 解决方案 当使用 `MapStruct` 并将其集成到 Spring 应用程序中时,通常会在 `@Mapper` 注解中指定 `componentModel = "spring"` 属性[^2]。这使得生成的映射器实现类会被标记为 Spring 组件,并能够通过依赖注入机制正常工作。 然而,在某些情况下可能会遇到 `BeanNotFoundException` 或类似的错误提示,表示 Spring 容器未能正确扫描并注册这些由 MapStruct 自动生成的 Bean。以下是可能的原因以及解决方案: #### 1. **未启用组件扫描** 如果项目的包结构设计不合理或者缺少必要的配置文件,则可能导致 Spring 无法找到 MapStruct 自动生成的 Bean 实现类。确保在应用程序的主要启动类或配置类上启用了组件扫描功能: ```java @SpringBootApplication(scanBasePackages = {"com.example.mapper", "com.example.service"}) public class Application { public static void main(String[] args) { SpringApplication.run(Application.class, args); } } ``` 此外,确认 `scanBasePackages` 参数覆盖了所有包含 MapStruct 映射器接口及其对应实现类所在的路径[^3]。 #### 2. **POM 配置不当** 使用 Maven 构建工具时,需确保引入了正确的 MapStruct 和其处理器依赖项。例如: ```xml <dependency> <groupId>org.mapstruct</groupId> <artifactId>mapstruct</artifactId> <version>1.5.0.Final</version> </dependency> <dependency> <groupId>org.mapstruct</groupId> <artifactId>mapstruct-processor</artifactId> <version>1.5.0.Final</version> <scope>provided</scope> </dependency> ``` 同时建议按照引用中的方法设置全局的 `componentModel="spring"` 编译参数来减少重复劳动[^1]。修改 pom.xml 文件如下所示: ```xml <build> <plugins> <plugin> <groupId>org.apache.maven.plugins</groupId> <artifactId>maven-compiler-plugin</artifactId> <version>3.8.1</version> <configuration> <annotationProcessorPaths> <path> <groupId>org.mapstruct</groupId> <artifactId>mapstruct-processor</artifactId> <version>1.5.0.Final</version> </path> </annotationProcessorPaths> <compilerArgs> <arg>-Amapstruct.defaultComponentModel=spring</arg> </compilerArgs> </configuration> </plugin> </plugins> </build> ``` #### 3. **命名冲突或其他技术限制** 当存在多个同名的方法签名(即函数重载),即使它们返回不同类型也可能引发混淆问题,从而阻止 MapStruct 成功创建预期的行为逻辑[^4]。针对这种情况应调整方法名称以消除歧义,比如将两个相似的功能分别命名为不同的标识符如 `toBudgetOccupyBO` 和 `toSingleBudgetOccupyBO`。 --- ### 总结 综上所述,要彻底解决 UserConverter 接口定义下的 Mapper 被 Spring 所忽略的情况,可以从以下几个方面入手:一是核查是否已开启全面的组件扫描;二是审查 POM 中关于 MapStruct 的各项设定是否准确无误;三是留意潜在的名字碰撞现象并对相应部分加以修正。 ```java // 示例代码片段展示如何声明一个简单的 MapStruct Mapper @Mapper(componentModel = "spring") public interface UserConverter { UserDto userToUserDto(User entity); } ```
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值