首页 > 其他 > 详细

OpenFeign

时间:2020-08-14 17:35:43      阅读:59      评论:0      收藏:0      [点我收藏+]

OpenFeign

前面无论是基本调用,还是 Hystrix,我们实际上都是通过手动调用 RestTemplate 来实现远程调用的。使用 RestTemplate 存在一个问题:繁琐,每一个请求,参数不同,请求地址不同,返回数据类型不同,其他都是一样的,所以我们希望能够对请求进行简化。我们希望对请求进行简化,简化方案就是 OpenFeign。一开始这个组件不叫这个名字,一开始就叫 Feign,Netflix Feign,但是 Netflix 中的组件现在已经停止开源工作,OpenFeign 是 Spring Cloud 团队在 Netflix Feign 的基础上开发出来的声明式服务调用组
件。关于 OpenFeign 组件的 Issue:https://github.com/OpenFeign/feign/issues/373

HelloWorld

继续使用之前的Provider

@RestController
public class HelloController {
    @Value("${server.port}")
    Integer port;
    @GetMapping("/hello")
    public String hello() {
        return "hello javaboy:" + port;
    }

    @GetMapping("/hello2")
    public String hello2(String name){
        System.out.println(new Date()+">>>"+name);
        return "hello "+ name;
    }

    @PostMapping("/user1")
    public User addUser1(User user) {
        return user;
    }

    @PostMapping("/user2")
    public User addUser2(@RequestBody User user){
        return user;
    }
    @PutMapping("/user1")
    public void updateUser1(User user1){
        System.out.println(user1);
    }

    @PutMapping("/hello2")
    public void updateUser2(@RequestBody User user) {
        System.out.println(user);
    }

    @DeleteMapping("/user1")
    public void deleteUser1(Integer id) {
        System.out.println(id);
    }

    @DeleteMapping("/user2/{id}")
    public void deleteUser2(@PathVariable Integer id){
        System.out.println(id);
    }

    @GetMapping("/user3")
    public void getUserByName(@RequestHeader String name) throws UnsupportedEncodingException {
        System.out.println(URLDecoder.decode(name,"UTF-8"));
    }
}

新建一个Spring Boot模块,创建时,选择OpenFeign依赖,如下:
技术分享图片
项目创建成功后,在application.properties中进行配置,使项目注册到Eureka上:

spring.application.name=openfeign
server.port=4000
eureka.client.service-url.defaultZone=http://localhost:1111/eureka

接下来在启动类上添加注解,开启Feign的支持

@SpringBootApplication
@EnableFeignClients
public class OpenfeignApplication {

    public static void main(String[] args) {
        SpringApplication.run(OpenfeignApplication.class, args);
    }

}

接下来,定义HelloService接口,去使用OpenFeign:

@FeignClient("provider")
public interface HelloService {
    @GetMapping("/hello")
    String hello();//这里的方法名无所谓,随意取
}

最后调用HelloController中,调用HelloService进行测试

@RestController
public class HelloController {
    @Autowired
    HelloService helloService;

    @GetMapping("/hello")
    public String hello(){
        return helloService.hello();
    }
}

参数传递

和普通参数传递的区别:

  1. 参数一定要绑定参数名。
  2. 如果通过header来传递参数,一定要记得中文要转码。
    测试的服务接口,继续使用provider提供的接口。
    这里,我们主要在openfeign中添加调用接口即可:
@FeignClient("provider")
public interface HelloService {
    @GetMapping("/hello")
    String hello();//这里的方法名无所谓,随意取

    @GetMapping("/hello2")
    String hello2(@RequestParam("name") String name);

    @PostMapping("/user2")
    User addUser(@RequestBody User user);

    @DeleteMapping("/user2/{id}")
    void deleteUserById(@PathVariable("id") Integer id);

    @GetMapping("/user3")
    void getUserByName(@RequestHeader("name") String name);
}

注意,凡是key/value形式的参数,一定要标记参数的名称
HelloController中调用HelloService:

@RestController
public class HelloController {
    @Autowired
    HelloService helloService;

    @GetMapping("/hello")
    public String hello() throws UnsupportedEncodingException {
        String s = helloService.hello2("江南一点雨");
        System.out.println(s);
        User user = new User();
        user.setId(1);
        user.setName("javaboy");
        user.setPassword("123");
        User u = helloService.addUser(user);
        System.out.println(u);
        helloService.deleteUserById(1);
        helloService.getUserByName(URLEncoder.encode("江南一点雨","UTF-8"));
        return helloService.hello();
    }
}

注意放在header中的中文参数,一定要编码之后传递

继承特性

将provider和openfeign中公共的部分提取出来,一起使用:
我们新建一个Module,叫做hello-api,注意,由于这个模块要被其他模块所依赖,所以这个模块是一个maven项目,但是由于这个模块要用到 SpringMVC 的东西,因此在创建成功后,给这个模块添加一个 web 依赖,导入 SpringMVC 需要的一套东西。
项目创建成功后,首先添加依赖:

<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
<version>2.2.4.RELEASE</version>
</dependency>
<dependency>
<groupId>org.javaboy</groupId>
<artifactId>commons</artifactId>
<version>1.0-SNAPSHOT</version>
</dependency>

然后定义公共接口,就是provider 和 openfeign 中公共的部分:

public interface IUserService {
    @GetMapping("/hello")
    String hello();

    @GetMapping("/hello2")
    String hello2(@RequestParam("name") String name);

    @PostMapping("/user2")
    User addUser2(@RequestBody User user);

    @DeleteMapping("/user2/{id}")
    void deleteUser2(@PathVariable("id") Integer id);

    @GetMapping("/user3")
    void getUserByName(@RequestHeader("name") String name) throws UnsupportedEncodingException;
}

定义完成后,接下来,在provider和openfeign中,分别引用该模块

<dependency>
    <groupId>com.qwl</groupId>
    <artifactId>hello-api</artifactId>
    <version>0.0.1-SNAPSHOT</version>
</dependency>

添加成功之后,在 provider 中实现该接口:

@RestController
public class HelloController implements IUserService {
    @Value("${server.port}")
    Integer port;
    
    @Override
    public String hello() {
        return "hello javaboy:" + port;
    }

    @Override
    public String hello2(String name){
        System.out.println(new Date()+">>>"+name);
        return "hello "+ name;
    }

    @PostMapping("/user1")
    public User addUser1(User user) {
        return user;
    }

    @Override
    public User addUser2(@RequestBody User user){
        return user;
    }
    @PutMapping("/user1")
    public void updateUser1(User user1){
        System.out.println(user1);
    }

    @PutMapping("/hello2")
    public void updateUser2(@RequestBody User user) {
        System.out.println(user);
    }

    @DeleteMapping("/user1")
    public void deleteUser1(Integer id) {
        System.out.println(id);
    }
    @Override
    public void deleteUser2(@PathVariable Integer id){
        System.out.println(id);
    }

    @Override
    public void getUserByName(@RequestHeader String name) throws UnsupportedEncodingException {
        System.out.println(URLDecoder.decode(name,"UTF-8"));
    }
}

在 openfeign 中,定义接口继承自公共接口:

@FeignClient("provider")
public interface HelloService extends IUserService {
}

接下来,测试代码不变。
关于继承特性:

  1. 使用继承特性,代码简洁明了不易出错。服务端和消费端的代码统一,一改俱改,不易出错。这是
    优点也是缺点,这样会提高服务端和消费端的耦合度。
  2. 9.2 中所讲的参数传递,在使用了继承之后,依然不变,参数该怎么传还是怎么传。

日志

OpenFeign 中,我们可以通过配置日志,来查看整个请求的调用过程。日志级别一共分为四种:

  1. NONE:不开启日志,默认就是这个
  2. BASIC:记录请求方法、URL、响应状态码、执行时间
  3. HEADERS:在 BASIC 的基础上,加载请求/响应头
  4. FULL:在 HEADERS 基础上,再增加 body 以及请求元数据。
    四种级别,可以通过 Bean 来配置
@SpringBootApplication
@EnableFeignClients
public class OpenfeignApplication {

    public static void main(String[] args) {
        SpringApplication.run(OpenfeignApplication.class, args);
    }

    @Bean
    Logger.Level loggerLevel(){
        return Logger.Level.FULL;
    }

}

最后,在application.properties中开启日志级别

logging.level.com.qwl.openfeign.HelloService=debug

重启OpenFeign,进行测试

数据压缩

# 开启请求的数据压缩
feign.compression.request.enabled=true
# 开启响应的数据压缩
feign.compression.response.enabled=true
# 压缩的数据类型
feign.compression.request.mime-types=text/html,application/json
# 压缩的数据下限,2048 表示当要传输的数据大于 2048 时,才会进行数据压缩
feign.compression.request.min-request-size=2048

+Hystrix

Hystrix 中的容错、服务降级等功能,在 OpenFeign 中一样要使用。
首先定义服务降级的方法:

@Component
@RequestMapping("/javaboy")//防止请求地址重复
public class HelloServiceFallback implements HelloService{

    @Override
    public String hello() {
        return "error";
    }

    @Override
    public String hello2(String name) {
        return null;
    }

    @Override
    public User addUser2(User user) {
        return null;
    }

    @Override
    public void deleteUser2(Integer id) {

    }

    @Override
    public void getUserByName(String name) throws UnsupportedEncodingException {

    }
}

然后,在 HelloService 中配置这个服务降级类:

@FeignClient(value = "provider",fallback = HelloServiceFallback.class)
public interface HelloService extends IUserService {
}

然后,在application.properties中开启Hystrix

feign.hystrix.enabled=true

也可以通过自定义FallbackFactory来实现请求降级

public class HelloServiceFallbackFactory implements FallbackFactory<HelloService> {
    @Override
    public HelloService create(Throwable throwable) {
        return new HelloService() {
            @Override
            public String hello() {
                return "error";
            }

            @Override
            public String hello2(String name) {
                return "error2";
            }

            @Override
            public User addUser2(User user) {
                return null;
            }

            @Override
            public void deleteUser2(Integer id) {

            }

            @Override
            public void getUserByName(String name) throws UnsupportedEncodingException {

            }
        };
    }
}

HelloService中进行配置

@FeignClient(value = "provider",fallbackFactory = HelloServiceFallbackFactory.class)
public interface HelloService extends IUserService {
}

OpenFeign

原文:https://www.cnblogs.com/qiuwenli/p/13503181.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!