前言
在结束技术选择和模块化项目搭建之后。我们的spring boot项目就可以启动了。这时候就需要聊到微服务的一个重大作用,RPC调用
feign
feign接口调用是spring cloud下常用的RPC调用,自带Spring Cloud Ribbon 与 Spring Cloud Hystrix,提供了方便的负载均衡和断融,降级等服务
代码
pom文件
<dependencies>
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-openfeign</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-netflix-eureka-client</artifactId>
</dependency>
</dependencies>
Service文件
import com.chen.service.requestDTO.TestHelloRequestDTO;
import org.springframework.cloud.openfeign.FeignClient;
import org.springframework.web.bind.annotation.RequestMapping;
@FeignClient(name = "SERVICE-A")
@RequestMapping("/test")
//@FeignClient注解value必须与服务客户端application.yml配置中服务命名对应
public interface TestService {
@RequestMapping("/hello")
public String hello(TestHelloRequestDTO requestDTO);
@RequestMapping("/hi")
public String hi();
}
serviceimpl文件
import com.alibaba.fastjson.JSON;
import com.chen.service.requestDTO.TestHelloRequestDTO;
import com.chen.service.TestService;
import lombok.extern.slf4j.Slf4j;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RestController;
import javax.annotation.Resource;
@Slf4j
@RestController
public class TestServiceImpl implements TestService {
@Override
public String hello(@RequestBody TestHelloRequestDTO requestDTO) {
log.info(JSON.toJSONString(requestDTO));
return "hello";
}
@Override
public String hi() {
return "hi";
}
}
启动类注解
@SpringBootApplication
//Spring boot启动类
@EnableEurekaClient
//eureka 注册类
@RestController
//一个spring mvc控制
@EnableDiscoveryClient
//eureka 发现,如果单机应用,需要注释
//@EnableFeignClients(basePackages = {"com.chen.service"})
//feign包扫描
使用
坑点:
1.spring cloud feign接口无法调用。接口404或者调用后500,因为我把启动类的位置调整了。我们的默认包为com.chen.xxx,则启动类要放在com.chen下面
2.项目启动失败,原因有二,一是@FeignClient(name = “SERVICE-A”)注意括号中的name要和ymp配置中的服务命名对应,3.@RequestMapping这个注解我喜欢在类上注解一个,方法上也注解一个,这样可以方便的控制url的头部内容,但是feign有一个bug,无法做到,虽然这个bug之后修复了,但是老版本的feign不能再类上注解@RequestMapping,问题解决可以参考我上面发的链接
2020.8.19重新更新
重写:这部分的的时候我当初没有搞明白服务提供的原理,上面代码的@EnableFeignClients这个注释会扫描包。但是如果你这个应用只提供服务但是不调用其他服务。不需要开启这个注释,如果有其他应用的服务需要调用,才需要开启。而且这个扫描包的范围,只能扫描其他服务的包,千万不能扫描应用本身的包。不然就会提示方法重复。
核心说法就是使用@EnableFeignClients的扫描的时候如果把自己方法的feign接口也扫描的话会和feign的实现类互相冲突,导致spring认为有2个相同的url地址而启动失败。也有一些解决办法推荐重写spring的相关方法,就是我开始的链接上放的,但是其实是没有必要的,因为你调用微服务本来就不会调用到自身,所以本来@EnableFeignClients这个注解的扫描包就是要控制的颗粒度很细才行。不能直接一个大范围直接扫描。这样才会导致我之前遇到的3的问题
具体使用方法:
1.直接用serviceimpl+@Controller注解,即可调用对应的web接口返回结果
2.其他微服务,调用@Resource注解注入Service然后调用方法即可