likes
comments
collection
share

Feign服务通信的实现

作者站长头像
站长
· 阅读数 28

feign的用法

还是通过spring官网,首先创建两个项目,demoA和demoB,引入openfeign;我把demoA的端口设置为9000,demoB的端口设置为9001。

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

@EnableFeignClients的使用

在启动类上添加注解,并且把basePackages指向feign的包

package com.example.demoA;

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cloud.openfeign.EnableFeignClients;

@SpringBootApplication
@EnableFeignClients(basePackages="com.example.demoA.feign")
public class DemoAApplication {

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

}

@FeignClient的使用

然后创建一个接口,添加@FeignClient注解

package com.example.demoA.feign;

import org.springframework.cloud.openfeign.FeignClient;
import org.springframework.web.bind.annotation.GetMapping;

@FeignClient(name = "remoteTestB", url = "http://localhost:9001")
public interface RemoteTestB {

    @GetMapping("/testB/test1")
    String testB();
}

就能实现两个服务互相调用了,以上是demoA调用demoB的示例,检验一下

package com.example.demoA.controller;

import com.example.demoA.feign.RemoteTestB;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;


@RestController
@RequestMapping("/testA")
public class TestA {

    @Autowired
    RemoteTestB remoteTestB;

    @GetMapping("/test1")
    public Object test1() {
        return "this is test1 of testA";
    }

    @GetMapping("/test2")
    public Object test2() {
        return remoteTestB.testB();
    }


}

当我调用demoA的test2的方法时,出现

Feign服务通信的实现

说明demoA成功调用了demoB,此时,我们已经会使用feign了。

如何实现服务间通信

那么feign是怎么实现服务间通信的呢?通过这个调用方式,我们可以大概反推出来一个过程:

1.启动类里加上@EnableFeignClients后,在启动时,被@FeignClient注解过的类会注册到容器中。

2.被@FeignClient注解的是接口,需要用到动态代理方式补全通信逻辑。

所以现在我们拆解这两步,为了搞懂这两个注解的逻辑,我们新建一个项目仿写一个逻辑简单的feign,虽然逻辑比较简单,过程比较粗暴,不过正适合新手学习。

自定义注解

第一步当然是创建两个注解

@EnableFeignClients中的逻辑就是:利用@Import和ImportBeanDefinitionRegistrar配合,重写一个registerBeanDefinitions方法,在这个方法中实现自定义的bean注册逻辑。

这段源码研究好几天,只提炼出最核心的两个步骤:

1.扫描文件,把带有MyFeignClient的接口都找到;

2.把带有MyFeignClient的接口生成一个代理类,注册到容器中;

其他一概没看懂,不过在此挖个坑,以后详细写一篇spring自定义注册。

package com.example.myopenfeign;


import org.springframework.context.annotation.Import;

import java.lang.annotation.*;

@Import(MyFeignRegister.class)
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
@Documented
public @interface MyEnableFeignClients {
}
package com.example.myopenfeign;

import org.springframework.beans.factory.annotation.AnnotatedBeanDefinition;
import org.springframework.beans.factory.config.BeanDefinition;
import org.springframework.beans.factory.support.BeanDefinitionBuilder;
import org.springframework.beans.factory.support.BeanDefinitionRegistry;
import org.springframework.context.EnvironmentAware;
import org.springframework.context.ResourceLoaderAware;
import org.springframework.context.annotation.ClassPathScanningCandidateComponentProvider;
import org.springframework.context.annotation.ImportBeanDefinitionRegistrar;
import org.springframework.core.env.Environment;
import org.springframework.core.io.ResourceLoader;
import org.springframework.core.type.AnnotationMetadata;
import org.springframework.core.type.filter.AnnotationTypeFilter;
import org.springframework.util.Assert;

import java.util.LinkedHashSet;
import java.util.Map;


public class MyFeignRegister implements ImportBeanDefinitionRegistrar, ResourceLoaderAware, EnvironmentAware {

    private ResourceLoader resourceLoader;

    private Environment environment;

    @Override
    public void setEnvironment(Environment environment) {
        this.environment = environment;
    }

    @Override
    public void setResourceLoader(ResourceLoader resourceLoader) {
        this.resourceLoader = resourceLoader;
    }

