监控系统简介(二):使用 App Metrics 在 ASP.NET Web API 中记录指标

回顾

在《监控系统简介:使用 Prometheus 与 Grafana》一文中,我们了解了什么是监控系统,Prometheus 这一监控工具及它提供的数据类型、PromQL 以及 Grafana 可视化工具的基本用法。今天这一篇我们将在 ASP.NET Web API 项目中进行实战,将 Web API 接口的请求次数、响应耗时、错误率等指标记录下来,并提供给 Prometheus 和 Grafana,用于分析和呈现。

我们主要采用一个名为 App Metrics 的类库记录指标。App Metrics 是以 Apache v2 协议开源的一款类库,支持 .NET Framework 4.5.2 以上,以及 .NET Core 的应用程序。除了记录各种程序生成的指标,它还提供健康检查的功能,但这不在本文的范围内。

为什么没有使用 Prometheus 推荐的 .NET 类库,主要是因为 App Metrics 在 GitHub 的 star 比较多,另外 API 用起来比较顺手而已……

本文示例代码已提交至 Github https://github.com/huhubun/AppMetricsPrometheusSample 欢迎一同讨论。

在 ASP.NET Web API 中记录指标

因为还有一些项目在 .NET Framework 下,所以先以 .NET Framework 的 ASP.NET Web API 开始,通过 Visual Studio 创建“ASP.NET Web 应用程序(.NET Framework)”,框架版本高于或等于 .NET Framework 4.5.2 即可,然后选择 “Web API”。

首先,通过 nuget,将 App Metrics 添加至项目中

Install-Package App.Metrics
Install-Package App.Metrics.Formatters.Prometheus

App Metrics 支持各种各样的监控系统或时序数据库。因为我们最终要将数据提供给 Prometheus,所以除了 App Metrics 的包外,还需要安装一个用于格式化数据的包 App.Metrics.Formatters.Prometheus

由于这是一个新建的项目,简单起见这里创建一个名为 ApiMetrics 的类,保证 Web API 整个生命周期中只初始化一次 App Metrics。如果项目中有依赖注入容器(例如 AutoFac),则直接将 IMetricsRoot 注册为单例即可(通过 InitAppMetrics() 的代码来创建)。

