千家信息网

SQL函数计算流量的95值

发表于:2025-01-23 作者:千家信息网编辑
千家信息网最后更新 2025年01月23日,我用的是Solarwinds系统,部分内容就结合Solarwinds系统一起写了。最后落地也是通过系统的Report定时自动出报告并且可以发邮件。不过计算方法是通过定义SQL函数,然后使用SQL查询来
千家信息网最后更新 2025年01月23日SQL函数计算流量的95值

我用的是Solarwinds系统,部分内容就结合Solarwinds系统一起写了。最后落地也是通过系统的Report定时自动出报告并且可以发邮件。
不过计算方法是通过定义SQL函数,然后使用SQL查询来获取到的,这部分内容是通用的。

95th 计算方法

从 Solarwinds 官方网站搜索"95th"关键字能获取到说明的文档。
95th Percentile Calculations in the Orion Platform:
https://documentation.solarwinds.com/en/Success_Center/orionplatform/Content/Core-95th-Percentile-Calculations-sw80.htm

  1. Over the 10 hours, the following 120 values were collected for inbound traffic (Mb/s):

    0.149 0.623 0.281 0.136 0.024 0.042 0.097 0.185 0.198 0.243 0.274 0.390 0.971 0.633 0.238 0.142 0.119 0.176 0.131 0.127 0.169 0.223 0.291 0.236 0.124 0.072 0.197 0.105 0.138 0.233 0.374 0.290 0.871 0.433 0.248 0.242 0.169 0.116 0.121 0.427 0.249 0.223 0.231 0.336 0.014 0.442 0.197 0.125 0.108 0.244 0.264 0.190 0.471 0.033 0.228 0.942 0.219 0.076 0.331 0.227 0.849 0.323 0.221 0.196 0.223 0.642 0.197 0.385 0.098 0.263 0.174 0.690 0.571 0.233 0.208 0.242 0.139 0.186 0.331 0.124 0.249 0.643 0.481 0.936 0.124 0.742 0.497 0.085 0.398 0.643 0.074 0.590 0.771 0.833 0.438 0.242 0.092 0.376 0.231 0.627 0.249 0.663 0.181 0.636 0.224 0.342 0.697 0.285 0.108 0.211 0.074 0.490 0.271 0.133 0.338 0.242 0.519 0.376 0.331 0.227 
  2. The values are reordered from high to low.

    0.971 0.942 0.936 0.871 0.849 0.833 0.771 0.742 0.697 0.690 0.663 0.643 0.643 0.642 0.636 0.633 0.627 0.623 0.590 0.571 0.519 0.497 0.490 0.481 0.471 0.442 0.438 0.433 0.427 0.398 0.390 0.385 0.376 0.376 0.374 0.342 0.338 0.336 0.331 0.331 0.331 0.323 0.291 0.290 0.285 0.281 0.274 0.271 0.264 0.263 0.249 0.249 0.249 0.248 0.244 0.243 0.242 0.242 0.242 0.242 0.238 0.236 0.233 0.233 0.231 0.231 0.228 0.227 0.227 0.224 0.223 0.223 0.223 0.221 0.219 0.211 0.208 0.198 0.197 0.197 0.197 0.196 0.190 0.186 0.185 0.181 0.176 0.174 0.169 0.169 0.149 0.142 0.139 0.138 0.136 0.133 0.131 0.127 0.125 0.124 0.124 0.124 0.121 0.119 0.116 0.108 0.108 0.105 0.098 0.097 0.092 0.085 0.076 0.074 0.074 0.072 0.042 0.033 0.024 0.014 
  3. The first 6 values are dropped, as these equal the top 5% of the values.

    0.771 0.742 0.697 0.690 0.663 0.643 0.643 0.642 0.636 0.633 0.627 0.623 0.590 0.571 0.519 0.497 0.490 0.481 0.471 0.442 0.438 0.433 0.427 0.398 0.390 0.385 0.376 0.376 0.374 0.342 0.338 0.336 0.331 0.331 0.331 0.323 0.291 0.290 0.285 0.281 0.274 0.271 0.264 0.263 0.249 0.249 0.249 0.248 0.244 0.243 0.242 0.242 0.242 0.242 0.238 0.236 0.233 0.233 0.231 0.231 0.228 0.227 0.227 0.224 0.223 0.223 0.223 0.221 0.219 0.211 0.208 0.198 0.197 0.197 0.197 0.196 0.190 0.186 0.185 0.181 0.176 0.174 0.169 0.169 0.149 0.142 0.139 0.138 0.136 0.133 0.131 0.127 0.125 0.124 0.124 0.124 0.121 0.119 0.116 0.108 0.108 0.105 0.098 0.097 0.092 0.085 0.076 0.074 0.074 0.072 0.042 0.033 0.024 0.014 
  4. The 95th percentile is 0.771.

