:
服务注册:Nacos Clientapp开发定制公司会通过发送RESTapp开发定制公司请求的方式向Nacos Serverapp开发定制公司注册自己的服务,app开发定制公司提供自身的元数据,比如ip地址、app开发定制公司端口等信息。
Nacos Server接收到注册请求后,就会把这些元数据信息存储在一个双层的内存Map中。
服务心跳:在服务注册后,Nacos Client会维护一个定时心跳来持续通知Nacos Server,说明服务一直处于可用状态,防止被剔除。默认5s发送一次心跳。
服务同步:Nacos Server集群之间会互相同步服务实例,用来保证服务信息的一致性。
服务发现:服务消费者(Nacos Client)在调用服务提供者的服务时,会发送一个REST请求给Nacos Server,获取上面注册的服务清单,并且缓存在Nacos Client本地,同时会在Nacos Client本地开启一个定时任务定时拉取服务端最新的注册表信息更新到本地缓存
服务健康检查:Nacos Server会开启一个定时任务用来检查注册服务实例的健康情况,对于超过15s没有收到客户端心跳的实例会将它的healthy属性置为false(客户端服务发现时不会发现),如果某个实例超过30秒没有收到心跳,直接剔除该实例(被剔除的实例如果恢复发送心跳则会重新注册)
的作用
从拉取服务,对http客户端封装,根据负载均衡策略对服务调用进行负载均衡
自定义负载均衡策略可以实现:
1、金丝雀发布(灰度发布),要在nacos服务元数据配置版本号,在调用时过滤掉不同版本的服务即可
2、同集群调用,根据nacos中的clusterName进行过滤
注意:不同的服务可以设置不同的负载均衡策略
feign的作用
服务调用,维护管理rest调用地址,内置拦截器对http请求可以进行拦截
feign-ribbon-feign相关联的源码调用链
定义Feign客户端
- @FeignClient(name = "product", contextId = "ds-mgr-ds-base"
- , configuration = {DefaultFallback.class})
- public interface TestService {
-
- @RequestMapping(value = "/product/service", method = RequestMethod.GET)
- public String product();
- }
其中product服务有三个实例:
在启动类中对product方法进行调用:
- @SpringBootApplication
- @EnableFeignClients
- public class NacosConfigApplication {
-
- public static void main(String[] args) {
- ApplicationContext ac = SpringApplication.run(NacosConfigApplication.class, args);
- TestService testService = ac.getBean(TestService.class);
- System.out.println(testService.product());
- }
- }
首先TestService是通过factoryBean生成的bean,其实是一个代理对象
- class FeignClientFactoryBean
- implements FactoryBean<Object>, InitializingBean, ApplicationContextAware {
- ..........................................
-
- @Override
- public Object getObject() throws Exception {
- return getTarget();
- }
-
- /**
- * @param <T> the target type of the Feign client
- * @return a {@link Feign} client created with the specified data and the context
- * information
- */
- <T> T getTarget() {
- FeignContext context = this.applicationContext.getBean(FeignContext.class);
- Feign.Builder builder = feign(context);
-
- if (!StringUtils.hasText(this.url)) {
- if (!this.name.startsWith("http")) {
- this.url = "http://" + this.name;
- }
- else {
- this.url = this.name;
- }
- this.url += cleanPath();
- return (T) loadBalance(builder, context,
- new HardCodedTarget<>(this.type, this.name, this.url));
- }
- if (StringUtils.hasText(this.url) && !this.url.startsWith("http")) {
- this.url = "http://" + this.url;
- }
- String url = this.url + cleanPath();
- Client client = getOptional(context, Client.class);
- if (client != null) {
- if (client instanceof LoadBalancerFeignClient) {
- // not load balancing because we have a url,
- // but ribbon is on the classpath, so unwrap
- client = ((LoadBalancerFeignClient) client).getDelegate();
- }
- builder.client(client);
- }
- Targeter targeter = get(context, Targeter.class);
- return (T) targeter.target(this, builder, context,
- new HardCodedTarget<>(this.type, this.name, url));
- }
-
- ................................................................................
-
- }
此时 TestService testService = ac.getBean(TestService.class);已经获取了代理对象,在执行testService.product()中,会执行以下invoke方法
- static class FeignInvocationHandler implements InvocationHandler {
- .......................................
- return dispatch.get(method).invoke(args);
- }
点击invoke方法进去SynchronousMethodHandler中执行invoke方法
- @Override
- public Object invoke(Object[] argv) throws Throwable {
- RequestTemplate template = buildTemplateFromArgs.create(argv);
- Retryer retryer = this.retryer.clone();
- while (true) {
- ..............
- return executeAndDecode(template);
- ..................
- }
- }
在执行executeAndDecode方法的时候,会调用client去执行rest调用,这个client此时是LoadBalancerFeignClient
- Object executeAndDecode(RequestTemplate template) throws Throwable {
- ...............................
- response = client.execute(request, options);
- ...................................
- }
这个类就是feign和ribbon的桥梁
-
- public class LoadBalancerFeignClient implements Client {
-
- .............................
- @Override
- public Response execute(Request request, Request.Options options) throws IOException {
- try {
- URI asUri = URI.create(request.url());
- String clientName = asUri.getHost();
- URI uriWithoutHost = cleanUrl(request.url(), clientName);
- FeignLoadBalancer.RibbonRequest ribbonRequest = new FeignLoadBalancer.RibbonRequest(
- this.delegate, request, uriWithoutHost);
-
- IClientConfig requestConfig = getClientConfig(options, clientName);
- return lbClient(clientName)
- .executeWithLoadBalancer(ribbonRequest, requestConfig).toResponse();
- }
- catch (ClientException e) {
- IOException io = findIOException(e);
- if (io != null) {
- throw io;
- }
- throw new RuntimeException(e);
- }
- }
-
- IClientConfig getClientConfig(Request.Options options, String clientName) {
- IClientConfig requestConfig;
- if (options == DEFAULT_OPTIONS) {
- requestConfig = this.clientFactory.getClientConfig(clientName);
- }
- else {
- requestConfig = new FeignOptionsClientConfig(options);
- }
- return requestConfig;
- }
-
- ....................................
- }
requestConfig = this.clientFactory.getClientConfig(clientName);
- @Override
- public <C> C getInstance(String name, Class<C> type) {
- C instance = super.getInstance(name, type);
- if (instance != null) {
- return instance;
- }
- IClientConfig config = getInstance(name, IClientConfig.class);
- return instantiateWithConfig(getContext(name), type, config);
- }
C instance = super.getInstance(name, type);
- public <T> T getInstance(String name, Class<T> type) {
- AnnotationConfigApplicationContext context = getContext(name);
- if (BeanFactoryUtils.beanNamesForTypeIncludingAncestors(context,
- type).length > 0) {
- return context.getBean(type);
- }
- return null;
- }
- protected AnnotationConfigApplicationContext getContext(String name) {
- if (!this.contexts.containsKey(name)) {
- synchronized (this.contexts) {
- if (!this.contexts.containsKey(name)) {
- this.contexts.put(name, createContext(name));
- }
- }
- }
- return this.contexts.get(name);
- }
- protected AnnotationConfigApplicationContext createContext(String name) {
- AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext();
- if (this.configurations.containsKey(name)) {
- for (Class<?> configuration : this.configurations.get(name)
- .getConfiguration()) {
- context.register(configuration);
- }
- }
- for (Map.Entry<String, C> entry : this.configurations.entrySet()) {
- if (entry.getKey().startsWith("default.")) {
- for (Class<?> configuration : entry.getValue().getConfiguration()) {
- context.register(configuration);
- }
- }
- }
- context.register(PropertyPlaceholderAutoConfiguration.class,
- this.defaultConfigType);
- context.getEnvironment().getPropertySources().addFirst(new MapPropertySource(
- this.propertySourceName,
- Collections.<String, Object>singletonMap(this.propertyName, name)));
- if (this.parent != null) {
- // Uses Environment from parent as well as beans
- context.setParent(this.parent);
- // jdk11 issue
- // https://github.com/spring-cloud/spring-cloud-netflix/issues/3101
- context.setClassLoader(this.parent.getClassLoader());
- }
- context.setDisplayName(generateDisplayName(name));
- context.refresh();
- return context;
- }
我觉得这个方法是ribbion中最核心的方法,就是为每一个服务创建一个ioc容器上下文,这样的做的目的就是让每一个服务都能够自定义自己的负载均衡策略,还有调用不同注册中心的服务列表
他会把RibbonClientConfiguration这个配置类加载进容器
在RibbonClientConfiguration类中有两个核心的bean
- @Bean
- @ConditionalOnMissingBean
- public IRule ribbonRule(IClientConfig config) {
- if (this.propertiesFactory.isSet(IRule.class, name)) {
- return this.propertiesFactory.get(IRule.class, config, name);
- }
- ZoneAvoidanceRule rule = new ZoneAvoidanceRule();
- rule.initWithNiwsConfig(config);
- return rule;
- }
- @Bean
- @ConditionalOnMissingBean
- public ILoadBalancer ribbonLoadBalancer(IClientConfig config,
- ServerList<Server> serverList, ServerListFilter<Server> serverListFilter,
- IRule rule, IPing ping, ServerListUpdater serverListUpdater) {
- if (this.propertiesFactory.isSet(ILoadBalancer.class, name)) {
- return this.propertiesFactory.get(ILoadBalancer.class, config, name);
- }
- return new ZoneAwareLoadBalancer<>(config, rule, ping, serverList,
- serverListFilter, serverListUpdater);
- }
ribbonRule是配置负载均衡策略的,ribbonLoadBalancer主要是去nocas中获取服务列表的
接下来看ribbonLoadBalancer方法:
- public ZoneAwareLoadBalancer(IClientConfig clientConfig, IRule rule,
- IPing ping, ServerList<T> serverList, ServerListFilter<T> filter,
- ServerListUpdater serverListUpdater) {
- super(clientConfig, rule, ping, serverList, filter, serverListUpdater);
- }
- public DynamicServerListLoadBalancer(IClientConfig clientConfig, IRule rule, IPing ping,
- ServerList<T> serverList, ServerListFilter<T> filter,
- ServerListUpdater serverListUpdater) {
- super(clientConfig, rule, ping);
- this.serverListImpl = serverList;
- this.filter = filter;
- this.serverListUpdater = serverListUpdater;
- if (filter instanceof AbstractServerListFilter) {
- ((AbstractServerListFilter) filter).setLoadBalancerStats(getLoadBalancerStats());
- }
- restOfInit(clientConfig);
- }
- void restOfInit(IClientConfig clientConfig) {
- boolean primeConnection = this.isEnablePrimingConnections();
- // turn this off to avoid duplicated asynchronous priming done in BaseLoadBalancer.setServerList()
- this.setEnablePrimingConnections(false);
- enableAndInitLearnNewServersFeature();
-
- updateListOfServers();
- if (primeConnection && this.getPrimeConnections() != null) {
- this.getPrimeConnections()
- .primeConnections(getReachableServers());
- }
- this.setEnablePrimingConnections(primeConnection);
- LOGGER.info("DynamicServerListLoadBalancer for client {} initialized: {}", clientConfig.getClientName(), this.toString());
- }
- @VisibleForTesting
- public void updateListOfServers() {
- List<T> servers = new ArrayList<T>();
- if (serverListImpl != null) {
- servers = serverListImpl.getUpdatedListOfServers();
- LOGGER.debug("List of Servers for {} obtained from Discovery client: {}",
- getIdentifier(), servers);
-
- if (filter != null) {
- servers = filter.getFilteredListOfServers(servers);
- LOGGER.debug("Filtered List of Servers for {} obtained from Discovery client: {}",
- getIdentifier(), servers);
- }
- }
- updateAllServerList(servers);
- }
- @Override
- public List<NacosServer> getUpdatedListOfServers() {
- return getServers();
- }
接下来就是去获取nacos服务,则程序中可以注入ILoadBalancer获取服务列表