public class ApiMetrics
{private static IMetricsRoot _metrics;public static IMetricsRoot GetMetrics(){if (_metrics == null){_metrics = InitAppMetrics();}return _metrics;}private static IMetricsRoot InitAppMetrics(){var metrics = new MetricsBuilder().Configuration.Configure(options =>{options.DefaultContextLabel = "API";options.AddAppTag(Assembly.GetExecutingAssembly().GetName().Name);options.AddServerTag(Environment.MachineName);#if DEBUGoptions.AddEnvTag("Dev");
#elseoptions.AddEnvTag("Release");
#endifoptions.GlobalTags.Add("my_custom_tag", "MyCustomValue");}).Build();return metrics;}
}
  1. DefaultContextLabel 的值会成为指标的前缀,这里设置成 API,则默认所有指标都为 api_ 开头

  2. AddAppTag() 会为所有指标添加一个名为 app 的 tag,内容为当前程序的名称

  3. AddServerTag() 会为所有指标添加一个名为 server 的 tag,内容是运行程序的机器名称

  4. AddEnvTag() 会为所有指标添加一个名为 env 的 tag,用于区分运行程序的环境

  5. 也可以通过 GlobalTags 属性,来添加自定义的 tag

因为没有依赖注入容器,还需要在 Global.asax 的 Application_Start() 中手动调用一下 GetMetrics() 方法以完成初始化。

protected void Application_Start()
{// 省略其他内容ApiMetrics.GetMetrics();
}

记录程序启动时间

我们把程序启动的时间作为一项指标,在 Grafana 中就能显示出程序已经运行了多长时间。Prometheus 通过 time() 能得到当前时间的 unix 时间戳,所以我们只需要将程序启动时的时间以 unix 时间戳的方式记录下来即可。

在 Application_Start() 中,当一切准备就绪后通过 App Metrics 创建一个 Gauge:

    var metrics = ApiMetrics.GetMetrics();    // 如果有依赖注入容器,请替换为注入 IMetricsRoot 的代码var unixTimestamp = DateTimeOffset.UtcNow.ToUnixTimeSeconds();metrics.Measure.Gauge.SetValue(new GaugeOptions{Name = "Boot Time Seconds"}, unixTimestamp);

通过 App Metrics 的 Measure 属性可以找到 Gauge 属性,然后通过 SetValue() 方法即可记录指标。指标的各种设置(例如名称)通过参数传入。指标名称 Name 我习惯按可读性高的方式来写,因为 App Metrics 的 Prometheus 格式化器会自动帮我们处理它,后文会说明。

另外,虽然我们创建的是 Gauge,但对于启动时间而言,除了这时的赋值外,这个指标的值是不会改变的。

添加 /metrics 终结点

现在我们已经有一个内容为程序启动时间的指标了,还缺少一个能让 Prometheus 抓取指标数据的地方。因为这是一个 Web API 项目,很简单来创建一个 Web API 控制器 MetricsController

    [RoutePrefix("metrics")]public class MetricsController : ApiController{[HttpGet][Route("")]public async Task<HttpResponseMessage> GetMetricsAsync(){var formatter = new App.Metrics.Formatters.Prometheus.MetricsPrometheusTextOutputFormatter();var snapshot = ApiMetrics.GetMetrics().Snapshot.Get();using (var ms = new MemoryStream()){await formatter.WriteAsync(ms, snapshot);var result = Encoding.UTF8.GetString(ms.ToArray());var response = Request.CreateResponse(HttpStatusCode.OK);response.Content = new StringContent(result, Encoding.UTF8, formatter.MediaType.ContentType);return response;}}}

现在启动程序,访问 localhost:端口/metrics 就能看到类似这样的效果:

# HELP api_boot_time_seconds
# TYPE api_boot_time_seconds gauge
api_boot_time_seconds{app="WebAPISample",server="BUNPC",env="Dev",my_custom_tag="MyCustonValue"} 1580913792

App Metrics 的指标类型及转换

由于 App Metrics 的指标类型与 Prometheus 的并不是一一对应的,我们先看看 App Metrics 中提供的类型有哪些:

  • Apdex 应用性能指数评分,它的含义可以参考 《应用性能指标apdex》 https://www.cnblogs.com/tetu/p/4968666.html

  • Counter 计数器

  • Gauge gauge

  • Histogram 直方图

  • Meter 一个可增减的计数器,一般用于统计次数和速率

  • Timer 计时器,根据统计的时间,自动进行分组

可以看到,ApdexMeter 和 Timer 是 Prometheus 中没有的。通过 App.Metrics.Formatters.Prometheus 可以转换成 Prometheus 的指标:

  • Apdex -> Gauge

  • Counter -> Counter

  • Gauge -> Gauge

  • Histogram -> Histogram

  • Meter -> Counter,用起来和 Counter 好像也没什么区别…

  • Timer -> Summary,会自动帮我们计算好 0.5、0.75、0.95、0.99 的分位数

还需要提到的是,通过 App Metrics Prometheus 格式化器,指标的名称也会发生变化,指标名称 Boot Time Seconds 会被转换为 api_boot_time_seconds,空格会自动变为下划线,大写也会被转为小写。所以代码中可以按习惯的方式编写,只要统一即可。

App Metrics 的 API

在 IMetricsRoot 下,我们常用的有这两个属性:

  • Measure

  • Provider

通过 Measure 和 Provider 属性都可以访问到所有的指标类型,仔细观察可以发现, 通过 Measure 操作指标,方法返回的都是 XXXContext 或者 void,而 Provider 返回的都是 IXXX,来看看方法的定义:

  • void IMetricsRoot.Measure.Counter.Increment(CounterOptions options, long amount),只能通过参数列表直接传入值

  • ICounter IMetricsRoot.Provider.Counter.Instance(CounterOptions options),可以对该计数器执行 Increment() 增加值、Decrement() 减少值、Reset() 重置等操作(当然,Prometheus 的计数器应该是只增不减的,但因为 App Metrics 并不是专为 Prometheus 设计,所以它的 API 可以这样操作也是可以理解的)

总的来说,区别在于 Measure 中的 API 相当于去测量某些指标,而 Provider 的 API 可以直接为指标赋值。通过 Timer 来看更为明显:

  • void IMetricsRoot.Measure.Timer.Time(TimerOptions options, Action action) 要求将要统计时间的操作,直接在 Action 中执行,这个 API 会自动开始计时,当 Action 执行完毕后停止计时

  • TimerContext IMetricsRoot.Measure.Timer.Time(TimerOptions options) 当创建 TimerContext 后开始计时,通过 TimerContext 提供的 Dispose() 方法来停止计时

  • ITimer IMetricsRoot.Provider.Timer.Instance(TimerOptions options) 通过 Record() 直接设置时间,另外也有 StartRecording()EndRecording() 等方法手动开始和停止计时

记录 API 响应耗时和请求次数

在 Web API 中,可以通过消息处理程序在请求进入控制器之前,以及响应被生成后,执行一些操作。我们可以通过一个计时器,在收到请求时计时,处理完请求后停止计时的方式,统计一次 HTTP 请求所需要的时间。

确定计时的方案后,需要确定维度。对于 API 的响应耗时,我们应该关注 API 的请求方式(GET、POST、PUT、DELETE等)、API 的路由(/api/values/api/values/{id}等)、响应状态码这些信息。所以需要在指标中,体现出这几个标签。

最后确认使用何种数据类型。App Metrics 提供了 Timer 类型,能自动生成 0.5、0.99 等分位数,并且转换为 Prometheus 后,它是 summary 类型,意味着还会产生 XXX_sum 和 XXX_count 两个指标。通过 XXX_count ,我们顺便还能把请求次数给计算出来。

新建一个 MetricsHandler 类,代码如下:

    public class MetricsHandler : DelegatingHandler{private const string API_METRICS_RESPONSE_TIME_KEY = "__ApiMetrics.ResponseTime__";private const string API_METRICS_ROUTE = "metrics";protected async override Task<HttpResponseMessage> SendAsync(HttpRequestMessage request, CancellationToken cancellationToken){var routeTemplate = GetRouteTemplate(request);// 如果访问的是 /metrics ,则不计入统计中if (routeTemplate == API_METRICS_ROUTE){return await base.SendAsync(request, cancellationToken);}StartRecordingResponseTime(request);var response = await base.SendAsync(request, cancellationToken);EndRecordingResponseTime(routeTemplate, request, response);return response;}private string GetRouteTemplate(HttpRequestMessage request){// MS_SubRoutes 适用于 Route Attribute 的情况request.GetRouteData().Values.TryGetValue("MS_SubRoutes", out var routes);return (routes as System.Web.Http.Routing.IHttpRouteData[])?.FirstOrDefault()?.Route?.RouteTemplate ?? "unknown";}#region Response Time/// <summary>/// 开始记录响应时间/// </summary>/// <param name="request"></param>/// <param name="routeTemplate"></param>private void StartRecordingResponseTime(HttpRequestMessage request){var stopwatch = new Stopwatch();stopwatch.Start();request.Properties.Add(API_METRICS_RESPONSE_TIME_KEY, stopwatch);}/// <summary>/// 停止记录响应时间/// </summary>/// <param name="response"></param>private void EndRecordingResponseTime(string routeTemplate, HttpRequestMessage request, HttpResponseMessage response){var stopwatch = response.RequestMessage.Properties[API_METRICS_RESPONSE_TIME_KEY] as Stopwatch;ApiMetrics.GetMetrics().Provider.Timer.Instance(new TimerOptions{Name = "Response Time",Tags = new MetricTags(new string[] { "method", "route", "status" },new string[] { request.Method.Method, routeTemplate, ((int)response.StatusCode).ToString() }),DurationUnit = TimeUnit.Milliseconds,RateUnit = TimeUnit.Milliseconds,MeasurementUnit = Unit.Requests}).Record(stopwatch.ElapsedMilliseconds, TimeUnit.Milliseconds);response.RequestMessage.Properties.Remove(API_METRICS_RESPONSE_TIME_KEY);}#endregion}

MetricsHandler 的原理是:

  1. 请求进入后,首先触发 StartRecordingResponseTime() 方法,该方法创建了一个 Stopwatch 并开始计时,同时将 Stopwatch 储存在当前请求的缓存中

  2. 等待 await base.SendAsync() 完成,这会执行其它的 Handler、Filter 以及 Action 中的内容,这里执行完成意味着所有的操作都已经完成,并且响应体也已经生成

  3. 触发 EndRecordingResponseTime() 停止计时,并将记录的时间直接储存到 App Metrics 的 Timer 类型的 Response Time 指标中

需要注意的是,GetRouteTemplate() 方法通过 MS_SubRoutes 获取路由的方式仅适用于使用特性路由的方式,根据需要可以使用不同的获取路由的方式。

为了使 MetricsHandler 能正常工作,首先修改默认生成的 ValuesController,将其修改为使用特性路由的方式注册路由:

    [RoutePrefix("api/values")]public class ValuesController : ApiController{// GET api/values[HttpGet, Route("")]public IEnumerable<string> Get(){return new string[] { "value1", "value2" };}// GET api/values/5[HttpGet, Route("{id:int}")]public string Get([FromUri]int id){return "value" + id;}// POST api/values[HttpPost, Route("")]public void Post([FromBody]string value){}// PUT api/values/5[HttpPut, Route("{id:int}")]public void Put([FromUri]int id, [FromBody]string value){}// DELETE api/values/5[HttpDelete, Route("{id:int}")]public void Delete([FromUri]int id){}}

接着修改 WebApiConfig 的 Register() ,将 config.Routes.MapHttpRoute() 路由模板注释掉,然后注册 MetricsHandler。现在 Register() 看起来类似这样:

    public static void Register(HttpConfiguration config){config.MapHttpAttributeRoutes();// 注释掉这部分代码//config.Routes.MapHttpRoute(//    name: "DefaultApi",//    routeTemplate: "api/{controller}/{id}",//    defaults: new { id = RouteParameter.Optional }//);// Metrics Handlerconfig.MessageHandlers.Add(new MetricsHandler());}

完成后我们启动程序,先通过浏览器或者 Postman 随意访问几个接口,例如 localhost:端口/api/values ,之后再访问 /metrics,就能看到我们新增的 api_response_time 指标了:

# HELP api_response_time
# TYPE api_response_time summary
api_response_time_sum{method="GET",route="api/values",status="200",app="WebAPISample",server="BUNPC",env="Dev",my_custom_tag="MyCustomValue"} 0.158
api_response_time_count{method="GET",route="api/values",status="200",app="WebAPISample",server="BUNPC",env="Dev",my_custom_tag="MyCustomValue"} 1
api_response_time{method="GET",route="api/values",status="200",app="WebAPISample",server="BUNPC",env="Dev",my_custom_tag="MyCustomValue",quantile="0.5"} 0.158
api_response_time{method="GET",route="api/values",status="200",app="WebAPISample",server="BUNPC",env="Dev",my_custom_tag="MyCustomValue",quantile="0.75"} 0.158
api_response_time{method="GET",route="api/values",status="200",app="WebAPISample",server="BUNPC",env="Dev",my_custom_tag="MyCustomValue",quantile="0.95"} 0.158
api_response_time{method="GET",route="api/values",status="200",app="WebAPISample",server="BUNPC",env="Dev",my_custom_tag="MyCustomValue",quantile="0.99"} 0.158

虽然我们的例子是基于 .NET Framework 的,但其实对于 .NET Core 而言也是类似。App Metrics 的 API 是一致的, MetricsHandler 由 Middleware 实现即可,这里就不展开说了。

通过 Prometheus 分析

Prometheus 的配置参考上一篇文章,这里直接通过 PromQL 来查询,默认地址为 http://localhost:9090/ 打开 Graph 页面。

计算每个接口总请求数量,因为 api_response_time_count 中包含响应状态,同一个 method 和 route 有时可能返回 200,有时可能返回 400,所以我们需要根据 method 和 route 进行分组再求和:

sum by (method, route)(api_response_time_count)

还可以统计1分钟内的错误率,我们对“错误”的定义为所有非 2XX 的响应,所以非 2 开头的 status 都属于错误:

sum(rate(api_response_time_count{status!~'2.*'}[1m]))

请注意,一定要先 rate() 再 sum(),参考文章 Rate then sum, never sum then rate https://www.robustperception.io/rate-then-sum-never-sum-then-rate

统计每个接口 95% 情况下的响应时间

api_response_time{quantile='0.95'}

与 Grafana 图表结合的例子,可以参考本文 demo 的 https://github.com/huhubun/AppMetricsPrometheusSample

链接

  • App Metrics 官方网站 https://www.app-metrics.io/

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/303440.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

剧透人生!你什么时候结婚换工作甚至狗带,Facebook都知道

来源&#xff1a;大数据文摘即将换工作&#xff1f;要结婚了&#xff1f;有亲人朋友要去世了&#xff1f;你关心的这些人生重大节点&#xff0c;有人希望比你提前知道它们何时发生&#xff0c;并基于此对你精准投放广告。惊悚&#xff1f;这是Facebook在2010年以来提交的一系列…

如何在 NET 程序万种死法中有效的生成 Dump (下)

一&#xff1a;背景 上一篇我们聊到了如何通过 procdump 抓取 cpu爆高 和 内存暴涨 两种情况&#xff0c;这一篇再聊聊如何去抓程序 挂死 和 意外退出。二&#xff1a;程序挂死 1. 定义程序挂死 简单的说就是程序没有响应&#xff0c;既然没响应了&#xff0c;可能 死锁, 可能 …

C++程序运行时内存布局之--无继承情况下的虚函数

2019独角兽企业重金招聘Python工程师标准>>> 虚函数是C实现多态的关键&#xff0c;没有虚函数&#xff0c;C只能是OB&#xff0c;不能完成OO。 本文介绍的是没有继承情况下&#xff0c;带有虚函数的类在内存中布局&#xff0c;以及其实例&#xff08;对象&#xff0…

纠结学哪种编程语言?请看这个回答

各位童鞋&#xff0c;如果最近/未来几年有人还纠结学习哪种编程语言&#xff0c;那你可以请他/她参考一下这个 Quora 帖子的最高赞回答&#xff1a;> 能找一份工作&#xff1a;Java> 能找一份高薪工作&#xff1a;C> 啥样工作都能找到&#xff1a;Python> 总能找到…

NET问答: 有最干净利落的读写文件方式吗?

咨询区 ApprenticeHacker&#xff1a;在 C# 中有很多种读写文件的方式 (文本文件&#xff0c;非二进制)。为了践行 do more, write less 的思想&#xff0c;现寻找一种最简单最少代码量的方式&#xff0c;因为在我的项目中有太多的功能需要读写文件了。回答区 vc 74&#xff1a…

写好一份数据分析报告的13个要点

先说说写一份好的数据分析报告的重要性&#xff0c;很简单&#xff0c;因为分析报告的输出是你整个分析过程的成果&#xff0c;是评定一个产品、一个运营事件的定性结论&#xff0c;很可能是产品决策的参考依据&#xff0c;既然这么重要那当然要写好它了。我认为一份好的分析报…

oracle11g安装和基本的使用,手把手看图教你用起来。

前面技术架构和系统选型说到了使用oracle&#xff0c;看到园友们一些评论。我也不想卖弄&#xff0c;不想争论什么数据库好。喜欢就是理由&#xff0c;用了多年了为何不可。 我可以想象&#xff0c;大家没有使用oracle的原因可能有以下几点&#xff1a; 被oracle价格吓破胆的有…

java高并发类_Java 高并发之魂

前置知识了解Java基本语法了解多线程基本知识知识介绍Synchronized简介&#xff1a;作用、地位、不控制并发的后果两种用法&#xff1a;对象锁和类锁多线程访问同步方法的7种情况&#xff1a;是否是static、Synchronized方法等Synchronized的性质&#xff1a;可重入、不可中断原…

.NET 开源配置组件 AgileConfig 初体验

介绍在微服务大行其道的今天&#xff0c;系统会被拆分成多个模块&#xff0c;作为单独的服务运行&#xff0c;同时为了集中化管理&#xff0c;我们还需要日志中心&#xff0c;配置中心等&#xff0c;很多开发人员可能更熟悉 ApolloConfig&#xff0c;这个组件功能也很完善&…

来领资料咯!计算机专业教科书礼包

现在计算机行业越来越火爆&#xff0c;技术的更新也越来越快&#xff0c;不好好充实自己的话&#xff0c;就会被淘汰。小编这几年一直有意识地收集计算机行业相关的书籍&#xff0c;现在已经拥有5G左右的资料&#xff0c;如C、Java、Python、机器学习、网页开发Web Dev、数据分…

Asp.Net Core 5 REST API - Step by Step(一)

翻译自 Mohamad Lawand 2021年1月19日的文章 《Asp.Net Core 5 Rest API Step by Step》 [1]在本文中&#xff0c;我们将创建一个简单的 Asp.Net Core REST API Todo 应用程序&#xff0c;在其中我们可以添加、编辑、删除和查看待办事项&#xff0c;并且将使用 SQLite 来存储数…

身为程序员碰到最奇葩的需求是怎样的?

梁大折腾部门老大&#xff1a;你&#xff0c;做个微信小游戏吧我&#xff1a;啥样的&#xff1f;老大&#xff1a;反正你就做个小游戏吧&#xff0c;这个火我&#xff1a;&#xff1f;&#xff1f;&#xff1f;&#xff1f;&#xff1f;柔情领导:那个运维啊&#xff0c;你来把他…

15个未来高科技产品会让你无法想象!这些开脑洞的设计太牛了!

导读&#xff1a;从衣食住行到生活的方方面面&#xff0c;未来必将会有天翻地覆的变化。大数据、云计算、物联网和人工智能这些年的发展&#xff0c;让我们对并不遥远的未来有了更多想象和期待。那些我们现阶段不可企及的所思所想&#xff0c;将在未来成为大部分人的日常。这么…

解决 .NET Core 在 Linux Container 中获取 CurrentCulture 不正确的问题

背景在将公司一款基于 .NET Framework 的控制台程序迁移到 .NET Core 3.1 时&#xff0c;发现程序中本地化的部分失效&#xff0c;症状类似于对 Thread.CurrentThread.CurrentCulture.Name 的值进行 Substring() 操作时抛出 ArgumentOutOfRangeException 异常。该程序在 Window…

机器学习资料升级版来了!!!

机器学习一直是一个热门的领域。上次分享的机器学习资料【资源】机器学习资料包来袭受到大家的广泛好评&#xff0c;今天小编打算分享一份机器学习升级版的资料&#xff0c;有斯坦福大学-深度学习基础教程、机器学习实战、人工智能与大数据、Tagging等。顺序最下优化算法Error …

K-Means算法的10个有趣用例

K-means算法具有悠久的历史&#xff0c;并且也是最常用的聚类算法之一。K-means算法实施起来非常简单&#xff0c;因此&#xff0c;它非常适用于机器学习新手爱好者。首先我们来回顾K-Means算法的起源&#xff0c;然后介绍其较为典型的应用场景。起源1967年&#xff0c;James M…

C# $的用法

今天闲来无事&#xff0c;就随便在网上乱看&#xff0c;突然想到Jquery的$符号很强大&#xff0c;那么C#有没有这个东西呢&#xff0c;一查&#xff0c;果然有。经查证发现&#xff0c;这个是在C#6.0出现的一个新特性&#xff0c;也就是一个小语法糖&#xff0c;其作用相当于对…

想转行人工智能?机会来了!!!

一个坏消息&#xff1a;2018年1月 教育部印发的《普通高中课程方案和语文等学科课程标准》新加入了数据结构、人工智能、开源硬件设计等 AI 相关的课程。这意味着职场新人和准备找工作的同学们&#xff0c;为了在今后十年内不被淘汰&#xff0c;你们要补课了&#xff0c;从初中…

tankwar java_TankWar 单机(JAVA版) 版本0.3 画出坦克

其实就是通过自定义的panel 重写里面的paint方法 使用Graphics类画一个圆然后把自定的panel添加到窗口中由于想到tank不能只画一个 所以我们封装一个tank类 里面有一个draw方法 用来画 坦克通过实例化tank 调用draw方法就能实现画tank了具体代码如下&#xff1a;Tank类pac…

轻量NuGet服务—BaGet

相信大家都受益过nuget.org&#xff0c;上面的海量的库、工具、模板为我们开发提供了极大的帮助&#xff0c;其中有很多都是非常宽松的开源协议&#xff0c;在此感谢那无私奉献的人。有的时候&#xff0c;在企业内部&#xff0c;有些库是私有的&#xff0c;专项的&#xff0c;要…