上一篇博文中说到Prometheus有四种指标类型:Counter(计数器)、Gauge(仪表盘)、Histogram(直方图)、Summary(摘要),并且我们做了一个Counter的Demo,接下来看看Histogram。

3、Summary:摘要

summary是采集展示百分位数,百分位定义:在一组由小到大的数字中,某个数字大于90%的数字,这个数字就是第90个的百分位数。

通过demo的来理解一下吧,假如我们业务需求是要知道订单金额10,30,50,70,90的百分位数,该怎么实现呢?

需要在MetricsHub.cs中添加Summary类型的指标收集集合:

using Prometheus;
using System.Collections.Generic;namespace PrometheusSample.Middlewares
{public class MetricsHub{private static Dictionary<string, Counter> _counterDictionary = new Dictionary<string, Counter>();private static Dictionary<string, Dictionary<string, Gauge>> _gaugeDictionary = new Dictionary<string, Dictionary<string, Gauge>>();private static Dictionary<string, Summary> _summaryDictionary = new Dictionary<string, Summary>();private static Dictionary<string, Histogram> _histogramDictionary = new Dictionary<string, Histogram>();public Counter GetCounter(string key){if (_counterDictionary.ContainsKey(key)){return _counterDictionary[key];}else{return null;}}public Dictionary<string, Gauge> GetGauge(string key){if (_gaugeDictionary.ContainsKey(key)){return _gaugeDictionary[key];}else{return null;}}public Summary GetSummary(string key){if (_summaryDictionary.ContainsKey(key)){return _summaryDictionary[key];}else{return null;}}public Histogram GetHistogram(string key){if (_histogramDictionary.ContainsKey(key)){return _histogramDictionary[key];}else{return null;}}public void AddCounter(string key, Counter counter){_counterDictionary.Add(key, counter);}public void AddGauge(string key, Dictionary<string, Gauge> gauges){_gaugeDictionary.Add(key, gauges);}public void AddSummary(string key, Summary summary){_summaryDictionary.Add(key, summary);}public void AddHistogram(string key, Histogram histogram){_histogramDictionary.Add(key, histogram);}}
}

接下来就要在BusinessMetricsMiddleware的中间件中添加处理Summary指标的代码了:

using Microsoft.AspNetCore.Http;
using PrometheusSample.Models;
using System.IO;
using System.Threading.Tasks;namespace PrometheusSample.Middlewares
{/// <summary>/// 请求记录中间件/// </summary>public class BusinessMetricsMiddleware{private readonly RequestDelegate _next;public BusinessMetricsMiddleware(RequestDelegate next){_next = next;}public async Task InvokeAsync(HttpContext context, MetricsHub metricsHub){var originalBody = context.Response.Body;try{using (var memStream = new MemoryStream()){//从管理返回的Response中取出返回数据,根据返回值进行监控指标计数context.Response.Body = memStream;await _next(context);memStream.Position = 0;string responseBody = new StreamReader(memStream).ReadToEnd();memStream.Position = 0;await memStream.CopyToAsync(originalBody);if (metricsHub.GetCounter(context.Request.Path) != null || metricsHub.GetGauge(context.Request.Path) != null){//这里约定所有action返回值是一个APIResult类型var result = System.Text.Json.JsonSerializer.Deserialize<APIResult>(responseBody, new System.Text.Json.JsonSerializerOptions { PropertyNameCaseInsensitive = true });if (result != null && result.Result){//获取到Countervar counter = metricsHub.GetCounter(context.Request.Path);if (counter != null){//计数counter.Inc();}var gauges = metricsHub.GetGauge(context.Request.Path);if (gauges != null){//存在增加指标+就Incif (gauges.ContainsKey("+")){gauges["+"].Inc();} //存在减少指标-就Decif (gauges.ContainsKey("-")){gauges["-"].Dec();}}var histogram = metricsHub.GetHistogram(context.Request.Path);if (histogram != null){var parseResult = int.TryParse(result.Data.ToString(), out int i);if (parseResult){histogram.Observe(i);}}var summary = metricsHub.GetSummary(context.Request.Path);if (summary != null){var parseResult = int.TryParse(result.Data.ToString(), out int i);if (parseResult){summary.Observe(i);}}                            }}}}finally{context.Response.Body = originalBody;}}}
}

