亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Spring Cloud Ribbon的踩坑記錄與原理詳析

 更新時(shí)間:2018年10月21日 09:07:21   作者:aCoder2013  
這篇文章主要給大家介紹了關(guān)于Spring Cloud Ribbon踩坑記錄與原理的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧

簡(jiǎn)介

Spring Cloud Ribbon 是一個(gè)基于Http和TCP的客服端負(fù)載均衡工具,它是基于Netflix Ribbon實(shí)現(xiàn)的。它不像服務(wù)注冊(cè)中心、配置中心、API網(wǎng)關(guān)那樣獨(dú)立部署,但是它幾乎存在于每個(gè)微服務(wù)的基礎(chǔ)設(shè)施中。包括前面的提供的聲明式服務(wù)調(diào)用也是基于該Ribbon實(shí)現(xiàn)的。理解Ribbon對(duì)于我們使用Spring Cloud來(lái)講非常的重要,因?yàn)樨?fù)載均衡是對(duì)系統(tǒng)的高可用、網(wǎng)絡(luò)壓力的緩解和處理能力擴(kuò)容的重要手段之一。在上節(jié)的例子中,我們采用了聲明式的方式來(lái)實(shí)現(xiàn)負(fù)載均衡。實(shí)際上,內(nèi)部調(diào)用維護(hù)了一個(gè)RestTemplate對(duì)象,該對(duì)象會(huì)使用Ribbon的自動(dòng)化配置,同時(shí)通過(guò)@LoadBalanced開(kāi)啟客戶(hù)端負(fù)載均衡。其實(shí)RestTemplate是Spring自己提供的對(duì)象,不是新的內(nèi)容。讀者不知道RestTemplate可以查看相關(guān)的文檔。

現(xiàn)象

前兩天碰到一個(gè)ribbon相關(guān)的問(wèn)題,覺(jué)得值得記錄一下。表象是對(duì)外的接口返回內(nèi)部異常,這個(gè)是封裝的統(tǒng)

一錯(cuò)誤信息,Spring的異常處理器catch到未捕獲異常統(tǒng)一返回的信息。因此到日志平臺(tái)查看實(shí)際的異常:

org.springframework.web.client.HttpClientErrorException: 404 null

這里介紹一下背景,出現(xiàn)問(wèn)題的開(kāi)放網(wǎng)關(guān),做點(diǎn)事情說(shuō)白了就是轉(zhuǎn)發(fā)對(duì)應(yīng)的請(qǐng)求給后端的服務(wù)。這里用到了ribbon去做服務(wù)負(fù)載均衡、eureka負(fù)責(zé)服務(wù)發(fā)現(xiàn)。

這里出現(xiàn)404,首先看了下請(qǐng)求的url以及對(duì)應(yīng)的參數(shù),都沒(méi)有發(fā)現(xiàn)問(wèn)題,對(duì)應(yīng)的后端服務(wù)也沒(méi)有收到請(qǐng)求。這就比較詭異了,開(kāi)始懷疑是ribbon或者Eureka的緩存導(dǎo)致請(qǐng)求到了錯(cuò)誤的ip或端口,但由于日志中打印的是Eureka的serviceId而不是實(shí)際的ip:port,因此先加了個(gè)日志:

@Slf4j
public class CustomHttpRequestInterceptor implements ClientHttpRequestInterceptor {

 @Override
 public ClientHttpResponse intercept(HttpRequest request, byte[] body, ClientHttpRequestExecution execution) throws IOException {
  log.info("Request , url:{},method:{}.", request.getURI(), request.getMethod());
  return execution.execute(request, body);
 }
}

這里是通過(guò)給RestTemplate添加攔截器的方式,但要注意,ribbon也是通過(guò)給RestTemplate添加攔截器實(shí)現(xiàn)的解析serviceId到實(shí)際的ip:port,因此需要注意下優(yōu)先級(jí)添加到ribbon的 LoadBalancerInterceptor 之后,我這里是通過(guò)Spring的初始化完成事件的回調(diào)中添加的,另外也添加了另一條日志,在catch到這個(gè)異常的時(shí)候,利用Eureka的 DiscoveryClient#getInstances 獲取到當(dāng)前的實(shí)例信息。

