成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

如何優(yōu)雅地自定義Prometheus監(jiān)控指標(biāo)

安全 應(yīng)用安全
目前大部分使用Spring Boot構(gòu)建微服務(wù)體系的公司,大都在使用Prometheus來(lái)構(gòu)建微服務(wù)的度量指標(biāo)(Metrics)類監(jiān)控系統(tǒng)。而一般做法是通過(guò)在微服務(wù)應(yīng)用中集成Prometheus指標(biāo)采集SDK,從而使得Spring Boot暴露相關(guān)Metrics采集端點(diǎn)來(lái)實(shí)現(xiàn)。

[[389927]]

本文轉(zhuǎn)載自微信公眾號(hào)「無(wú)敵碼農(nóng)」,作者無(wú)敵碼農(nóng)。轉(zhuǎn)載本文請(qǐng)聯(lián)系無(wú)敵碼農(nóng)公眾號(hào)。

大家好!我是"無(wú)敵碼農(nóng)",今天要和大家分享的是在實(shí)際工作中“如何優(yōu)雅地自定義Prometheus監(jiān)控指標(biāo)”!目前大部分使用Spring Boot構(gòu)建微服務(wù)體系的公司,大都在使用Prometheus來(lái)構(gòu)建微服務(wù)的度量指標(biāo)(Metrics)類監(jiān)控系統(tǒng)。而一般做法是通過(guò)在微服務(wù)應(yīng)用中集成Prometheus指標(biāo)采集SDK,從而使得Spring Boot暴露相關(guān)Metrics采集端點(diǎn)來(lái)實(shí)現(xiàn)。

但一般來(lái)說(shuō),Spring Boot默認(rèn)暴露的Metrics數(shù)量及類型是有限的,如果想要建立針對(duì)微服務(wù)應(yīng)用更豐富的監(jiān)控維度(例如TP90/TP99分位值指標(biāo)之類),那么還需要我們?cè)赟pring Boot默認(rèn)已經(jīng)打開(kāi)的Metrics基礎(chǔ)之上,配置Prometheus類庫(kù)(micrometer-registry-prometheus)所提供的其他指標(biāo)類型。

但怎么樣才能在Spring Boot框架中以更優(yōu)雅地方式實(shí)現(xiàn)呢?難道需要在業(yè)務(wù)代碼中編寫(xiě)各種自定義監(jiān)控指標(biāo)代碼的暴露邏輯嗎?接下來(lái)的內(nèi)容我們將通過(guò)@注解+AOP的方式來(lái)演示如何以更加優(yōu)雅的方式來(lái)實(shí)現(xiàn)Prometheus監(jiān)控指標(biāo)的自定義!

自定義監(jiān)控指標(biāo)配置注解

