系统开发到一定的阶段,线上的机器越来越多,就需要一些监控了,除了服务器的监控,业务方面也需要一些监控服务。Metrics
作为一款监控指标
的度量类库,提供了许多工具帮助开发者来完成自定义的监控工作。
通过构建一个Spring Boot
的基本应用来演示Metrics
的工作方式。
在Maven的Metrics
:
<dependency><groupId>ics</groupId><artifactId>metrics-core</artifactId><version>${metrics.version}</version>
</dependency>
目前Metrics
的最新版本是3.1.2
。
Metrics
提供了五个基本的度量类型:
Metrics
中MetricRegistry
是中心容器,它是程序中所有度量的容器,所有新的度量工具都要注册到一个MetricRegistry
实例中才可以使用,尽量在一个应用中保持让这个MetricRegistry
实例保持单例。
在代码中配置好这个MetricRegistry
容器:
@Bean
public MetricRegistry metrics() {return new MetricRegistry();
}
TPS计算器
这个名称并不准确,Meters
工具会帮助我们统计系统中某一个事件的速率。比如每秒请求数(TPS),每秒查询数(QPS)等等。这个指标能反应系统当前的处理能力,帮助我们判断资源是否已经不足。Meters
本身是一个自增计数器。
通过MetricRegistry
可以获得一个Meter
:
@Bean
public Meter requestMeter(MetricRegistry metrics) {("request");
}
在请求中调用mark()
方法,来增加计数,我们可以在不同的请求中添加不同的Meter
,针对自己的系统完成定制的监控需求。
@RequestMapping("/hello")
@ResponseBody
public String helloWorld() {requestMeter.mark();return "Hello World";
}
应用运行的过程中,在console中反馈的信息:
-- Meters ----------------------------------------------------------------------
requestcount = 21055mean rate = 133.35 events/second1-minute rate = 121.66 events/second5-minute rate = 36.99 events/second15-minute rate = 13.33 events/second
从以上信息中可以看出Meter
可以为我们提供平均速率,以及采样后的1分钟,5分钟,15分钟的速率。
直方图是一种非常常见的统计图表,Metrics
通过这个Histogram
这个度量类型提供了一些方便实时绘制直方图的数据。
和之前的Meter
相同,我们可以通过MetricRegistry
来获得一个Histogram
。
@Bean
public Histogram responseSizes(MetricRegistry metrics) {return metrics.histogram("response-sizes");
}
在应用中,需要统计的位置调用Histogram
的update()
方法。
responseSizes.update(new Random().nextInt(10));
比如我们需要统计某个方法的网络流量,通过Histogram
就非常的方便。
在console中Histogram
反馈的信息:
-- Histograms ------------------------------------------------------------------
response-sizescount = 21051min = 0max = 9mean = 4.55stddev = 2.88median = 4.0075% <= 7.0095% <= 9.0098% <= 9.0099% <= 9.0099.9% <= 9.00
Histogram
为我们提供了最大值,最小值和平均值等数据,利用这些数据,我们就可以开始绘制自定义的直方图了。
Counter
的本质就是一个AtomicLong
实例,可以增加或者减少值,可以用它来统计队列中Job的总数。
通过MetricRegistry
也可以获得一个Counter
实例。
@Bean
public Counter pendingJobs(MetricRegistry metrics) {unter("requestCount");
}
在需要统计数据的位置调用inc()
和dec()
方法。
// 增加计数
pendingJobs.inc();
// 减去计数
pendingJobs.dec();
console的输出非常简单:
-- Counters --------------------------------------------------------------------
requestCountcount = 21051
只是输出了当前度量的值。
Timer
是一个Meter
和Histogram
的组合。这个度量单位可以比较方便地统计请求的速率和处理时间。对于接口中调用的延迟等信息的统计就比较方便了。如果发现一个方法的RPS(请求速率)
很低,而且平均的处理时间很长,那么这个方法八成出问题了。
同样,通过MetricRegistry
获取一个Timer
的实例:
@Bean
public Timer responses(MetricRegistry metrics) {return metrics.timer("executeTime");
}
在需要统计信息的位置使用这样的代码:
final Timer.Context context = responses.time();
try {// handle request
} finally {context.stop();
}
console中就会实时返回这个Timer
的信息:
-- Timers ----------------------------------------------------------------------
executeTimecount = 21061mean rate = 133.39 calls/second1-minute rate = 122.22 calls/second5-minute rate = 37.11 calls/second15-minute rate = 13.37 calls/secondmin = 0.00 millisecondsmax = 0.01 millisecondsmean = 0.00 millisecondsstddev = 0.00 millisecondsmedian = 0.00 milliseconds75% <= 0.00 milliseconds95% <= 0.00 milliseconds98% <= 0.00 milliseconds99% <= 0.00 milliseconds99.9% <= 0.01 milliseconds
除了Metrics
提供的几个度量类型,我们可以通过Gauges
完成自定义的度量类型。比方说很简单的,我们想看我们缓存里面的数据大小,就可以自己定义一个Gauges
。
这样Metrics
就会一直监控Cache
的大小。
除此之外有时候,我们需要计算自己定义的一直单位,比如消息队列里面消费者(consumers)消费的速率和生产者(producers)的生产速率的比例,这也是一个度量。
public class CompareRatio extends RatioGauge {private final Meter consumers;private final Meter producers;public CacheHitRatio(Meter consumers, Meter producers) {sumers = consumers;this.producers = producers;}@Overrideprotected Ratio getRatio() {return Ratio.OneMinuteRate(),OneMinuteRate());}
}
把这个类也注册到Metrics
容器里面:
@Bean
public CompareRatio cacheHitRatio(MetricRegistry metrics, Meter requestMeter, Meter producers) {CompareRatio compareRatio = new CompareRatio(consumers, producers);ister("生产者消费者比率", compareRatio);return cacheHitRatio;
}
Metrics
通过报表,将采集的数据展现到不同的位置,这里比如我们注册一个ConsoleReporter
到MetricRegistry
中,那么console中就会打印出对应的信息。
@Bean
public ConsoleReporter consoleReporter(MetricRegistry metrics) {return ConsoleReporter.forRegistry(metrics).convertRatesTo(TimeUnit.SECONDS).convertDurationsTo(TimeUnit.MILLISECONDS).build();
}
除此之外Metrics
还支持JMX
、HTTP
、Slf4j
等等,可以访问 .1.0/manual/core/#reporters 来查看Metrics
提供的报表,如果还是不能满足自己的业务,也可以自己继承Metrics
提供的ScheduledReporter
类完成自定义的报表类。
这个demo是在一个很简单的spring boot下运行的,关键的几个类完整代码如下。
配置类MetricConfig.java
fig;ics.*;
import org.slf4j.LoggerFactory;
import t.annotation.Bean;
import t.annotation.Configuration;import urrent.TimeUnit;@Configuration
public class MetricConfig {@Beanpublic MetricRegistry metrics() {return new MetricRegistry();}/*** Reporter 数据的展现位置** @param metrics* @return*/@Beanpublic ConsoleReporter consoleReporter(MetricRegistry metrics) {return ConsoleReporter.forRegistry(metrics).convertRatesTo(TimeUnit.SECONDS).convertDurationsTo(TimeUnit.MILLISECONDS).build();}@Beanpublic Slf4jReporter slf4jReporter(MetricRegistry metrics) {return Slf4jReporter.forRegistry(metrics).Logger(ics")).convertRatesTo(TimeUnit.SECONDS).convertDurationsTo(TimeUnit.MILLISECONDS).build();}@Beanpublic JmxReporter jmxReporter(MetricRegistry metrics) {return JmxReporter.forRegistry(metrics).build();}/*** 自定义单位** @param metrics* @return*/@Beanpublic ListManager listManager(MetricRegistry metrics) {return new ListManager(metrics);}/*** TPS 计算器** @param metrics* @return*/@Beanpublic Meter requestMeter(MetricRegistry metrics) {("request");}/*** 直方图** @param metrics* @return*/@Beanpublic Histogram responseSizes(MetricRegistry metrics) {return metrics.histogram("response-sizes");}/*** 计数器** @param metrics* @return*/@Beanpublic Counter pendingJobs(MetricRegistry metrics) {unter("requestCount");}/*** 计时器** @param metrics* @return*/@Beanpublic Timer responses(MetricRegistry metrics) {return metrics.timer("executeTime");}}
接收请求的类MainController.java
ics.action;ics.Counter;
ics.Histogram;
ics.Meter;
ics.Timer;
fig.ListManager;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.ResponseBody;import java.util.Random;@Controller
@RequestMapping("/")
public class MainController {@Autowiredprivate Meter requestMeter;@Autowiredprivate Histogram responseSizes;@Autowiredprivate Counter pendingJobs;@Autowiredprivate Timer responses;@Autowiredprivate ListManager listManager;@RequestMapping("/hello")@ResponseBodypublic String helloWorld() {requestMeter.mark();pendingJobs.inc();responseSizes.update(new Random().nextInt(10));List().add(1);final Timer.Context context = responses.time();try {return "Hello World";} finally {context.stop();}}
}
项目启动类DemoApplication.java
:
ics;ics.ConsoleReporter;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import t.ApplicationContext;import urrent.TimeUnit;@SpringBootApplication
public class DemoApplication {public static void main(String[] args) {ApplicationContext ctx = SpringApplication.run(DemoApplication.class, args);// 启动ReporterConsoleReporter reporter = Bean(ConsoleReporter.class);reporter.start(1, TimeUnit.SECONDS);}
}
本文发布于:2024-02-01 05:13:28,感谢您对本站的认可!
本文链接:https://www.4u4v.net/it/170673561134132.html
版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系,我们将在24小时内删除。
留言与评论(共有 0 条评论) |