之后在測(cè)試環(huán)境中復(fù)現(xiàn)了這個(gè)問(wèn)題,看了下日志,eurek中緩存的實(shí)例信息是對(duì)的,但是實(shí)際調(diào)用的確實(shí)另外一個(gè)服務(wù)的地址,從而導(dǎo)致了接口404。

源碼解析

從上述的信息中可以知道,問(wèn)題出在ribbon中,具體的原因后面會(huì)說(shuō),這里先講一下Spring Cloud Ribbon的初始化流程。

@Configuration
@ConditionalOnClass({ IClient.class, RestTemplate.class, AsyncRestTemplate.class, Ribbon.class})
@RibbonClients
@AutoConfigureAfter(name = "org.springframework.cloud.netflix.eureka.EurekaClientAutoConfiguration")
@AutoConfigureBefore({LoadBalancerAutoConfiguration.class, AsyncLoadBalancerAutoConfiguration.class})
@EnableConfigurationProperties({RibbonEagerLoadProperties.class, ServerIntrospectorProperties.class})
public class RibbonAutoConfiguration {
}

注意這個(gè)注解 @RibbonClients , 如果想要覆蓋Spring Cloud提供的默認(rèn)Ribbon配置就可以使用這個(gè)注解,最終的解析類(lèi)是:

public class RibbonClientConfigurationRegistrar implements ImportBeanDefinitionRegistrar {

 @Override
 public void registerBeanDefinitions(AnnotationMetadata metadata,
 BeanDefinitionRegistry registry) {
 Map<String, Object> attrs = metadata.getAnnotationAttributes(
 RibbonClients.class.getName(), true);
 if (attrs != null && attrs.containsKey("value")) {
 AnnotationAttributes[] clients = (AnnotationAttributes[]) attrs.get("value");
 for (AnnotationAttributes client : clients) {
 registerClientConfiguration(registry, getClientName(client),
  client.get("configuration"));
 }
 }
 if (attrs != null && attrs.containsKey("defaultConfiguration")) {
 String name;
 if (metadata.hasEnclosingClass()) {
 name = "default." + metadata.getEnclosingClassName();
 } else {
 name = "default." + metadata.getClassName();
 }
 registerClientConfiguration(registry, name,
  attrs.get("defaultConfiguration"));
 }
 Map<String, Object> client = metadata.getAnnotationAttributes(
 RibbonClient.class.getName(), true);
 String name = getClientName(client);
 if (name != null) {
 registerClientConfiguration(registry, name, client.get("configuration"));
 }
 }

 private String getClientName(Map<String, Object> client) {
 if (client == null) {
 return null;
 }
 String value = (String) client.get("value");
 if (!StringUtils.hasText(value)) {
 value = (String) client.get("name");
 }
 if (StringUtils.hasText(value)) {
 return value;
 }
 throw new IllegalStateException(
 "Either 'name' or 'value' must be provided in @RibbonClient");
 }

 private void registerClientConfiguration(BeanDefinitionRegistry registry,
 Object name, Object configuration) {
 BeanDefinitionBuilder builder = BeanDefinitionBuilder
 .genericBeanDefinition(RibbonClientSpecification.class);
 builder.addConstructorArgValue(name);
 builder.addConstructorArgValue(configuration);
 registry.registerBeanDefinition(name + ".RibbonClientSpecification",
 builder.getBeanDefinition());
 }
}

atrrs包含defaultConfiguration,因此會(huì)注冊(cè)RibbonClientSpecification類(lèi)型的bean,注意名稱(chēng)以 default. 開(kāi)頭,類(lèi)型是RibbonAutoConfiguration,注意上面說(shuō)的RibbonAutoConfiguration被@RibbonClients修飾。