SQL 函数

系统上安装的工具是:SQL Server Management Studio

新建函数的命令模板

新建内联表值函数:

-- ================================================-- Template generated from Template Explorer using:-- Create Inline Function (New Menu).SQL---- Use the Specify Values for Template Parameters -- command (Ctrl-Shift-M) to fill in the parameter -- values below.---- This block of comments will not be included in-- the definition of the function.-- ================================================SET ANSI_NULLS ONGOSET QUOTED_IDENTIFIER ONGO-- =============================================-- Author:      -- Create date: -- Description: -- =============================================CREATE FUNCTION  (       -- Add the parameters for the function here    <@param1, sysname, @p1> ,     <@param2, sysname, @p2> )RETURNS TABLE ASRETURN (    -- Add the SELECT statement with parameter references here    SELECT 0)GO

新建多语句表值函数:

-- ================================================-- Template generated from Template Explorer using:-- Create Multi-Statement Function (New Menu).SQL---- Use the Specify Values for Template Parameters -- command (Ctrl-Shift-M) to fill in the parameter -- values below.---- This block of comments will not be included in-- the definition of the function.-- ================================================SET ANSI_NULLS ONGOSET QUOTED_IDENTIFIER ONGO-- =============================================-- Author:      -- Create date: -- Description: -- =============================================CREATE FUNCTION  (    -- Add the parameters for the function here    <@param1, sysname, @p1> ,     <@param2, sysname, @p2> )RETURNS <@Table_Variable_Name, sysname, @Table_Var> TABLE (    -- Add the column definitions for the TABLE variable here     ,      )ASBEGIN    -- Fill the table variable with the rows for your result set    RETURN ENDGO

新建标量值函数:

-- ================================================-- Template generated from Template Explorer using:-- Create Scalar Function (New Menu).SQL---- Use the Specify Values for Template Parameters -- command (Ctrl-Shift-M) to fill in the parameter -- values below.---- This block of comments will not be included in-- the definition of the function.-- ================================================SET ANSI_NULLS ONGOSET QUOTED_IDENTIFIER ONGO-- =============================================-- Author:      -- Create date: -- Description: -- =============================================CREATE FUNCTION  (    -- Add the parameters for the function here    <@Param1, sysname, @p1> )RETURNS ASBEGIN    -- Declare the return variable here    DECLARE <@ResultVar, sysname, @Result>     -- Add the T-SQL statements to compute the return value here    SELECT <@ResultVar, sysname, @Result> = <@Param1, sysname, @p1>    -- Return the result of the function    RETURN <@ResultVar, sysname, @Result>ENDGO

Solarwinds系统使用的计算95th的函数

通过标量值函数来实现的。
一共有3个,一个In方向GetInBps95th,一个Out方向GetOutBps95th。
还有一个单个时间点的In和Out方向取大值计算结果。
函数已经在系统中了,这里是修改函数的命令模板。

计算In方向的95th

USE [SolarwindsOrion]GO/****** Object:  UserDefinedFunction [dbo].[GetInBps95th]    Script Date: 2019/11/14 11:07:12 ******/SET ANSI_NULLS ONGOSET QUOTED_IDENTIFIER ONGOALTER FUNCTION [dbo].[GetInBps95th] (    @InterfaceId int,    @StartDate DateTime,    @EndDate DateTime)RETURNS realASBEGIN    DECLARE @ResultVar real    SELECT @ResultVar = MAX(In_Maxbps)    FROM (        SELECT TOP 95 PERCENT In_Maxbps        FROM dbo.InterfaceTraffic WITH (NOLOCK)        WHERE InterfaceID = @InterfaceId AND DateTime >= @StartDate AND DateTime <= @EndDate        ORDER BY In_Maxbps ASC    ) AS AA    RETURN @ResultVarEND

Out方向的只是换一个字段,其他都一样。

计算双向的95th

USE [SolarwindsOrion]GO/****** Object:  UserDefinedFunction [dbo].[GetMaxBps95th]    Script Date: 2019/11/14 11:08:42 ******/SET ANSI_NULLS ONGOSET QUOTED_IDENTIFIER ONGOALTER FUNCTION [dbo].[GetMaxBps95th] (    @InterfaceId int,    @StartDate DateTime,    @EndDate DateTime)RETURNS realASBEGIN    DECLARE @ResultVar real    SELECT @ResultVar = MAX(Maxbps)    FROM (        SELECT TOP 95 PERCENT Maxbps        FROM (SELECT (CASE WHEN Out_Maxbps > In_Maxbps THEN Out_Maxbps ELSE In_Maxbps END) AS Maxbps            FROM dbo.InterfaceTraffic WITH (NOLOCK)            WHERE InterfaceID = @InterfaceId AND DateTime >= @StartDate AND DateTime <= @EndDate) AS MaxbpsSet        ORDER BY Maxbps ASC    ) AS AA    RETURN @ResultVarEND

