大家都知道@Service注入的是实现类serviceImpl,那使用时怎么能获取到接口,而且还能调用到实现类的方法。
接口:
public interface TestService{ public String test(); }
实现类:
@Service public class TestServiceImpl implements TestService{ @Override public String test(){ return "TestServiceImpl "; } }
Controller类:
@RestController public class TestController{ @Autowired private TestService testService; @RequestMapping("/test") public String test(){ return testService.test(); } }
请求结果:
其中只注入了实现类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呢?
增加一个新的实现类:
@Service public class TestServiceImpl2 implements TestService{ @Override public String test() { return "TestServiceImpl2"; } }
1.通过指定bean的名称来明确到底要实例哪一个类
@Autowired需要结合@Qualifier,注:实现类的开头小写类名TestServiceImpl2->testServiceImpl2
@Autowired @Qualifier("testServiceImpl2") private TestService testService;
@Resource也可以使用(name="testServiceImpl2"),如果不显示的指定name值,就会自动把实例变量的名称作为name的值
@Resource private TestService testServiceImpl2;
2.通过在实现类上添加@Primary注解来指定默认加载类,如果任然指定了bean的名字则以指定的为准
@Service @Primary public class TestServiceImpl2 implements TestService{ @Override public String test() { return "TestServiceImpl2"; } }
问:为什么非要调用接口来多此一举,而不直接调用实现类serviceImpl的bean来得简单明了呢?
直接使用serviceImpl的bean是可以的,这样加一层接口的原因:1.AOP程序的思想,给别人调用的接口,调用这只想知道方法和功能,而对于这个方法内部逻辑实现并不关心。2.降低各个模块间的关联,实现松耦合、程序分成、最主要的体现了继承只能单继承,接口却可以多实现