然后再回到上面的源碼:

public class RibbonAutoConfiguration {
  

  //上文中會(huì)解析被@RibbonClients注解修飾的類(lèi),然后注冊(cè)類(lèi)型為RibbonClientSpecification的bean。
  //主要有兩個(gè): RibbonAutoConfiguration、RibbonEurekaAutoConfiguration
 @Autowired(required = false)
 private List<RibbonClientSpecification> configurations = new ArrayList<>();
  
 @Bean
 public SpringClientFactory springClientFactory() {
    //初始化SpringClientFactory,并將上面的配置注入進(jìn)去,這段很重要。
 SpringClientFactory factory = new SpringClientFactory();
 factory.setConfigurations(this.configurations);
 return factory;
 }
   //其他的都是提供一些默認(rèn)的bean配置

 @Bean
 @ConditionalOnMissingBean(LoadBalancerClient.class)
 public LoadBalancerClient loadBalancerClient() {
 return new RibbonLoadBalancerClient(springClientFactory());
 }

 @Bean
 @ConditionalOnClass(name = "org.springframework.retry.support.RetryTemplate")
 @ConditionalOnMissingBean
 public LoadBalancedRetryPolicyFactory loadBalancedRetryPolicyFactory(SpringClientFactory clientFactory) {
 return new RibbonLoadBalancedRetryPolicyFactory(clientFactory);
 }

 @Bean
 @ConditionalOnMissingClass(value = "org.springframework.retry.support.RetryTemplate")
 @ConditionalOnMissingBean
 public LoadBalancedRetryPolicyFactory neverRetryPolicyFactory() {
 return new LoadBalancedRetryPolicyFactory.NeverRetryFactory();
 }

 @Bean
 @ConditionalOnClass(name = "org.springframework.retry.support.RetryTemplate")
 @ConditionalOnMissingBean
 public LoadBalancedBackOffPolicyFactory loadBalancedBackoffPolicyFactory() {
 return new LoadBalancedBackOffPolicyFactory.NoBackOffPolicyFactory();
 }

 @Bean
 @ConditionalOnClass(name = "org.springframework.retry.support.RetryTemplate")
 @ConditionalOnMissingBean
 public LoadBalancedRetryListenerFactory loadBalancedRetryListenerFactory() {
 return new LoadBalancedRetryListenerFactory.DefaultRetryListenerFactory();
 }

 @Bean
 @ConditionalOnMissingBean
 public PropertiesFactory propertiesFactory() {
 return new PropertiesFactory();
 }
 
 @Bean
 @ConditionalOnProperty(value = "ribbon.eager-load.enabled", matchIfMissing = false)
 public RibbonApplicationContextInitializer ribbonApplicationContextInitializer() {
 return new RibbonApplicationContextInitializer(springClientFactory(),
 ribbonEagerLoadProperties.getClients());
 }

 @Configuration
 @ConditionalOnClass(HttpRequest.class)
 @ConditionalOnRibbonRestClient
 protected static class RibbonClientConfig {

 @Autowired
 private SpringClientFactory springClientFactory;

 @Bean
 public RestTemplateCustomizer restTemplateCustomizer(
 final RibbonClientHttpRequestFactory ribbonClientHttpRequestFactory) {
 return new RestTemplateCustomizer() {
 @Override
 public void customize(RestTemplate restTemplate) {
  restTemplate.setRequestFactory(ribbonClientHttpRequestFactory);
 }
 };
 }

 @Bean
 public RibbonClientHttpRequestFactory ribbonClientHttpRequestFactory() {
 return new RibbonClientHttpRequestFactory(this.springClientFactory);
 }
 }

 //TODO: support for autoconfiguring restemplate to use apache http client or okhttp

 @Target({ ElementType.TYPE, ElementType.METHOD })
 @Retention(RetentionPolicy.RUNTIME)
 @Documented
 @Conditional(OnRibbonRestClientCondition.class)
 @interface ConditionalOnRibbonRestClient { }