    @Override
    public void registerBeanDefinitions(AnnotationMetadata importingClassMetadata, BeanDefinitionRegistry registry) {
        LinkedHashSet<BeanDefinition> candidateComponents = new LinkedHashSet<>();
        //扫描文件,把带有MyFeignClient的接口都找到
        ClassPathScanningCandidateComponentProvider scanner = getScanner();
        scanner.setResourceLoader(resourceLoader);
        scanner.addIncludeFilter(new AnnotationTypeFilter(MyFeignClient.class));
        candidateComponents.addAll(scanner.findCandidateComponents("com.example"));
        for (BeanDefinition candidateComponent : candidateComponents) {
            if (candidateComponent instanceof AnnotatedBeanDefinition beanDefinition) {
                AnnotationMetadata annotationMetadata = beanDefinition.getMetadata();
                Assert.isTrue(annotationMetadata.isInterface(), "@FeignClient can only be specified on an interface");

                Map<String, Object> attributes = annotationMetadata
                        .getAnnotationAttributes(MyFeignClient.class.getCanonicalName());

                String name = attributes.get("value").toString();
                String beanClassName = candidateComponent.getBeanClassName();

                //把带有MyFeignClient的接口生成一个代理类,注册到容器中
                try {
                    Class<?> aClass1 = Class.forName(beanClassName);
                    BeanDefinitionBuilder beanDefinitionBuilder = BeanDefinitionBuilder.genericBeanDefinition(MyFeignFactoryBean.class);
                    beanDefinitionBuilder.addConstructorArgValue(aClass1);
                    registry.registerBeanDefinition(name, beanDefinitionBuilder.getBeanDefinition());
                } catch (ClassNotFoundException e) {
                    throw new RuntimeException(e);
                }

            }
        }
    }

    protected ClassPathScanningCandidateComponentProvider getScanner() {
        return new ClassPathScanningCandidateComponentProvider(false, this.environment) {
            @Override
            protected boolean isCandidateComponent(AnnotatedBeanDefinition beanDefinition) {
                boolean isCandidate = false;
                if (beanDefinition.getMetadata().isIndependent()) {
                    if (!beanDefinition.getMetadata().isAnnotation()) {
                        isCandidate = true;
                    }
                }
                return isCandidate;
            }
        };
    }

}
package com.example.myopenfeign;

import java.lang.annotation.*;

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Inherited
public @interface MyFeignClient {

    String value() default "";

    String url() default "";
}

代理生成bean

package com.example.myopenfeign;

import org.springframework.beans.factory.FactoryBean;

import java.lang.reflect.Proxy;

public class MyFeignFactoryBean <T> implements FactoryBean {

    private Class<T> interfaceClass;

    public MyFeignFactoryBean(Class<T> interfaceClass){
        this.interfaceClass = interfaceClass;
    }

    @Override
    public T getObject() throws Exception {
        return (T) Proxy.newProxyInstance(
                interfaceClass.getClassLoader(), // 目标类的类加载器
                new Class[]{interfaceClass},  // 代理需要实现的接口,可指定多个
                new MyFeignInvocationHandler()
        );
    }

    @Override
    public Class<?> getObjectType() {
        return interfaceClass;
    }
}

实现真正的代理方法逻辑

package com.example.myopenfeign;

import cn.hutool.http.HttpUtil;
import org.springframework.web.bind.annotation.GetMapping;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;

public class MyFeignInvocationHandler implements InvocationHandler {

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) {
        System.out.println("before method " + method.getName());
        Class<?> declaringClass = method.getDeclaringClass();
        MyFeignClient annotation = declaringClass.getAnnotation(MyFeignClient.class);
        String url = annotation.url();
        GetMapping annotation1 = method.getAnnotation(GetMapping.class);
        String[] value1 = annotation1.value();
        String s = HttpUtil.get(url + value1[0]);
        return s;
    }
}

以上就完成了简单的feign逻辑。

测试

首先在启动类里加上我们自定义的@MyEnableFeignClients

package com.example.myopenfeign;

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.context.ConfigurableApplicationContext;

@SpringBootApplication
@MyEnableFeignClients
public class MyOpenFeignApplication {

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

}

然后创建一个用于服务通信的接口,还是指向刚刚新建的demoA服务

package com.example.myopenfeign.test;

import com.example.myopenfeign.MyFeignClient;
import org.springframework.web.bind.annotation.GetMapping;

@MyFeignClient(value = "remote",url = "http://localhost:9000")
public interface Remote {

    @GetMapping("/testA/test1")
    Object getA();
}

最后使用这个接口

package com.example.myopenfeign.test;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

@RestController
@RequestMapping("/myFeignTest")
public class TestController {

    @Autowired
    Remote remote;

    @GetMapping("/test1")
    public Object test(){
        Object a = remote.getA();
        return a;
    }
}

启动后调用 http://localhost:8080/myFeignTest/test1

结果成功!

Feign服务通信的实现

以上就是实现feign的过程,掌握这个过程,我们可以照此模式扩展写出自定义的bean注册逻辑。

转载自:https://juejin.cn/post/7360976761838714880
评论
请登录