需要說(shuō)明的是在Spring Boot應(yīng)用中,對(duì)程序運(yùn)行信息的收集(如指標(biāo)、日志),比較常用的方法是通過(guò)Spring的AOP代理攔截來(lái)實(shí)現(xiàn),但這種攔截程序運(yùn)行過(guò)程的邏輯多少會(huì)損耗點(diǎn)系統(tǒng)性能,因此在自定義Prometheus監(jiān)控指標(biāo)的過(guò)程中,可以將是否上報(bào)指標(biāo)的選擇權(quán)交給開(kāi)發(fā)人員,而從易用性角度來(lái)說(shuō),可以通過(guò)注解的方式實(shí)現(xiàn)。例如:

  1. package com.wudimanong.monitor.metrics.annotation; 
  2.  
  3. import java.lang.annotation.ElementType; 
  4. import java.lang.annotation.Inherited; 
  5. import java.lang.annotation.Retention; 
  6. import java.lang.annotation.RetentionPolicy; 
  7. import java.lang.annotation.Target; 
  8.  
  9. @Target({ElementType.METHOD}) 
  10. @Retention(RetentionPolicy.RUNTIME) 
  11. @Inherited 
  12. public @interface Tp { 
  13.  
  14.     String description() default ""

如上所示代碼,我們定義了一個(gè)用于標(biāo)注上報(bào)計(jì)時(shí)器指標(biāo)類型的注解,如果想統(tǒng)計(jì)接口的想TP90、TP99這樣的分位值指標(biāo),那么就可以通過(guò)該注解標(biāo)注。除此之外,還可以定義上報(bào)其他指標(biāo)類型的注解,例如:

  1. package com.wudimanong.monitor.metrics.annotation; 
  2.  
  3. import java.lang.annotation.ElementType; 
  4. import java.lang.annotation.Inherited; 
  5. import java.lang.annotation.Retention; 
  6. import java.lang.annotation.RetentionPolicy; 
  7. import java.lang.annotation.Target; 
  8.  
  9. @Target({ElementType.METHOD}) 
  10. @Retention(RetentionPolicy.RUNTIME) 
  11. @Inherited 
  12. public @interface Count { 
  13.  
  14.     String description() default ""

如上所示,我們定義了一個(gè)用于上報(bào)計(jì)數(shù)器類型指標(biāo)的注解!如果要統(tǒng)計(jì)接口的平均響應(yīng)時(shí)間、接口的請(qǐng)求量之類的指標(biāo),那么可以通過(guò)該注解標(biāo)注!

而如果覺(jué)得分別定義不同指標(biāo)類型的注解比較麻煩,對(duì)于某些接口上述各種指標(biāo)類型都希望上報(bào)到Prometheus,那么也可以定義一個(gè)通用注解,用于同時(shí)上報(bào)多個(gè)指標(biāo)類型,例如:

  1. package com.wudimanong.monitor.metrics.annotation; 
  2.  
  3. import java.lang.annotation.ElementType; 
  4. import java.lang.annotation.Inherited; 
  5. import java.lang.annotation.Retention; 
  6. import java.lang.annotation.RetentionPolicy; 
  7. import java.lang.annotation.Target; 
  8.  
  9. @Target({ElementType.METHOD}) 
  10. @Retention(RetentionPolicy.RUNTIME) 
  11. @Inherited 
  12. public @interface Monitor { 
  13.  
  14.     String description() default ""

總之,無(wú)論是分開(kāi)定義特定指標(biāo)注解還是定義一個(gè)通用的指標(biāo)注解,其目標(biāo)都是希望以更靈活的方式來(lái)擴(kuò)展Spring Boot微服務(wù)應(yīng)用的監(jiān)控指標(biāo)類型。

自定義監(jiān)控指標(biāo)注解AOP代理邏輯實(shí)現(xiàn)

上面我們靈活定義了上報(bào)不同指標(biāo)類型的注解,而上述注解的具體實(shí)現(xiàn)邏輯,可以通過(guò)定義一個(gè)通用的AOP代理類來(lái)實(shí)現(xiàn),具體實(shí)現(xiàn)代碼如下:

  1. package com.wudimanong.monitor.metrics.aop; 
  2.  
  3. import com.wudimanong.monitor.metrics.Metrics; 
  4. import com.wudimanong.monitor.metrics.annotation.Count
  5. import com.wudimanong.monitor.metrics.annotation.Monitor; 
  6. import com.wudimanong.monitor.metrics.annotation.Tp; 
  7. import io.micrometer.core.instrument.Counter; 
  8. import io.micrometer.core.instrument.MeterRegistry; 
  9. import io.micrometer.core.instrument.Tag; 
  10. import io.micrometer.core.instrument.Tags; 
  11. import io.micrometer.core.instrument.Timer; 
  12. import java.lang.reflect.Method; 
  13. import java.util.function.Function
  14. import org.aspectj.lang.ProceedingJoinPoint; 
  15. import org.aspectj.lang.annotation.Around; 
  16. import org.aspectj.lang.annotation.Aspect; 
  17. import org.aspectj.lang.reflect.MethodSignature; 
  18. import org.springframework.stereotype.Component; 
  19.  
  20. @Aspect 
  21. @Component 
  22. public class MetricsAspect { 
  23.  
  24.     /** 
  25.      * Prometheus指標(biāo)管理 
  26.      */ 
  27.     private MeterRegistry registry; 
  28.  
  29.     private Function<ProceedingJoinPoint, Iterable<Tag>> tagsBasedOnJoinPoint; 
  30.  
  31.     public MetricsAspect(MeterRegistry registry) { 
  32.         this.init(registry, pjp -> Tags 
  33.                 .of(new String[]{"class", pjp.getStaticPart().getSignature().getDeclaringTypeName(), "method"
  34.                         pjp.getStaticPart().getSignature().getName()})); 
  35.     } 
  36.  
  37.     public void init(MeterRegistry registry, Function<ProceedingJoinPoint, Iterable<Tag>> tagsBasedOnJoinPoint) { 
  38.         this.registry = registry; 
  39.         this.tagsBasedOnJoinPoint = tagsBasedOnJoinPoint; 
  40.     } 
  41.  
  42.     /** 
  43.      * 針對(duì)@Tp指標(biāo)配置注解的邏輯實(shí)現(xiàn) 
  44.      */ 
  45.     @Around("@annotation(com.wudimanong.monitor.metrics.annotation.Tp)"
  46.     public Object timedMethod(ProceedingJoinPoint pjp) throws Throwable { 
  47.         Method method = ((MethodSignature) pjp.getSignature()).getMethod(); 
  48.         method = pjp.getTarget().getClass().getMethod(method.getName(), method.getParameterTypes()); 
  49.         Tp tp = method.getAnnotation(Tp.class); 
  50.         Timer.Sample sample = Timer.start(this.registry); 
  51.         String exceptionClass = "none"
  52.         try { 
  53.             return pjp.proceed(); 
  54.         } catch (Exception ex) { 
  55.             exceptionClass = ex.getClass().getSimpleName(); 
  56.             throw ex; 
  57.         } finally { 
  58.             try { 
  59.                 String finalExceptionClass = exceptionClass; 
  60.                 //創(chuàng)建定義計(jì)數(shù)器,并設(shè)置指標(biāo)的Tags信息(名稱可以自定義) 
  61.                 Timer timer = Metrics.newTimer("tp.method.timed"
  62.                         builder -> builder.tags(new String[]{"exception", finalExceptionClass}) 
  63.                                 .tags(this.tagsBasedOnJoinPoint.apply(pjp)).tag("description", tp.description()) 
  64.                                 .publishPercentileHistogram().register(this.registry)); 
  65.                 sample.stop(timer); 
  66.             } catch (Exception exception) { 
  67.             } 
  68.         } 
  69.     } 
  70.  
  71.     /** 
  72.      * 針對(duì)@Count指標(biāo)配置注解的邏輯實(shí)現(xiàn) 
  73.      */ 
  74.     @Around("@annotation(com.wudimanong.monitor.metrics.annotation.Count)"
  75.     public Object countMethod(ProceedingJoinPoint pjp) throws Throwable { 
  76.         Method method = ((MethodSignature) pjp.getSignature()).getMethod(); 
  77.         method = pjp.getTarget().getClass().getMethod(method.getName(), method.getParameterTypes()); 
  78.         Count count = method.getAnnotation(Count.class); 
  79.         String exceptionClass = "none"
  80.         try { 
  81.             return pjp.proceed(); 
  82.         } catch (Exception ex) { 
  83.             exceptionClass = ex.getClass().getSimpleName(); 
  84.             throw ex; 
  85.         } finally { 
  86.             try { 
  87.                 String finalExceptionClass = exceptionClass; 
  88.                 //創(chuàng)建定義計(jì)數(shù)器,并設(shè)置指標(biāo)的Tags信息(名稱可以自定義) 
  89.                 Counter counter = Metrics.newCounter("count.method.counted"
  90.                         builder -> builder.tags(new String[]{"exception", finalExceptionClass}) 
  91.                                 .tags(this.tagsBasedOnJoinPoint.apply(pjp)).tag("description"count.description()) 
  92.                                 .register(this.registry)); 
  93.                 counter.increment(); 
  94.             } catch (Exception exception) { 
  95.             } 
  96.         } 
  97.     } 
  98.  
  99.     /** 
  100.      * 針對(duì)@Monitor通用指標(biāo)配置注解的邏輯實(shí)現(xiàn) 
  101.      */ 
  102.     @Around("@annotation(com.wudimanong.monitor.metrics.annotation.Monitor)"
  103.     public Object monitorMethod(ProceedingJoinPoint pjp) throws Throwable { 
  104.         Method method = ((MethodSignature) pjp.getSignature()).getMethod(); 
  105.         method = pjp.getTarget().getClass().getMethod(method.getName(), method.getParameterTypes()); 
  106.         Monitor monitor = method.getAnnotation(Monitor.class); 
  107.         String exceptionClass = "none"
  108.         try { 
  109.             return pjp.proceed(); 
  110.         } catch (Exception ex) { 
  111.             exceptionClass = ex.getClass().getSimpleName(); 
  112.             throw ex; 
  113.         } finally { 
  114.             try { 
  115.                 String finalExceptionClass = exceptionClass; 
  116.                 //計(jì)時(shí)器Metric 
  117.                 Timer timer = Metrics.newTimer("tp.method.timed"
  118.                         builder -> builder.tags(new String[]{"exception", finalExceptionClass}) 
  119.                                 .tags(this.tagsBasedOnJoinPoint.apply(pjp)).tag("description", monitor.description()) 
  120.                                 .publishPercentileHistogram().register(this.registry)); 
  121.                 Timer.Sample sample = Timer.start(this.registry); 
  122.                 sample.stop(timer); 
  123.  
  124.                 //計(jì)數(shù)器Metric 
  125.                 Counter counter = Metrics.newCounter("count.method.counted"
  126.                         builder -> builder.tags(new String[]{"exception", finalExceptionClass}) 
  127.                                 .tags(this.tagsBasedOnJoinPoint.apply(pjp)).tag("description", monitor.description()) 
  128.                                 .register(this.registry)); 
  129.                 counter.increment(); 
  130.             } catch (Exception exception) { 
  131.             } 
  132.         } 
  133.     } 

上述代碼完整的實(shí)現(xiàn)了前面我們定義的指標(biāo)配置注解的邏輯,其中針對(duì)@Monitor注解的邏輯就是@Tp和@Count注解邏輯的整合。如果還需要定義其他指標(biāo)類型,可以在此基礎(chǔ)上繼續(xù)擴(kuò)展!

需要注意,在上述邏輯實(shí)現(xiàn)中對(duì)“Timer”及“Counter”等指標(biāo)類型的構(gòu)建這里并沒(méi)有直接使用“micrometer-registry-prometheus”依賴包中的構(gòu)建對(duì)象,而是通過(guò)自定義的Metrics.newTimer()這樣的方式實(shí)現(xiàn),其主要用意是希望以更簡(jiǎn)潔、靈活的方式去實(shí)現(xiàn)指標(biāo)的上報(bào),其代碼定義如下:

  1. package com.wudimanong.monitor.metrics; 
  2.  
  3. import io.micrometer.core.instrument.Counter; 
  4. import io.micrometer.core.instrument.Counter.Builder; 
  5. import io.micrometer.core.instrument.DistributionSummary; 
  6. import io.micrometer.core.instrument.Gauge; 
  7. import io.micrometer.core.instrument.MeterRegistry; 
  8. import io.micrometer.core.instrument.Timer; 
  9. import io.micrometer.core.lang.NonNull; 
  10. import java.util.function.Consumer; 
  11. import java.util.function.Supplier; 
  12. import org.springframework.beans.BeansException; 
  13. import org.springframework.context.ApplicationContext; 
  14. import org.springframework.context.ApplicationContextAware; 
  15.  
  16. public class Metrics implements ApplicationContextAware { 
  17.  
  18.     private static ApplicationContext context; 
  19.  
  20.     @Override 
  21.     public void setApplicationContext(@NonNull ApplicationContext applicationContext) throws BeansException { 
  22.         context = applicationContext; 
  23.     } 
  24.  
  25.     public static ApplicationContext getContext() { 
  26.         return context; 
  27.     } 
  28.  
  29.     public static Counter newCounter(String name, Consumer<Builder> consumer) { 
  30.         MeterRegistry meterRegistry = context.getBean(MeterRegistry.class); 
  31.         return new CounterBuilder(meterRegistry, name, consumer).build(); 
  32.     } 
  33.  
  34.     public static Timer newTimer(String name, Consumer<Timer.Builder> consumer) { 
  35.         return new TimerBuilder(context.getBean(MeterRegistry.class), name, consumer).build(); 
  36.     } 

上述代碼通過(guò)接入Spring容器上下文,獲取了MeterRegistry實(shí)例,并以此來(lái)構(gòu)建像Counter、Timer這樣的指標(biāo)類型對(duì)象。而這里之所以將獲取方法定義為靜態(tài)的,主要是便于在業(yè)務(wù)代碼中進(jìn)行引用!

而在上述代碼中涉及的CounterBuilder、TimerBuilder構(gòu)造器代碼定義分別如下:

  1. package com.wudimanong.monitor.metrics; 
  2.  
  3. import io.micrometer.core.instrument.Counter; 
  4. import io.micrometer.core.instrument.Counter.Builder; 
  5. import io.micrometer.core.instrument.MeterRegistry; 
  6. import java.util.function.Consumer; 
  7.  
  8. public class CounterBuilder { 
  9.  
  10.     private final MeterRegistry meterRegistry; 
  11.  
  12.     private Counter.Builder builder; 
  13.  
  14.     private Consumer<Builder> consumer; 
  15.  
  16.     public CounterBuilder(MeterRegistry meterRegistry, String name, Consumer<Counter.Builder> consumer) { 
  17.         this.builder = Counter.builder(name); 
  18.         this.meterRegistry = meterRegistry; 
  19.         this.consumer = consumer; 
  20.     } 
  21.  
  22.     public Counter build() { 
  23.         consumer.accept(builder); 
  24.         return builder.register(meterRegistry); 
  25.     } 

上述代碼為CounterBuilder構(gòu)造器代碼!TimerBuilder構(gòu)造器代碼如下:

  1. package com.wudimanong.monitor.metrics; 
  2.  
  3. import io.micrometer.core.instrument.MeterRegistry; 
  4. import io.micrometer.core.instrument.Timer; 
  5. import io.micrometer.core.instrument.Timer.Builder; 
  6. import java.util.function.Consumer; 
  7.  
  8. public class TimerBuilder { 
  9.  
  10.     private final MeterRegistry meterRegistry; 
  11.  
  12.     private Timer.Builder builder; 
  13.  
  14.     private Consumer<Builder> consumer; 
  15.  
  16.     public TimerBuilder(MeterRegistry meterRegistry, String name, Consumer<Timer.Builder> consumer) { 
  17.         this.builder = Timer.builder(name); 
  18.         this.meterRegistry = meterRegistry; 
  19.         this.consumer = consumer; 
  20.     } 
  21.  
  22.     public Timer build() { 
  23.         this.consumer.accept(builder); 
  24.         return builder.register(meterRegistry); 
  25.     } 

之所以還特地將構(gòu)造器代碼單獨(dú)定義,主要是從代碼的優(yōu)雅性考慮!如果涉及其他指標(biāo)類型的構(gòu)造,也可以通過(guò)類似的方法進(jìn)行擴(kuò)展!

自定義指標(biāo)注解配置類

在上述代碼中我們已經(jīng)定義了幾個(gè)自定義指標(biāo)注解及其實(shí)現(xiàn)邏輯代碼,為了使其在Spring Boot環(huán)境中運(yùn)行,還需要編寫(xiě)如下配置類,代碼如下:

  1. package com.wudimanong.monitor.metrics.config; 
  2.  
  3. import com.wudimanong.monitor.metrics.Metrics; 
  4. import io.micrometer.core.instrument.MeterRegistry; 
  5. import org.springframework.boot.actuate.autoconfigure.metrics.MeterRegistryCustomizer; 
  6. import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; 
  7. import org.springframework.context.annotation.Bean; 
  8. import org.springframework.context.annotation.Configuration; 
  9. import org.springframework.core.env.Environment; 
  10.  
  11. @Configuration 
  12. public class CustomMetricsAutoConfiguration { 
  13.  
  14.     @Bean 
  15.     @ConditionalOnMissingBean 
  16.     public MeterRegistryCustomizer<MeterRegistry> meterRegistryCustomizer(Environment environment) { 
  17.         return registry -> { 
  18.             registry.config() 
  19.                     .commonTags("application", environment.getProperty("spring.application.name")); 
  20.         }; 
  21.     } 
  22.  
  23.     @Bean 
  24.     @ConditionalOnMissingBean 
  25.     public Metrics metrics() { 
  26.         return new Metrics(); 
  27.     } 

上述配置代碼主要是約定了上報(bào)Prometheus指標(biāo)信息中所攜帶的應(yīng)用名稱,并對(duì)自定義了Metrics類進(jìn)行了Bean配置!

業(yè)務(wù)代碼的使用方式及效果

接下來(lái)我們演示在業(yè)務(wù)代碼中如果要上報(bào)Prometheus監(jiān)控指標(biāo)應(yīng)該怎么寫(xiě),具體如下:

  1. package com.wudimanong.monitor.controller; 
  2.  
  3. import com.wudimanong.monitor.metrics.annotation.Count
  4. import com.wudimanong.monitor.metrics.annotation.Monitor; 
  5. import com.wudimanong.monitor.metrics.annotation.Tp; 
  6. import com.wudimanong.monitor.service.MonitorService; 
  7. import org.springframework.beans.factory.annotation.Autowired; 
  8. import org.springframework.web.bind.annotation.GetMapping; 
  9. import org.springframework.web.bind.annotation.RequestMapping; 
  10. import org.springframework.web.bind.annotation.RequestParam; 
  11. import org.springframework.web.bind.annotation.RestController; 
  12.  
  13. @RestController 
  14. @RequestMapping("/monitor"
  15. public class MonitorController { 
  16.  
  17.     @Autowired 
  18.     private MonitorService monitorServiceImpl; 
  19.  
  20.     //監(jiān)控指標(biāo)注解使用 
  21.     //@Tp(description = "/monitor/test"
  22.     //@Count(description = "/monitor/test"
  23.     @Monitor(description = "/monitor/test"
  24.     @GetMapping("/test"
  25.     public String monitorTest(@RequestParam("name") String name) { 
  26.         monitorServiceImpl.monitorTest(name); 
  27.         return "監(jiān)控示范工程測(cè)試接口返回->OK!"
  28.     } 

如上述代碼所示,在實(shí)際的業(yè)務(wù)編程中就可以比較簡(jiǎn)單的通過(guò)注解來(lái)配置接口所上傳的Prometheus監(jiān)控指標(biāo)了!此時(shí)在本地啟動(dòng)程序,可以通過(guò)訪問(wèn)微服務(wù)應(yīng)用的“/actuator/prometheus”指標(biāo)采集端點(diǎn)來(lái)查看相關(guān)指標(biāo),如下圖所示:

有了這些自定義上報(bào)的監(jiān)控指標(biāo),那么Promethues在采集后,我們就可以通過(guò)像Grafana這樣的可視化工具,來(lái)構(gòu)建起多維度界面友好地監(jiān)控視圖了,例如以TP90/TP99為例:

如上所示,在Grafana中可以同時(shí)定義多個(gè)PromeQL來(lái)定于不同的監(jiān)控指標(biāo)信息,這里我們分別通過(guò)Prometheus所提供的“histogram_quantile”函數(shù)統(tǒng)計(jì)了接口方法“monitorTest”的TP90及TP95分位值!而所使用的指標(biāo)就是自定義的“tp_method_timed_xx”指標(biāo)類型!

后記

以上就是我最近在工作中封裝的一組關(guān)于Prometheus自定義監(jiān)控指標(biāo)的SDK代碼,在實(shí)際工作中可以將其封住為Spring Boot Starter依賴的形式,從而更好地被Spring Boot項(xiàng)目集成!至此我已經(jīng)毫無(wú)保留的將最近兩天的工作成果分享給大家了,也希望各位老鐵可以多多點(diǎn)贊支持,多多轉(zhuǎn)發(fā)傳播!

 

責(zé)任編輯:武曉燕 來(lái)源: 無(wú)敵碼農(nóng)
相關(guān)推薦

2023-12-29 08:01:52

自定義指標(biāo)模板

2020-12-14 10:26:48

Prometheus 監(jiān)控Services

2021-10-28 08:39:22

Node Export自定義 監(jiān)控

2021-05-28 08:58:41

Golang網(wǎng)卡metrics

2023-03-26 08:41:37

2013-01-10 09:36:19

NagiosNagios插件

2022-07-08 08:00:31

Prometheus監(jiān)控

2022-05-12 08:01:26

vmagentprometheus

2021-10-14 08:07:33

Go 應(yīng)用Prometheus監(jiān)控

2021-03-24 10:20:50

Fonts前端代碼

2023-05-28 13:11:43

Plotly指標(biāo)圖表

2023-09-06 08:46:47

2013-06-27 11:10:01

iOS開(kāi)發(fā)自定義UISlider

2022-03-07 07:33:24

Spring自定義機(jī)制線程池

2024-11-13 16:37:00

Java線程池

2009-09-07 22:00:15

LINQ自定義

2016-02-26 14:57:50

飛象網(wǎng)

2020-03-26 11:04:00

Linux命令光標(biāo)

2021-01-18 13:17:04

鴻蒙HarmonyOSAPP

2021-05-12 22:07:43

并發(fā)編排任務(wù)
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)

主站蜘蛛池模板: 一级黄色片在线免费观看 | 色视频成人在线观看免 | 国产一区二区 | 久久人体 | 亚洲一区视频在线 | www.99re| 欧美高清视频 | 狠狠干狠狠操 | 精品国产青草久久久久福利 | 欧美aⅴ | 综合网中文字幕 | 天堂久久天堂综合色 | 欧美精品福利 | 丁香五月缴情综合网 | 国产亚洲精品成人av久久ww | 在线观看中文字幕 | 超碰免费在线观看 | 黄色网址在线免费观看 | 久久久精品国产 | 亚洲欧美日韩电影 | 一级看片免费视频 | 伊人久久在线 | 91精品国产自产精品男人的天堂 | 久久av一区二区 | 99久久免费精品 | 亚洲成人精品影院 | 中文字幕国产精品视频 | 一级欧美一级日韩片免费观看 | 玖玖玖在线| 新av在线| 日韩α片 | 久久中文字幕一区 | 1级黄色大片 | 蜜桃av一区二区三区 | 午夜影院视频在线观看 | 91黄在线观看 | 黄色av观看 | 中文一区 | 狠狠av| 国产一区二区三区四区五区加勒比 | 天天天操操操 |