 private static class OnRibbonRestClientCondition extends AnyNestedCondition {
 public OnRibbonRestClientCondition() {
 super(ConfigurationPhase.REGISTER_BEAN);
 }

 @Deprecated //remove in Edgware"
 @ConditionalOnProperty("ribbon.http.client.enabled")
 static class ZuulProperty {}

 @ConditionalOnProperty("ribbon.restclient.enabled")
 static class RibbonProperty {}
 }
}

注意這里的SpringClientFactory, ribbon默認(rèn)情況下,每個(gè)eureka的serviceId(服務(wù)),都會(huì)分配自己獨(dú)立的Spring的上下文,即ApplicationContext, 然后這個(gè)上下文中包含了必要的一些bean,比如: ILoadBalancer 、 ServerListFilter 等。而Spring Cloud默認(rèn)是使用RestTemplate封裝了ribbon的調(diào)用,核心是通過(guò)一個(gè)攔截器:

@Bean
 @ConditionalOnMissingBean
 public RestTemplateCustomizer restTemplateCustomizer(
 final LoadBalancerInterceptor loadBalancerInterceptor) {
 return new RestTemplateCustomizer() {
 @Override
 public void customize(RestTemplate restTemplate) {
  List<ClientHttpRequestInterceptor> list = new ArrayList<>(
  restTemplate.getInterceptors());
  list.add(loadBalancerInterceptor);
  restTemplate.setInterceptors(list);
 }
 };
 }

因此核心是通過(guò)這個(gè)攔截器實(shí)現(xiàn)的負(fù)載均衡:

public class LoadBalancerInterceptor implements ClientHttpRequestInterceptor {

 private LoadBalancerClient loadBalancer;
 private LoadBalancerRequestFactory requestFactory;

 @Override
 public ClientHttpResponse intercept(final HttpRequest request, final byte[] body,
 final ClientHttpRequestExecution execution) throws IOException {
 final URI originalUri = request.getURI(); //這里傳入的url是解析之前的,即http://serviceId/服務(wù)地址的形式
 String serviceName = originalUri.getHost(); //解析拿到對(duì)應(yīng)的serviceId
 Assert.state(serviceName != null, "Request URI does not contain a valid hostname: " + originalUri);
 return this.loadBalancer.execute(serviceName, requestFactory.createRequest(request, body, execution));
 }
}

然后將請(qǐng)求轉(zhuǎn)發(fā)給LoadBalancerClient:

public class RibbonLoadBalancerClient implements LoadBalancerClient {
@Override
 public <T> T execute(String serviceId, LoadBalancerRequest<T> request) throws IOException {
 ILoadBalancer loadBalancer = getLoadBalancer(serviceId); //獲取對(duì)應(yīng)的LoadBalancer
 Server server = getServer(loadBalancer); //獲取服務(wù)器,這里會(huì)執(zhí)行對(duì)應(yīng)的分流策略,比如輪訓(xùn)
    //、隨機(jī)等
 if (server == null) {
 throw new IllegalStateException("No instances available for " + serviceId);
 }
 RibbonServer ribbonServer = new RibbonServer(serviceId, server, isSecure(server,
 serviceId), serverIntrospector(serviceId).getMetadata(server));

 return execute(serviceId, ribbonServer, request);
 }
}

而這里的LoadBalancer是通過(guò)上文中提到的SpringClientFactory獲取到的,這里會(huì)初始化一個(gè)新的Spring上下文,然后將Ribbon默認(rèn)的配置類(lèi),比如說(shuō): RibbonAutoConfiguration 、 RibbonEurekaAutoConfiguration 等添加進(jìn)去, 然后將當(dāng)前spring的上下文設(shè)置為parent,再調(diào)用refresh方法進(jìn)行初始化。

public class SpringClientFactory extends NamedContextFactory<RibbonClientSpecification> {
 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);
 }
 context.refresh();
 return context;
 }
}

