SpringCloud微服务实践之六 Feign最佳实践(抽取)

传统Feign面临的问题:
1、每个子项目都要写所要调用服务的pojo
2、每个子项目都要写所要调用服务的feign client客户端
优化思路:由提供服务服务的子项目统一归集代码,统一对外提供接口服务、Feign子项目统一管理服务远程调用、
将FeignClient抽取为独立的模块,并且把接口有关的POJO、默认Feign配置、Feign-client服务都放到这个模块中,提供给所有消费者使用。换言之,就是新建一个Feign子项目,将其他子项目的中feign相关代码(包括pojo、feign配置和feign-client)全部抽取剥离并集中于此,需要调用服务的子项目只需要依赖(引入)这个Feign子项目即可。
SpringCloud微服务实践之六 Feign最佳实践(抽取)_第1张图片

一、在父项目上点击鼠标右键选择new→Module:

过程同本专栏前述文章,略过不表,子项目的文件结构如下:
SpringCloud微服务实践之六 Feign最佳实践(抽取)_第2张图片

二、在pom.xml中引入依赖:

    <dependencies>
        <dependency>
            <groupId>org.springframework.cloud</groupId>
            <artifactId>spring-cloud-starter-openfeign</artifactId>
        </dependency>
    </dependencies>

三、创建feign配置类

package cn.it32.feign.config;

import feign.Logger;
import org.springframework.context.annotation.Bean;

public class DefaultFeignConfiguration {

    @Bean
    public Logger.Level logLevel(){
        return Logger.Level.BASIC;
    }
}

四、创建pojo及feign客户端,目前仅有userservice提供服务接口,因此只需创建user类的pojo 及相应feig客户端:

pojo:

package cn.it32.feign.pojo;

public class User {
    private Long id;
    private String username;
    private String address;

    public Long getId() {
        return id;
    }

    public void setId(Long id) {
        this.id = id;
    }

    public String getUsername() {
        return username;
    }

    public void setUsername(String username) {
        this.username = username;
    }

    public String getAddress() {
        return address;
    }

    public void setAddress(String address) {
        this.address = address;
    }
}

feign client:

package cn.it32.feign.clients;

import cn.it32.feign.pojo.User;
import org.springframework.cloud.openfeign.FeignClient;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PathVariable;

@FeignClient("userservice")
public interface UserClient {

    @GetMapping("/user/{id}")
    User findById(@PathVariable("id") Long id);
}

五、最后一步,改造原order服务,具体包括:

(一)删除转移到feign-api类中的代码,包括:

1、pojo类User.java
2、feign配置类DefaultFeignConfiguration.java
3、feign客户端类 UserClient.java

(二)引入feign-api类,同时变更user类的导入路径

1、修改pom.xm文件,增加feign-api依赖:

        <!-- 引入feign的api-->
        <dependency>
            <groupId>cn.it32</groupId>
            <artifactId>feign-api</artifactId>
            <version>1.0-SNAPSHOT</version>
        </dependency>
    </dependencies>

2、更新涉及到的user类、UserClient类、DefaultFeignConfiguration类的导入路径,具体包括:
(1)Order类
(2)OrderService类
(3)启动类OrderApplication.java
3、至此,完成改造。但此时重启测试,还会报错:找不到UserClient.
SpringCloud微服务实践之六 Feign最佳实践(抽取)_第3张图片
原因分析:这里就奇怪了,在启动类OrderApplication类中明明引入了feign-api子项目中的UserClient类,为何还会找不到呢?这是因为spring的特点所决定的:
从Order-service子项目编译没有报错,可以说明userclient这个类是确实可以被发现的;但无法注入成功,证明是这个类没有创建对象,也就是在Spring的容器里找不到它,所以注入失败。
那为什么这个接口没有对应的对象呢?改造前,之所以有这个对象,是因为UserClient接口中有这个注解:

@FeignClient(“userservice”)
public interface UserClient {

当Spring扫描到了这个注解,就会给这个接口创建对象,而改造后呢,现在它没有扫描到UserClient所在的这个包了,order-service默认的扫描包是启动类所在的包,启动类所在的包是:

package cn.it32.order;

而feign-api所在的包是:

package cn.it32.feign;

这是两个不同的包,所以扫描不到feign这个包,因此UserClient的对象就不会被创建,所以Spring的容器里就没有这个bean,就无法完成注入。这就是原因!我们就可以寻求相应的解决办法了。

当定义的FeignClient不在SpringBootApplication的扫描包范围时,这些FeignClient无法使用,有两种解决方式:
方式一:指定FeignClient所在包

@EnableFeignClients(basePackages = “cn.it32.feign”)

方式二:指定FeignClient字节码

@EnableFeignClients(clients = {UserClient.class})

第一种方式是对整个包进行扫描,第二种方式是精准扫描,这里我们一般使用第二种方式:
在order-service启动类OrderApplication.java的注解中添加:

……
@EnableFeignClients(clients = {UserClient.class},defaultConfiguration = DefaultFeignConfiguration.class)
public class OrderApplication {
……

改造完成,重启order-service测试:
SpringCloud微服务实践之六 Feign最佳实践(抽取)_第4张图片
没有报错,浏览器访问测试也正常:
在这里插入图片描述
至此,我们就完成了SpringCloud的Feign最佳实践的抽取方案了。

六、总结一下

这个最佳实践,我们是做了以下步骤:
第一步:新建了一个模块module(未来就是一个jar包),并在这个module中引入了feign相关的依赖;
第二步:我们把order-service原来自己写的feign客户端、实体类和配置都拿到了这个独立模块中,将来它就作为一个共享的jar包,任何需要userclient的服务,无需自己编写了,直接使用它就行;
第三步:在order-service的pom文件里引入feign-api依赖,这样就具备了上边这些功能,然后就可以把order-service里的这些代码删除了,原来的user相关的那些类就找不到了,就把order-service中相关的依赖包换成feign-api中的类;
第四步:最终重启时遇到一个问题:找不到UserClient了,原因是因为这个FeignClient所在的包是Feign的包,而我们现在默认的包是order包,这就导致它扫描不到了。基于两种解决方案,我们在@EnableFeignClient注解中指定所需Clent字节码文件。
结束。

你可能感兴趣的:(springcloud,spring,cloud,微服务,java)