再就是在Starsup中配置对应url的Summary参数了:

using Microsoft.AspNetCore.Builder;
using Microsoft.AspNetCore.Hosting;
using Microsoft.Extensions.Configuration;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.Hosting;
using Microsoft.OpenApi.Models;
using Prometheus;
using PrometheusSample.Middlewares;
using PrometheusSample.Services;
using System.Collections.Generic;namespace PrometheusSample
{public class Startup{public Startup(IConfiguration configuration){Configuration = configuration;}public IConfiguration Configuration { get; }public void ConfigureServices(IServiceCollection services){MetricsHandle(services);services.AddScoped<IOrderService, OrderService>();services.AddControllers();services.AddSwaggerGen(c =>{c.SwaggerDoc("v1", new OpenApiInfo { Title = "PrometheusSample", Version = "v1" });});}public void Configure(IApplicationBuilder app, IWebHostEnvironment env){if (env.IsDevelopment()){app.UseDeveloperExceptionPage();app.UseSwagger();app.UseSwaggerUI(c => c.SwaggerEndpoint("/swagger/v1/swagger.json", "PrometheusSample v1"));}app.UseRouting();//http请求的中间件app.UseHttpMetrics();app.UseAuthorization();//自定义业务跟踪app.UseBusinessMetrics();app.UseEndpoints(endpoints =>{//映射监控地址为  /metricsendpoints.MapMetrics();endpoints.MapControllers();});}/// <summary>/// 处理监控事项/// </summary>/// <param name="services"></param>void MetricsHandle(IServiceCollection services){var metricsHub = new MetricsHub();//countermetricsHub.AddCounter("/register", Metrics.CreateCounter("business_register_user", "注册用户数。"));metricsHub.AddCounter("/order", Metrics.CreateCounter("business_order_total", "下单总数。"));metricsHub.AddCounter("/pay", Metrics.CreateCounter("business_pay_total", "支付总数。"));metricsHub.AddCounter("/ship", Metrics.CreateCounter("business_ship_total", "发货总数。"));//gaugevar orderGauge = Metrics.CreateGauge("business_order_count", "当前下单数量。");var payGauge = Metrics.CreateGauge("business_pay_count", "当前支付数量。");var shipGauge = Metrics.CreateGauge("business_ship_count", "当前发货数据。");metricsHub.AddGauge("/order", new Dictionary<string, Gauge> {{ "+", orderGauge}});metricsHub.AddGauge("/pay", new Dictionary<string, Gauge> {{"-",orderGauge},{"+",payGauge}});metricsHub.AddGauge("/ship", new Dictionary<string, Gauge> {{"+",shipGauge},{"-",payGauge}});//histogram   var orderHistogram = Metrics.CreateHistogram("business_order_histogram", "订单直方图。",new HistogramConfiguration{Buckets = Histogram.LinearBuckets(start: 1000, width: 1000, count: 6)}) ;         metricsHub.AddHistogram("/order", orderHistogram);//summary var orderSummary = Metrics.CreateSummary("business_order_summary", "10分钟内的订单数量",new SummaryConfiguration{Objectives = new[]{new QuantileEpsilonPair(0.1, 0.05),   new QuantileEpsilonPair(0.3, 0.05),      new QuantileEpsilonPair(0.5, 0.05),new QuantileEpsilonPair(0.7, 0.05),           new QuantileEpsilonPair(0.9, 0.05),}});metricsHub.AddSummary("/order", orderSummary);services.AddSingleton(metricsHub);}}
}

其实 new QuantileEpsilonPair(0.1, 0.05) 第一个参数是百分位,0.05是误差,范围是10%-5%,10%+5%。

最后一步,就是打开Grafana来配置展示图表了。

最终展示结果:

同时事例中给出了最大、最少、平均、汇总、当前值以供参考。

asp.net core监控—引入Prometheus(五)相关推荐

  1. asp.net core监控—引入Prometheus(六)

    在前面的系列博文中,我们说自定义业务计数器步骤: 1.分析业务,规划好监控跟踪指标 2.定义指标收集器 3.侵入编程(尽量在开发时分离业务实现与监控指票的收集代码)收集指标 4.开发grafana展示 ...

  2. asp.net core监控—引入Prometheus(二)

    上一篇博文中,说明了怎么引进Prometheus到asp.net core项目中,因为是Demo,所以Prometheus和Grafana都是windows版本,本地执行的,生产环境上这些服务可以根据 ...

  3. asp.net core监控—引入Prometheus(一)

    Prometheus是CNCF毕业的第二个项目,算是明星产品(可自行了解Prometheus的功能),asp.net core当然不能错过与之配套使用.在.net中是通过prometheus.net[ ...

  4. asp.net core监控—引入Prometheus(四)

    上一篇博文中说到Prometheus有四种指标类型:Counter(计数器).Gauge(仪表盘).Histogram(直方图).Summary(摘要),并且我们做了一个Counter的Demo,接下 ...

  5. ASP.NET Core 2.1 : 十五.图解路由(2.1 or earler)

    原文:ASP.NET Core 2.1 : 十五.图解路由(2.1 or earler) 本文通过一张图来看一下路由的配置以及请求处理的机制.(ASP.NET Core 系列目录) 一.概述 路由主要 ...

  6. ASP.NET Core 2.1 : 十五.图解路由(2.1 or earler)(转)

    ASP.NET Core 系列目录 本文通过一张图来看一下路由的配置以及请求处理的机制. 一.概述 路由主要有两个主要功能: 将请求的URL与已定义的路由进行匹配,找到该URL对应的处理程序并传入该请 ...

  7. 学习ASP.NET Core Razor 编程系列五——Asp.Net Core Razor新建模板页面

    学习ASP.NET Core Razor 编程系列目录 学习ASP.NET Core Razor 编程系列一 学习ASP.NET Core Razor 编程系列二--添加一个实体 学习ASP.NET ...

  8. ASP.NET Core微服务(一)——【完整API搭建及访问过程】

    ASP.NET Core微服务(一)--[完整API搭建及访问过程]: 环境:win10专业版+vs2019+sqlserver2014/2019 对应练习demo下载路径(1积分):[https:/ ...

  9. 学习ASP.NET Core Razor 编程系列九——增加查询功能

    原文:学习ASP.NET Core Razor 编程系列九--增加查询功能 学习ASP.NET Core Razor 编程系列目录 学习ASP.NET Core Razor 编程系列一 学习ASP.N ...

最新文章

  1. 七大科技巨头的最新人工智能布局
  2. ZOJ 1423 (Your)((Term)((Project))) (模拟+数据结构)
  3. Linux记录-普通用户下执行sudo xxx 找不到命令解决方案
  4. 肝一波 ~ 手写一个简易版的Mybatis,带你深入领略它的魅力!
  5. 20 条非常实用的 Python 代码,建议收藏!
  6. 【机器学习】传统目标检测算法之HOG
  7. LeetCode 285. 二叉搜索树中的顺序后继(中序遍历)
  8. 往map里的vector添加_面试官问我同步容器(如Vector)的所有操作一定是线程安全的吗?我懵了!...
  9. java动态代理_Java中的动态代理一
  10. adb devices 找不到夜神模拟器解决方法
  11. Linux原生日志系统Rsyslog详解
  12. 数据库设计多表关系、范式
  13. 关于for丶foreach丶iterator 迭代器
  14. Ubuntu快速安装或更新chrome
  15. 懂车帝:2018汽车行业大数据报告(附下载)
  16. oracle取去年的最后一天,oracle本月、上月、去年同月第一天最后一天
  17. 将台式机组成云服务器_四种旧PC台式电脑改造桌面云虚拟化的方案介绍
  18. 电脑微信聊天记录迁移——备份与恢复
  19. yolox的正负样本分配策略mmdet代码详解
  20. 云队友丨一战打败马云,四十岁的黄峥究竟有多恐怖?

热门文章

  1. 全向轮底盘磁导轨寻迹
  2. JIL 编译与 AOT 编译
  3. container 的背后
  4. 基于tiny4412的Linux内核移植 -- MMA7660驱动移植(九)
  5. Comparison of video container formats
  6. android studio no marked region found along edge Found along top edge
  7. Scala:First Steps in Scala
  8. 基于JavaScript技术的横排文字转古书式竖排工具
  9. Kinect开发笔记之四检测并调试Kinect设备
  10. nodejs和Vue和Idea