最核心的就在這一段,也就是說(shuō)對(duì)于每一個(gè)不同的serviceId來(lái)說(shuō),都擁有一個(gè)獨(dú)立的spring上下文,并且在第一次調(diào)用這個(gè)服務(wù)的時(shí)候,會(huì)初始化ribbon相關(guān)的所有bean, 如果不存在 才回去父context中去找。

再回到上文中根據(jù)分流策略獲取實(shí)際的ip:port的代碼段:

public class RibbonLoadBalancerClient implements LoadBalancerClient {
@Override
 public <T> T execute(String serviceId, LoadBalancerRequest<T> request) throws IOException {
 ILoadBalancer loadBalancer = getLoadBalancer(serviceId); //獲取對(duì)應(yīng)的LoadBalancer
 Server server = getServer(loadBalancer); //獲取服務(wù)器,這里會(huì)執(zhí)行對(duì)應(yīng)的分流策略,比如輪訓(xùn)
    //、隨機(jī)等
 if (server == null) {
 throw new IllegalStateException("No instances available for " + serviceId);
 }
 RibbonServer ribbonServer = new RibbonServer(serviceId, server, isSecure(server,
 serviceId), serverIntrospector(serviceId).getMetadata(server));

 return execute(serviceId, ribbonServer, request);
 }
}
 protected Server getServer(ILoadBalancer loadBalancer) {
 if (loadBalancer == null) {
 return null;
 }
    // 選擇對(duì)應(yīng)的服務(wù)器
 return loadBalancer.chooseServer("default"); // TODO: better handling of key
 }
public class ZoneAwareLoadBalancer<T extends Server> extends DynamicServerListLoadBalancer<T> {
@Override
 public Server chooseServer(Object key) {
  if (!ENABLED.get() || getLoadBalancerStats().getAvailableZones().size() <= 1) {
   logger.debug("Zone aware logic disabled or there is only one zone");
   return super.chooseServer(key); //默認(rèn)不配置可用區(qū),走的是這段
  }
  Server server = null;
  try {
   LoadBalancerStats lbStats = getLoadBalancerStats();
   Map<String, ZoneSnapshot> zoneSnapshot = ZoneAvoidanceRule.createSnapshot(lbStats);
   logger.debug("Zone snapshots: {}", zoneSnapshot);
   if (triggeringLoad == null) {
    triggeringLoad = DynamicPropertyFactory.getInstance().getDoubleProperty(
      "ZoneAwareNIWSDiscoveryLoadBalancer." + this.getName() + ".triggeringLoadPerServerThreshold", 0.2d);
   }

   if (triggeringBlackoutPercentage == null) {
    triggeringBlackoutPercentage = DynamicPropertyFactory.getInstance().getDoubleProperty(
      "ZoneAwareNIWSDiscoveryLoadBalancer." + this.getName() + ".avoidZoneWithBlackoutPercetage", 0.99999d);
   }
   Set<String> availableZones = ZoneAvoidanceRule.getAvailableZones(zoneSnapshot, triggeringLoad.get(), triggeringBlackoutPercentage.get());
   logger.debug("Available zones: {}", availableZones);
   if (availableZones != null && availableZones.size() < zoneSnapshot.keySet().size()) {
    String zone = ZoneAvoidanceRule.randomChooseZone(zoneSnapshot, availableZones);
    logger.debug("Zone chosen: {}", zone);
    if (zone != null) {
     BaseLoadBalancer zoneLoadBalancer = getLoadBalancer(zone);
     server = zoneLoadBalancer.chooseServer(key);
    }
   }
  } catch (Exception e) {
   logger.error("Error choosing server using zone aware logic for load balancer={}", name, e);
  }
  if (server != null) {
   return server;
  } else {
   logger.debug("Zone avoidance logic is not invoked.");
   return super.chooseServer(key);
  }
 }