多了一层子查询,使用WHEN子句用来判断取大的值。

自定义函数

因为原有的函数不能满足需求,需要改一下,增加一个时间段筛选的变量。需求就是一周或者一个月,每天只取工作时间的数据。
这里按时间的小时数 'DATEPART(hh,DateTime)' 只要是9到16这些数值就表示每天的9点到17点了。
另外还有一个时间段的需求是每天有2段时间,考虑到通用性,这里用了数组的方式来设置,把需要的小时的数值定义在数组里。实际没有用数组,用字符串来模拟了。

创建自定义函数

创建一个自定义函数,在原有的函数的基础上,增加时间段的筛选:

USE [SolarwindsOrion]GOSET ANSI_NULLS ONGOSET QUOTED_IDENTIFIER ONGO-- =============================================-- Author:      Steed Xu-- Create date: 2019/11/14-- Description: Base on GetInBps95th, add BusyHours.-- =============================================CREATE FUNCTION [dbo].[GetInBps95thBusyHours] (    @InterfaceId int,    @StartDate DateTime,    @EndDate DateTime,    @BusyHours VarChar(64))RETURNS realASBEGIN    DECLARE @ResultVar real    SELECT @ResultVar = MAX(In_Maxbps)    FROM (        SELECT TOP 95 PERCENT In_Maxbps        FROM dbo.InterfaceTraffic WITH (NOLOCK)        WHERE InterfaceID = @InterfaceId             AND DateTime >= @StartDate AND DateTime <= @EndDate            AND ','+RTRIM(@BusyHours)+',' LIKE '%,'+CAST(DATEPART(hh,DateTime) AS varchar)+',%'        ORDER BY In_Maxbps ASC    ) AS AA    RETURN @ResultVarENDGO

执行上面的语句后,就会添加到数据库中。
注意:这里定义字符串变量的时候要指定大小,否则虽然可以成功执行,但是结果会和预期的不一样。

解决数组的问题

筛选方式使用取时间的小时数,一天里哪几个时段是需要的,就定义在数组里。
使用的时候这样定义:

DECLARE @BusyHours VarChar(64)SET @BusyHours = '9,10,11,12,13,14,15,16'

这样就是要取9点到17点的数据。

对应的WHERE可以这样写:

WHERE ','+RTrim(@BusyHours)+',' LIKE '%,'+CAST(datepart(hh,DateTime) AS VarChar)+',%'

基本用字符串模拟出了数组的效果,在这里是够用了。

修改自定义函数

如果创建的时候写错了,则可以用下面的模板进行修改,更新到数据库中。

USE [SolarwindsOrion]GO/****** Object:  UserDefinedFunction [dbo].[GetInBps95thBusyHours]    Script Date: 2019/11/14 13:43:21 ******/SET ANSI_NULLS ONGOSET QUOTED_IDENTIFIER ONGO-- =============================================-- Author:      Steed Xu-- Create date: 2019/11/14-- Description: Base on GetInBps95th, add BusyHours.-- =============================================ALTER FUNCTION [dbo].[GetInBps95thBusyHours](    @InterfaceId int,    @StartDate DateTime,    @EndDate DateTime,    @BusyHours VarChar(64))RETURNS realASBEGIN    DECLARE @ResultVar real    SELECT @ResultVar = MAX(In_Maxbps)    FROM (        SELECT TOP 95 PERCENT In_Maxbps        FROM dbo.InterfaceTraffic WITH (NOLOCK)        WHERE InterfaceID = @InterfaceId             AND DateTime >= @StartDate AND DateTime <= @EndDate             AND ','+RTRIM(@BusyHours)+',' LIKE '%,'+CAST(DATEPART(hh,DateTime) AS varchar)+',%'        ORDER BY In_Maxbps ASC    ) AS AA    RETURN @ResultVarEND

在SQL中调用函数

供应商给的使用模板

完整的SQL查询语句:

完整的查询语句如下:SET NOCOUNT OFFSET ROWCOUNT 0DECLARE @StartDate DateTimeDECLARE @EndDate DateTimeSET @StartDate = CAST((ROUND(CAST(GetDate() - 7 AS FLOAT), 0, 1)) as datetime)SET @EndDate = GetDate()SELECT Interfaces.InterfaceId,    Nodes.NodeID,    Nodes.Caption AS NodeName,    Nodes.VendorIcon AS Vendor_Icon,    Interfaces.Caption AS Interface_Caption,    Interfaces.InterfaceIcon AS Interface_Icon,    Maxbps_In95,    Maxbps_Out95,    Maxbps_95FROM NodesINNER JOIN Interfaces ON Nodes.NodeID = Interfaces.NodeIDINNER JOIN (    SELECT InterfaceID,        dbo.GetInBps95th(InterfaceID, @StartDate, @EndDate) AS Maxbps_In95,        dbo.GetOutBps95th(InterfaceID, @StartDate, @EndDate) AS Maxbps_Out95,        dbo.GetMaxBps95th(InterfaceID, @StartDate, @EndDate) AS Maxbps_95    FROM InterfaceTraffic    WHERE InterfaceTraffic.DateTime >= @StartDate AND InterfaceTraffic.DateTime <= @EndDate    GROUP BY InterfaceID) TrafficStatON Interfaces.InterfaceID = TrafficStat.InterfaceIDWHERE (1=1)AND  (    (Nodes.Vendor = 'Cisco') AND    (Interfaces.Comments = 'MT'))ORDER BY Maxbps_In95 desc,Maxbps_Out95 desc

按需求修改的查询语句

需求是要计算一周的数据,这里是获取上一周的数据,从周一开始。
完整的查询语句:

SET NOCOUNT OFFSET ROWCOUNT 0DECLARE @StartDate DateTimeDECLARE @EndDate DateTimeDECLARE @BusyHours VarChar(64)SET @StartDate = DATEADD(week,-1,DATEADD(week,DATEDIFF(week,0,getdate()),0))SET @EndDate = DATEADD(week,DATEDIFF(week,0,getdate()),0)SET @BusyHours = '9,10,11,12,13,14,15,16'SELECT Interfaces.InterfaceId,    Nodes.NodeID,    Nodes.Caption AS NodeName,    Nodes.VendorIcon AS Vendor_Icon,    Nodes.IP_Address,    Interfaces.InterfaceName AS Interface_Name,    Interfaces.InterfaceAlias AS Description,    Interfaces.InterfaceIcon AS Interface_Icon,    Maxbps_In95,    Maxbps_Out95,    @StartDate AS StartDate,    @EndDate AS EndDate,    @BusyHours AS BusyHours,    Interfaces.Status AS Interfaces_StatusFROM NodesINNER JOIN Interfaces ON Nodes.NodeID = Interfaces.NodeIDINNER JOIN (    SELECT InterfaceID,        dbo.GetInBps95thBusyHours(InterfaceID, @StartDate, @EndDate, @BusyHours) AS Maxbps_In95,        dbo.GetOutBps95thBusyHours(InterfaceID, @StartDate, @EndDate, @BusyHours) AS Maxbps_Out95    FROM InterfaceTraffic    WHERE InterfaceTraffic.DateTime >= @StartDate        AND InterfaceTraffic.DateTime <= @EndDate        AND ','+RTRIM(@BusyHours)+',' LIKE '%,'+CAST(DATEPART(hh,DateTime) AS varchar)+',%'    GROUP BY InterfaceID) TrafficStatON Interfaces.InterfaceId = TrafficStat.InterfaceIdWHERE (1=1)    AND Nodes.IP_Address IN ('172.16.6.1','172.16.6.2','172.16.5.3','172.16.10.1','172.16.10.2','172.16.5.4')    AND Interfaces.InterfaceAlias <> ''    AND Interfaces.Status IN (1,2)ORDER BY Interfaces.InterfaceId

注意:这里定义字符串变量的时候要指定大小,否则虽然可以成功执行,但是结果会和预期的不一样。

验证计算结果

在写一个查询语句,把一个端口下的所有的流量数据都导出。验证函数的结果是否符合预期。
完整的查询语句:

SET NOCOUNT OFFSET ROWCOUNT 0DECLARE @StartDate DateTimeDECLARE @EndDate DateTimeDECLARE @BusyHours VarChar(64)DECLARE @InterfaceId intSET @StartDate = DATEADD(week,-1,DATEADD(week,DATEDIFF(week,0,getdate()),0))SET @EndDate = DATEADD(week,DATEDIFF(week,0,getdate()),0)SET @BusyHours = '9,10,11,12,13,14,15,16'SET @InterfaceId = 28993SELECT TOP 95 PERCENT     DateTime, InterfaceID, In_Maxbps, Out_Maxbps,     @StartDate AS StartDate,    @EndDate AS EndDate,    @BusyHours AS BusyHoursFROM InterfaceTrafficWHERE InterfaceTraffic.DateTime >= @StartDate    AND InterfaceTraffic.DateTime <= @EndDate    AND ','+RTrim(@BusyHours)+',' LIKE '%,'+CAST(datepart(hh,DateTime) AS VarChar)+',%'    AND InterfaceID = @InterfaceIdORDER BY In_Maxbps ASC
0