 //實(shí)際走到的方法
 public Server chooseServer(Object key) {
  if (counter == null) {
   counter = createCounter();
  }
  counter.increment();
  if (rule == null) {
   return null;
  } else {
   try {
    return rule.choose(key);
   } catch (Exception e) {
    logger.warn("LoadBalancer [{}]: Error choosing server for key {}", name, key, e);
    return null;
   }
  }
 }
}

也就是說(shuō)最終會(huì)調(diào)用 IRule 選擇到一個(gè)節(jié)點(diǎn),這里支持很多策略,比如隨機(jī)、輪訓(xùn)、響應(yīng)時(shí)間權(quán)重等:


public interface IRule{

 public Server choose(Object key);
 
 public void setLoadBalancer(ILoadBalancer lb);
 
 public ILoadBalancer getLoadBalancer(); 
}

這里的LoadBalancer是在BaseLoadBalancer的構(gòu)造器中設(shè)置的,上文說(shuō)過(guò),對(duì)于每一個(gè)serviceId服務(wù)來(lái)說(shuō),當(dāng)?shù)谝淮握{(diào)用的時(shí)候會(huì)初始化對(duì)應(yīng)的spring上下文,而這個(gè)上下文中包含了所有ribbon相關(guān)的bean,其中就包括ILoadBalancer、IRule。

原因

通過(guò)跟蹤堆棧,發(fā)現(xiàn)不同的serviceId,IRule是同一個(gè), 而上文說(shuō)過(guò),每個(gè)serviceId都擁有自己獨(dú)立的上下文,包括獨(dú)立的loadBalancer、IRule,而IRule是同一個(gè),因此懷疑是這個(gè)bean是通過(guò)parent context獲取到的,換句話說(shuō)應(yīng)用自己定義了一個(gè)這樣的bean。查看代碼果然如此。

這樣就會(huì)導(dǎo)致一個(gè)問(wèn)題,IRule是共享的,而其他bean是隔離開(kāi)的,因此后面的serviceId初始化的時(shí)候,會(huì)修改這個(gè)IRule的LoadBalancer, 導(dǎo)致之前的服務(wù)獲取到的實(shí)例信息是錯(cuò)誤的,從而導(dǎo)致接口404。

public class BaseLoadBalancer extends AbstractLoadBalancer implements
  PrimeConnections.PrimeConnectionListener, IClientConfigAware {
 public BaseLoadBalancer() {
  this.name = DEFAULT_NAME;
  this.ping = null;
  setRule(DEFAULT_RULE); // 這里會(huì)設(shè)置IRule的loadbalancer
  setupPingTask();
  lbStats = new LoadBalancerStats(DEFAULT_NAME);
 }
}

解決方案

解決方法也很簡(jiǎn)單,最簡(jiǎn)單就將這個(gè)自定義的IRule的bean干掉,另外更標(biāo)準(zhǔn)的做法是使用RibbonClients注解,具體做法可以參考文檔。

總結(jié)

核心原因其實(shí)還是對(duì)于Spring Cloud的理解不夠深刻,用法有錯(cuò)誤,導(dǎo)致出現(xiàn)了一些比較詭異的問(wèn)題。對(duì)于自己使用的組件、框架、甚至于每一個(gè)注解,都要了解其原理,能夠清楚的說(shuō)清楚這個(gè)注解有什么效果,有什么影響,而不是只著眼于解決眼前的問(wèn)題。

再次聲明:代碼不是我寫(xiě)的=_=

好了,以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問(wèn)大家可以留言交流,謝謝大家對(duì)腳本之家的支持。

相關(guān)文章

  • 詳解Java數(shù)組的一維和二維講解和內(nèi)存顯示圖

    詳解Java數(shù)組的一維和二維講解和內(nèi)存顯示圖

    這篇文章主要介紹了Java數(shù)組的一維和二維講解和內(nèi)存顯示圖,數(shù)組就相當(dāng)于一個(gè)容器,存放相同類(lèi)型數(shù)據(jù)的容器。而數(shù)組的本質(zhì)上就是讓我們能 "批量" 創(chuàng)建相同類(lèi)型的變量,需要的朋友可以參考下
    2023-05-05
  • Spring Boot 常用注解大全

    Spring Boot 常用注解大全

    這篇文章主要介紹了Spring Boot 常用注解大全,需要的朋友可以參考下
    2024-02-02
  • Java實(shí)現(xiàn)循環(huán)體的過(guò)濾器的方法

    Java實(shí)現(xiàn)循環(huán)體的過(guò)濾器的方法

    這篇文章主要介紹了Java實(shí)現(xiàn)循環(huán)體的過(guò)濾器的方法,需要的朋友可以參考下
    2014-02-02
  • Java類(lèi)加載器ClassLoader源碼層面分析講解

    Java類(lèi)加載器ClassLoader源碼層面分析講解

    ClassLoader翻譯過(guò)來(lái)就是類(lèi)加載器,普通的java開(kāi)發(fā)者其實(shí)用到的不多,但對(duì)于某些框架開(kāi)發(fā)者來(lái)說(shuō)卻非常常見(jiàn)。理解ClassLoader的加載機(jī)制,也有利于我們編寫(xiě)出更高效的代碼。ClassLoader的具體作用就是將class文件加載到j(luò)vm虛擬機(jī)中去,程序就可以正確運(yùn)行了
    2022-09-09
  • java轉(zhuǎn)換字符串編碼格式的方法

    java轉(zhuǎn)換字符串編碼格式的方法

    這篇文章主要介紹了java轉(zhuǎn)換字符串編碼格式的方法,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2018-08-08
  • Java協(xié)程編程之Loom項(xiàng)目實(shí)戰(zhàn)記錄

    Java協(xié)程編程之Loom項(xiàng)目實(shí)戰(zhàn)記錄

    這篇文章主要介紹了Java協(xié)程編程之Loom項(xiàng)目嘗鮮,如果用嘗鮮的角度去使用Loom項(xiàng)目,可以提前窺探JVM開(kāi)發(fā)者們是如何基于協(xié)程這個(gè)重大特性進(jìn)行開(kāi)發(fā)的,這對(duì)于提高學(xué)習(xí)JDK內(nèi)核代碼的興趣有不少幫助,需要的朋友可以參考下
    2021-08-08
  • springcloud之自定義簡(jiǎn)易消費(fèi)服務(wù)組件

    springcloud之自定義簡(jiǎn)易消費(fèi)服務(wù)組件

    這篇文章主要介紹了springcloud之自定義簡(jiǎn)易消費(fèi)服務(wù)組件,本篇來(lái)使用rest+ribbon消費(fèi)服務(wù),并且通過(guò)輪詢(xún)方式來(lái)自定義了個(gè)簡(jiǎn)易消費(fèi)組件,感興趣的小伙伴們可以參考一下
    2018-06-06
  • Java實(shí)現(xiàn)文本查重的方法詳解

    Java實(shí)現(xiàn)文本查重的方法詳解

    Ansj 是一個(gè)開(kāi)源的 Java 中文分詞工具,基于中科院的 ictclas 中文分詞算法,采用隱馬爾科夫模型(HMM),比其他常用的開(kāi)源分詞工具(如 MMseg4j)的分詞準(zhǔn)確率更高,下面我們就來(lái)使用它實(shí)現(xiàn)文本查重功能吧
    2024-04-04
  • springMVC返回復(fù)雜的json格式數(shù)據(jù)方法

    springMVC返回復(fù)雜的json格式數(shù)據(jù)方法

    下面小編就為大家分享一篇springMVC返回復(fù)雜的json格式數(shù)據(jù)方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2018-03-03
  • SpringBoot使用ApplicationEvent&Listener完成業(yè)務(wù)解耦

    SpringBoot使用ApplicationEvent&Listener完成業(yè)務(wù)解耦

    這篇文章主要介紹了SpringBoot使用ApplicationEvent&Listener完成業(yè)務(wù)解耦示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-05-05

最新評(píng)論