t-digest
t-digest is a probabilistic data structure that allows you to estimate the percentile of a data stream.
The t-digest is a sketch data structure in Redis Stack for estimating percentiles from a data stream or a large dataset using a compact sketch.
It can answer questions like:
Which fraction of the values in the data stream are smaller than a given value?
How many values in the data stream are smaller than a given value?
What's the highest value that's smaller than p percent of the values in the data stream? (what is the p-percentile value)?
What is t-digest?
t-digest is a data structure that will estimate a percentile point without having to store and order all the data points in a set. For example: to answer the question "What's the average latency for 99% of my database operations" we would have to store the average latency for every user, order the values, cut out the last 1% and only then find the average value of all the rest. This kind of process is costly not just in terms of the processing needed to order those values but also in terms of the space needed to store them. Those are precisely the problems t-digest solves.
t-digest can also be used to estimate other values related to percentiles, like trimmed means.
A trimmed mean is the mean value from the sketch, excluding observation values outside the low and high cutoff percentiles. For example, a 0.1 trimmed mean is the mean value of the sketch, excluding the lowest 10% and the highest 10% of the values.
Use cases
Hardware/software monitoring
You measure your online server response latency, and you like to query:
What are the 50th, 90th, and 99th percentiles of the measured latencies?
Which fraction of the measured latencies are less than 25 milliseconds?
What is the mean latency, ignoring outliers? or What is the mean latency between the 10th and the 90th percentile?
Online gaming
Millions of people are playing a game on your online gaming platform, and you want to give the following information to each player?
Your score is better than x percent of the game sessions played.
There were about y game sessions where people scored larger than you.
To have a better score than 90% of the games played, your score should be z.
Network traffic monitoring
You measure the IP packets transferred over your network each second and try to detect denial-of-service attacks by asking:
Does the number of packets in the last second exceed 99% of previously observed values?
How many packets do I expect to see under normal network conditions?
(Answer: between x and y, where x represents the 1st percentile and y represents the 99th percentile).
Predictive maintenance
Was the measured parameter (noise level, current consumption, etc.) irregular? (not within the [1st percentile...99th percentile] range)?
To which values should I set my alerts?
Examples
In the following example, you'll create a t-digest with a compression of 100 and add items to it. The COMPRESSION
argument is used to specify the tradeoff between accuracy and memory consumption. The default value is 100. Higher values mean more accuracy. Note: unlike some of the other probabilistic data structures, the TDIGEST.ADD
command will not create a new structure if the key does not exist.
>_ Redis CLI
> TDIGEST.CREATE bikes:sales COMPRESSION 100
OK
> TDIGEST.ADD bikes:sales 21
OK
> TDIGEST.ADD bikes:sales 150 95 75 34
OK
Copied!
Python
"""
Code samples for t-digest pages:
https://redis.io/docs/latest/develop/data-types/probabilistic/t-digest/
"""
import redis
r = redis . Redis ( decode_responses = True )
res1 = r . tdigest () . create ( "bikes:sales" , 100 )
print ( res1 ) # >>> True
res2 = r . tdigest () . add ( "bikes:sales" , [ 21 ])
print ( res2 ) # >>> OK
res3 = r . tdigest () . add ( "bikes:sales" , [ 150 , 95 , 75 , 34 ])
print ( res3 ) # >>> OK
res4 = r . tdigest () . create ( "racer_ages" )
print ( res4 ) # >>> True
res5 = r . tdigest () . add (
"racer_ages" ,
[
45.88 ,
44.2 ,
58.03 ,
19.76 ,
39.84 ,
69.28 ,
50.97 ,
25.41 ,
19.27 ,
85.71 ,
42.63 ,
],
)
print ( res5 ) # >>> OK
res6 = r . tdigest () . rank ( "racer_ages" , 50 )
print ( res6 ) # >>> [7]
res7 = r . tdigest () . rank ( "racer_ages" , 50 , 40 )
print ( res7 ) # >>> [7, 4]
res8 = r . tdigest () . quantile ( "racer_ages" , 0.5 )
print ( res8 ) # >>> [44.2]
res9 = r . tdigest () . byrank ( "racer_ages" , 4 )
print ( res9 ) # >>> [42.63]
res10 = r . tdigest () . min ( "racer_ages" )
print ( res10 ) # >>> 19.27
res11 = r . tdigest () . max ( "racer_ages" )
print ( res11 ) # >>> 85.71
res12 = r . tdigest () . reset ( "racer_ages" )
print ( res12 ) # >>> OK
Copied!
Node.js
import assert from 'assert' ;
import { createClient } from 'redis' ;
const client = createClient ();
await client . connect ();
const res1 = await client . tDigest . create ( 'bikes:sales' , 100 );
console . log ( res1 ); // >>> OK
const res2 = await client . tDigest . add ( 'bikes:sales' , [ 21 ]);
console . log ( res2 ); // >>> OK
const res3 = await client . tDigest . add ( 'bikes:sales' , [ 150 , 95 , 75 , 34 ]);
console . log ( res3 ); // >>> OK
const res4 = await client . tDigest . create ( 'racer_ages' );
console . log ( res4 ); // >>> OK
const res5 = await client . tDigest . add ( 'racer_ages' , [
45.88 , 44.2 , 58.03 , 19.76 , 39.84 , 69.28 , 50.97 , 25.41 , 19.27 , 85.71 , 42.63
]);
console . log ( res5 ); // >>> OK
const res6 = await client . tDigest . rank ( 'racer_ages' , [ 50 ]);
console . log ( res6 ); // >>> [7]
const res7 = await client . tDigest . rank ( 'racer_ages' , [ 50 , 40 ]);
console . log ( res7 ); // >>> [7, 4]
const res8 = await client . tDigest . quantile ( 'racer_ages' , [ 0.5 ]);
console . log ( res8 ); // >>> [44.2]
const res9 = await client . tDigest . byRank ( 'racer_ages' , [ 4 ]);
console . log ( res9 ); // >>> [42.63]
const res10 = await client . tDigest . min ( 'racer_ages' );
console . log ( res10 ); // >>> 19.27
const res11 = await client . tDigest . max ( 'racer_ages' );
console . log ( res11 ); // >>> 85.71
const res12 = await client . tDigest . reset ( 'racer_ages' );
console . log ( res12 ); // >>> OK
Copied!
Java
package io.redis.examples ;
public class TDigestExample {
public void run (){
UnifiedJedis unifiedJedis = new UnifiedJedis ( "redis://127.0.0.1:6379" );
String res1 = unifiedJedis . tdigestCreate ( "bikes:sales" , 100 );
System . out . println ( res1 ); // >>> True
String res2 = unifiedJedis . tdigestAdd ( "bikes:sales" , 21 );
System . out . println ( res2 ); // >>> OK
String res3 = unifiedJedis . tdigestAdd ( "bikes:sales" , 150 , 95 , 75 , 34 );
System . out . println ( res3 ); // >>> OK
String res4 = unifiedJedis . tdigestCreate ( "racer_ages" );
System . out . println ( res4 ); // >>> True
String res5 = unifiedJedis . tdigestAdd ( "racer_ages" , 45.88 ,
44.2 ,
58.03 ,
19.76 ,
39.84 ,
69.28 ,
50.97 ,
25.41 ,
19.27 ,
85.71 ,
42.63 );
System . out . println ( res5 ); // >>> OK
List < Long > res6 = unifiedJedis . tdigestRank ( "racer_ages" , 50 );
System . out . println ( res6 ); // >>> [7]
List < Long > res7 = unifiedJedis . tdigestRank ( "racer_ages" , 50 , 40 );
System . out . println ( res7 ); // >>> [7, 4]
List < Double > res8 = unifiedJedis . tdigestQuantile ( "racer_ages" , 0.5 );
System . out . println ( res8 ); // >>> [44.2]
List < Double > res9 = unifiedJedis . tdigestByRank ( "racer_ages" , 4 );
System . out . println ( res9 ); // >>> [42.63]
double res10 = unifiedJedis . tdigestMin ( "racer_ages" );
System . out . println ( res10 ); // >>> 19.27
double res11 = unifiedJedis . tdigestMax ( "racer_ages" );
System . out . println ( res11 ); // >>> 85.71
String res12 = unifiedJedis . tdigestReset ( "racer_ages" );
System . out . println ( res12 ); // >>> OK
}
}
Copied!
C#
using NRedisStack.RedisStackCommands ;
using NRedisStack.Tests ;
using StackExchange.Redis ;
public class Tdigest_tutorial
{
[SkipIfRedis(Is.OSSCluster)]
public void run ()
{
var muxer = ConnectionMultiplexer . Connect ( "localhost:6379" );
var db = muxer . GetDatabase ();
bool res1 = db . TDIGEST (). Create ( "bikes:sales" , 100 );
Console . WriteLine ( res1 ); // >>> True
bool res2 = db . TDIGEST (). Add ( "bikes:sales" , 21 );
Console . WriteLine ( res2 ); // >>> True
bool res3 = db . TDIGEST (). Add ( "bikes:sales" , 150 , 95 , 75 , 34 );
Console . WriteLine ( res3 ); // >>> true
// Tests for 'tdig_start' step.
bool res4 = db . TDIGEST (). Create ( "racer_ages" );
Console . WriteLine ( res4 ); // >>> True
bool res5 = db . TDIGEST (). Add ( "racer_ages" ,
45.88 ,
44.2 ,
58.03 ,
19.76 ,
39.84 ,
69.28 ,
50.97 ,
25.41 ,
19.27 ,
85.71 ,
42.63
);
Console . WriteLine ( res5 ); // >>> True
long [] res6 = db . TDIGEST (). Rank ( "racer_ages" , 50 );
Console . WriteLine ( string . Join ( ", " , res6 )); // >>> 7
long [] res7 = db . TDIGEST (). Rank ( "racer_ages" , 50 , 40 );
Console . WriteLine ( string . Join ( ", " , res7 )); // >>> 7, 4
// Tests for 'tdig_cdf' step.
double [] res8 = db . TDIGEST (). Quantile ( "racer_ages" , 0.5 ); ;
Console . WriteLine ( string . Join ( ", " , res8 )); // >>> 44.2
double [] res9 = db . TDIGEST (). ByRank ( "racer_ages" , 4 );
Console . WriteLine ( string . Join ( ", " , res9 )); // >>> 42.63
// Tests for 'tdig_quant' step.
double res10 = db . TDIGEST (). Min ( "racer_ages" );
Console . WriteLine ( res10 ); // >>> 19.27
double res11 = db . TDIGEST (). Max ( "racer_ages" );
Console . WriteLine ( res11 ); // >>> 85.71
// Tests for 'tdig_min' step.
bool res12 = db . TDIGEST (). Reset ( "racer_ages" );
Console . WriteLine ( res12 ); // >>> True
// Tests for 'tdig_reset' step.
}
}
Copied!
You can repeat calling TDIGEST.ADD whenever new observations are available
Estimating fractions or ranks by values
Another helpful feature in t-digest is CDF (definition of rank) which gives us the fraction of observations smaller or equal to a certain value. This command is very useful to answer questions like "What's the percentage of observations with a value lower or equal to X ".
More precisely, TDIGEST.CDF
will return the estimated fraction of observations in the sketch that are smaller than X plus half the number of observations that are equal to X. We can also use the TDIGEST.RANK
command, which is very similar. Instead of returning a fraction, it returns the ----estimated---- rank of a value. The TDIGEST.RANK
command is also variadic, meaning you can use a single command to retrieve estimations for one or more values.
Here's an example. Given a set of biker's ages, you can ask a question like "What's the percentage of bike racers that are younger than 50 years?"
>_ Redis CLI
> TDIGEST.CREATE racer_ages
OK
> TDIGEST.ADD racer_ages 45.88 44.2 58.03 19.76 39.84 69.28 50.97 25.41 19.27 85.71 42.63
OK
> TDIGEST.CDF racer_ages 50
1) "0.63636363636363635"
> TDIGEST.RANK racer_ages 50
1) (integer) 7
> TDIGEST.RANK racer_ages 50 40
1) (integer) 7
2) (integer) 4
Copied!
Python
"""
Code samples for t-digest pages:
https://redis.io/docs/latest/develop/data-types/probabilistic/t-digest/
"""
import redis
r = redis . Redis ( decode_responses = True )
res1 = r . tdigest () . create ( "bikes:sales" , 100 )
print ( res1 ) # >>> True
res2 = r . tdigest () . add ( "bikes:sales" , [ 21 ])
print ( res2 ) # >>> OK
res3 = r . tdigest () . add ( "bikes:sales" , [ 150 , 95 , 75 , 34 ])
print ( res3 ) # >>> OK
res4 = r . tdigest () . create ( "racer_ages" )
print ( res4 ) # >>> True
res5 = r . tdigest () . add (
"racer_ages" ,
[
45.88 ,
44.2 ,
58.03 ,
19.76 ,
39.84 ,
69.28 ,
50.97 ,
25.41 ,
19.27 ,
85.71 ,
42.63 ,
],
)
print ( res5 ) # >>> OK
res6 = r . tdigest () . rank ( "racer_ages" , 50 )
print ( res6 ) # >>> [7]
res7 = r . tdigest () . rank ( "racer_ages" , 50 , 40 )
print ( res7 ) # >>> [7, 4]
res8 = r . tdigest () . quantile ( "racer_ages" , 0.5 )
print ( res8 ) # >>> [44.2]
res9 = r . tdigest () . byrank ( "racer_ages" , 4 )
print ( res9 ) # >>> [42.63]
res10 = r . tdigest () . min ( "racer_ages" )
print ( res10 ) # >>> 19.27
res11 = r . tdigest () . max ( "racer_ages" )
print ( res11 ) # >>> 85.71
res12 = r . tdigest () . reset ( "racer_ages" )
print ( res12 ) # >>> OK
Copied!
Node.js
import assert from 'assert' ;
import { createClient } from 'redis' ;
const client = createClient ();
await client . connect ();
const res1 = await client . tDigest . create ( 'bikes:sales' , 100 );
console . log ( res1 ); // >>> OK
const res2 = await client . tDigest . add ( 'bikes:sales' , [ 21 ]);
console . log ( res2 ); // >>> OK
const res3 = await client . tDigest . add ( 'bikes:sales' , [ 150 , 95 , 75 , 34 ]);
console . log ( res3 ); // >>> OK
const res4 = await client . tDigest . create ( 'racer_ages' );
console . log ( res4 ); // >>> OK
const res5 = await client . tDigest . add ( 'racer_ages' , [
45.88 , 44.2 , 58.03 , 19.76 , 39.84 , 69.28 , 50.97 , 25.41 , 19.27 , 85.71 , 42.63
]);
console . log ( res5 ); // >>> OK
const res6 = await client . tDigest . rank ( 'racer_ages' , [ 50 ]);
console . log ( res6 ); // >>> [7]
const res7 = await client . tDigest . rank ( 'racer_ages' , [ 50 , 40 ]);
console . log ( res7 ); // >>> [7, 4]
const res8 = await client . tDigest . quantile ( 'racer_ages' , [ 0.5 ]);
console . log ( res8 ); // >>> [44.2]
const res9 = await client . tDigest . byRank ( 'racer_ages' , [ 4 ]);
console . log ( res9 ); // >>> [42.63]
const res10 = await client . tDigest . min ( 'racer_ages' );
console . log ( res10 ); // >>> 19.27
const res11 = await client . tDigest . max ( 'racer_ages' );
console . log ( res11 ); // >>> 85.71
const res12 = await client . tDigest . reset ( 'racer_ages' );
console . log ( res12 ); // >>> OK
Copied!
Java
package io.redis.examples ;
public class TDigestExample {
public void run (){
UnifiedJedis unifiedJedis = new UnifiedJedis ( "redis://127.0.0.1:6379" );
String res1 = unifiedJedis . tdigestCreate ( "bikes:sales" , 100 );
System . out . println ( res1 ); // >>> True
String res2 = unifiedJedis . tdigestAdd ( "bikes:sales" , 21 );
System . out . println ( res2 ); // >>> OK
String res3 = unifiedJedis . tdigestAdd ( "bikes:sales" , 150 , 95 , 75 , 34 );
System . out . println ( res3 ); // >>> OK
String res4 = unifiedJedis . tdigestCreate ( "racer_ages" );
System . out . println ( res4 ); // >>> True
String res5 = unifiedJedis . tdigestAdd ( "racer_ages" , 45.88 ,
44.2 ,
58.03 ,
19.76 ,
39.84 ,
69.28 ,
50.97 ,
25.41 ,
19.27 ,
85.71 ,
42.63 );
System . out . println ( res5 ); // >>> OK
List < Long > res6 = unifiedJedis . tdigestRank ( "racer_ages" , 50 );
System . out . println ( res6 ); // >>> [7]
List < Long > res7 = unifiedJedis . tdigestRank ( "racer_ages" , 50 , 40 );
System . out . println ( res7 ); // >>> [7, 4]
List < Double > res8 = unifiedJedis . tdigestQuantile ( "racer_ages" , 0.5 );
System . out . println ( res8 ); // >>> [44.2]
List < Double > res9 = unifiedJedis . tdigestByRank ( "racer_ages" , 4 );
System . out . println ( res9 ); // >>> [42.63]
double res10 = unifiedJedis . tdigestMin ( "racer_ages" );
System . out . println ( res10 ); // >>> 19.27
double res11 = unifiedJedis . tdigestMax ( "racer_ages" );
System . out . println ( res11 ); // >>> 85.71
String res12 = unifiedJedis . tdigestReset ( "racer_ages" );
System . out . println ( res12 ); // >>> OK
}
}
Copied!
C#
using NRedisStack.RedisStackCommands ;
using NRedisStack.Tests ;
using StackExchange.Redis ;
public class Tdigest_tutorial
{
[SkipIfRedis(Is.OSSCluster)]
public void run ()
{
var muxer = ConnectionMultiplexer . Connect ( "localhost:6379" );
var db = muxer . GetDatabase ();
bool res1 = db . TDIGEST (). Create ( "bikes:sales" , 100 );
Console . WriteLine ( res1 ); // >>> True
bool res2 = db . TDIGEST (). Add ( "bikes:sales" , 21 );
Console . WriteLine ( res2 ); // >>> True
bool res3 = db . TDIGEST (). Add ( "bikes:sales" , 150 , 95 , 75 , 34 );
Console . WriteLine ( res3 ); // >>> true
// Tests for 'tdig_start' step.
bool res4 = db . TDIGEST (). Create ( "racer_ages" );
Console . WriteLine ( res4 ); // >>> True
bool res5 = db . TDIGEST (). Add ( "racer_ages" ,
45.88 ,
44.2 ,
58.03 ,
19.76 ,
39.84 ,
69.28 ,
50.97 ,
25.41 ,
19.27 ,
85.71 ,
42.63
);
Console . WriteLine ( res5 ); // >>> True
long [] res6 = db . TDIGEST (). Rank ( "racer_ages" , 50 );
Console . WriteLine ( string . Join ( ", " , res6 )); // >>> 7
long [] res7 = db . TDIGEST (). Rank ( "racer_ages" , 50 , 40 );
Console . WriteLine ( string . Join ( ", " , res7 )); // >>> 7, 4
// Tests for 'tdig_cdf' step.
double [] res8 = db . TDIGEST (). Quantile ( "racer_ages" , 0.5 ); ;
Console . WriteLine ( string . Join ( ", " , res8 )); // >>> 44.2
double [] res9 = db . TDIGEST (). ByRank ( "racer_ages" , 4 );
Console . WriteLine ( string . Join ( ", " , res9 )); // >>> 42.63
// Tests for 'tdig_quant' step.
double res10 = db . TDIGEST (). Min ( "racer_ages" );
Console . WriteLine ( res10 ); // >>> 19.27
double res11 = db . TDIGEST (). Max ( "racer_ages" );
Console . WriteLine ( res11 ); // >>> 85.71
// Tests for 'tdig_min' step.
bool res12 = db . TDIGEST (). Reset ( "racer_ages" );
Console . WriteLine ( res12 ); // >>> True
// Tests for 'tdig_reset' step.
}
}
Copied!
And lastly, TDIGEST.REVRANK key value...
is similar to TDIGEST.RANK , but returns, for each input value, an estimation of the number of (observations larger than a given value + half the observations equal to the given value).
Estimating values by fractions or ranks
TDIGEST.QUANTILE key fraction...
returns, for each input fraction, an estimation of the value (floating point) that is smaller than the given fraction of observations. TDIGEST.BYRANK key rank...
returns, for each input rank, an estimation of the value (floating point) with that rank.
>_ Redis CLI
> TDIGEST.QUANTILE racer_ages .5
1) "44.200000000000003"
> TDIGEST.BYRANK racer_ages 4
1) "42.630000000000003"
Copied!
Python
"""
Code samples for t-digest pages:
https://redis.io/docs/latest/develop/data-types/probabilistic/t-digest/
"""
import redis
r = redis . Redis ( decode_responses = True )
res1 = r . tdigest () . create ( "bikes:sales" , 100 )
print ( res1 ) # >>> True
res2 = r . tdigest () . add ( "bikes:sales" , [ 21 ])
print ( res2 ) # >>> OK
res3 = r . tdigest () . add ( "bikes:sales" , [ 150 , 95 , 75 , 34 ])
print ( res3 ) # >>> OK
res4 = r . tdigest () . create ( "racer_ages" )
print ( res4 ) # >>> True
res5 = r . tdigest () . add (
"racer_ages" ,
[
45.88 ,
44.2 ,
58.03 ,
19.76 ,
39.84 ,
69.28 ,
50.97 ,
25.41 ,
19.27 ,
85.71 ,
42.63 ,
],
)
print ( res5 ) # >>> OK
res6 = r . tdigest () . rank ( "racer_ages" , 50 )
print ( res6 ) # >>> [7]
res7 = r . tdigest () . rank ( "racer_ages" , 50 , 40 )
print ( res7 ) # >>> [7, 4]
res8 = r . tdigest () . quantile ( "racer_ages" , 0.5 )
print ( res8 ) # >>> [44.2]
res9 = r . tdigest () . byrank ( "racer_ages" , 4 )
print ( res9 ) # >>> [42.63]
res10 = r . tdigest () . min ( "racer_ages" )
print ( res10 ) # >>> 19.27
res11 = r . tdigest () . max ( "racer_ages" )
print ( res11 ) # >>> 85.71
res12 = r . tdigest () . reset ( "racer_ages" )
print ( res12 ) # >>> OK
Copied!
Node.js
import assert from 'assert' ;
import { createClient } from 'redis' ;
const client = createClient ();
await client . connect ();
const res1 = await client . tDigest . create ( 'bikes:sales' , 100 );
console . log ( res1 ); // >>> OK
const res2 = await client . tDigest . add ( 'bikes:sales' , [ 21 ]);
console . log ( res2 ); // >>> OK
const res3 = await client . tDigest . add ( 'bikes:sales' , [ 150 , 95 , 75 , 34 ]);
console . log ( res3 ); // >>> OK
const res4 = await client . tDigest . create ( 'racer_ages' );
console . log ( res4 ); // >>> OK
const res5 = await client . tDigest . add ( 'racer_ages' , [
45.88 , 44.2 , 58.03 , 19.76 , 39.84 , 69.28 , 50.97 , 25.41 , 19.27 , 85.71 , 42.63
]);
console . log ( res5 ); // >>> OK
const res6 = await client . tDigest . rank ( 'racer_ages' , [ 50 ]);
console . log ( res6 ); // >>> [7]
const res7 = await client . tDigest . rank ( 'racer_ages' , [ 50 , 40 ]);
console . log ( res7 ); // >>> [7, 4]
const res8 = await client . tDigest . quantile ( 'racer_ages' , [ 0.5 ]);
console . log ( res8 ); // >>> [44.2]
const res9 = await client . tDigest . byRank ( 'racer_ages' , [ 4 ]);
console . log ( res9 ); // >>> [42.63]
const res10 = await client . tDigest . min ( 'racer_ages' );
console . log ( res10 ); // >>> 19.27
const res11 = await client . tDigest . max ( 'racer_ages' );
console . log ( res11 ); // >>> 85.71
const res12 = await client . tDigest . reset ( 'racer_ages' );
console . log ( res12 ); // >>> OK
Copied!
Java
package io.redis.examples ;
public class TDigestExample {
public void run (){
UnifiedJedis unifiedJedis = new UnifiedJedis ( "redis://127.0.0.1:6379" );
String res1 = unifiedJedis . tdigestCreate ( "bikes:sales" , 100 );
System . out . println ( res1 ); // >>> True
String res2 = unifiedJedis . tdigestAdd ( "bikes:sales" , 21 );
System . out . println ( res2 ); // >>> OK
String res3 = unifiedJedis . tdigestAdd ( "bikes:sales" , 150 , 95 , 75 , 34 );
System . out . println ( res3 ); // >>> OK
String res4 = unifiedJedis . tdigestCreate ( "racer_ages" );
System . out . println ( res4 ); // >>> True
String res5 = unifiedJedis . tdigestAdd ( "racer_ages" , 45.88 ,
44.2 ,
58.03 ,
19.76 ,
39.84 ,
69.28 ,
50.97 ,
25.41 ,
19.27 ,
85.71 ,
42.63 );
System . out . println ( res5 ); // >>> OK
List < Long > res6 = unifiedJedis . tdigestRank ( "racer_ages" , 50 );
System . out . println ( res6 ); // >>> [7]
List < Long > res7 = unifiedJedis . tdigestRank ( "racer_ages" , 50 , 40 );
System . out . println ( res7 ); // >>> [7, 4]
List < Double > res8 = unifiedJedis . tdigestQuantile ( "racer_ages" , 0.5 );
System . out . println ( res8 ); // >>> [44.2]
List < Double > res9 = unifiedJedis . tdigestByRank ( "racer_ages" , 4 );
System . out . println ( res9 ); // >>> [42.63]
double res10 = unifiedJedis . tdigestMin ( "racer_ages" );
System . out . println ( res10 ); // >>> 19.27
double res11 = unifiedJedis . tdigestMax ( "racer_ages" );
System . out . println ( res11 ); // >>> 85.71
String res12 = unifiedJedis . tdigestReset ( "racer_ages" );
System . out . println ( res12 ); // >>> OK
}
}
Copied!
C#
using NRedisStack.RedisStackCommands ;
using NRedisStack.Tests ;
using StackExchange.Redis ;
public class Tdigest_tutorial
{
[SkipIfRedis(Is.OSSCluster)]
public void run ()
{
var muxer = ConnectionMultiplexer . Connect ( "localhost:6379" );
var db = muxer . GetDatabase ();
bool res1 = db . TDIGEST (). Create ( "bikes:sales" , 100 );
Console . WriteLine ( res1 ); // >>> True
bool res2 = db . TDIGEST (). Add ( "bikes:sales" , 21 );
Console . WriteLine ( res2 ); // >>> True
bool res3 = db . TDIGEST (). Add ( "bikes:sales" , 150 , 95 , 75 , 34 );
Console . WriteLine ( res3 ); // >>> true
// Tests for 'tdig_start' step.
bool res4 = db . TDIGEST (). Create ( "racer_ages" );
Console . WriteLine ( res4 ); // >>> True
bool res5 = db . TDIGEST (). Add ( "racer_ages" ,
45.88 ,
44.2 ,
58.03 ,
19.76 ,
39.84 ,
69.28 ,
50.97 ,
25.41 ,
19.27 ,
85.71 ,
42.63
);
Console . WriteLine ( res5 ); // >>> True
long [] res6 = db . TDIGEST (). Rank ( "racer_ages" , 50 );
Console . WriteLine ( string . Join ( ", " , res6 )); // >>> 7
long [] res7 = db . TDIGEST (). Rank ( "racer_ages" , 50 , 40 );
Console . WriteLine ( string . Join ( ", " , res7 )); // >>> 7, 4
// Tests for 'tdig_cdf' step.
double [] res8 = db . TDIGEST (). Quantile ( "racer_ages" , 0.5 ); ;
Console . WriteLine ( string . Join ( ", " , res8 )); // >>> 44.2
double [] res9 = db . TDIGEST (). ByRank ( "racer_ages" , 4 );
Console . WriteLine ( string . Join ( ", " , res9 )); // >>> 42.63
// Tests for 'tdig_quant' step.
double res10 = db . TDIGEST (). Min ( "racer_ages" );
Console . WriteLine ( res10 ); // >>> 19.27
double res11 = db . TDIGEST (). Max ( "racer_ages" );
Console . WriteLine ( res11 ); // >>> 85.71
// Tests for 'tdig_min' step.
bool res12 = db . TDIGEST (). Reset ( "racer_ages" );
Console . WriteLine ( res12 ); // >>> True
// Tests for 'tdig_reset' step.
}
}
Copied!
TDIGEST.BYREVRANK key rank...
returns, for each input reverse rank , an estimation of the value (floating point) with that reverse rank.
Estimating trimmed mean
Use TDIGEST.TRIMMED_MEAN key lowFraction highFraction
to retrieve an estimation of the mean value between the specified fractions.
This is especially useful for calculating the average value ignoring outliers. For example - calculating the average value between the 20th percentile and the 80th percentile.
Merging sketches
Sometimes it is useful to merge sketches. For example, suppose we measure latencies for 3 servers, and we want to calculate the 90%, 95%, and 99% latencies for all the servers combined.
TDIGEST.MERGE destKey numKeys sourceKey... [COMPRESSION compression] [OVERRIDE]
merges multiple sketches into a single sketch.
If destKey
does not exist - a new sketch is created.
If destKey
is an existing sketch, its values are merged with the values of the source keys. To override the destination key contents, use OVERRIDE
.
Use TDIGEST.MIN
and TDIGEST.MAX
to retrieve the minimal and maximal values in the sketch, respectively.
>_ Redis CLI
> TDIGEST.MIN racer_ages
"19.27"
> TDIGEST.MAX racer_ages
"85.709999999999994"
Copied!
Python
"""
Code samples for t-digest pages:
https://redis.io/docs/latest/develop/data-types/probabilistic/t-digest/
"""
import redis
r = redis . Redis ( decode_responses = True )
res1 = r . tdigest () . create ( "bikes:sales" , 100 )
print ( res1 ) # >>> True
res2 = r . tdigest () . add ( "bikes:sales" , [ 21 ])
print ( res2 ) # >>> OK
res3 = r . tdigest () . add ( "bikes:sales" , [ 150 , 95 , 75 , 34 ])
print ( res3 ) # >>> OK
res4 = r . tdigest () . create ( "racer_ages" )
print ( res4 ) # >>> True
res5 = r . tdigest () . add (
"racer_ages" ,
[
45.88 ,
44.2 ,
58.03 ,
19.76 ,
39.84 ,
69.28 ,
50.97 ,
25.41 ,
19.27 ,
85.71 ,
42.63 ,
],
)
print ( res5 ) # >>> OK
res6 = r . tdigest () . rank ( "racer_ages" , 50 )
print ( res6 ) # >>> [7]
res7 = r . tdigest () . rank ( "racer_ages" , 50 , 40 )
print ( res7 ) # >>> [7, 4]
res8 = r . tdigest () . quantile ( "racer_ages" , 0.5 )
print ( res8 ) # >>> [44.2]
res9 = r . tdigest () . byrank ( "racer_ages" , 4 )
print ( res9 ) # >>> [42.63]
res10 = r . tdigest () . min ( "racer_ages" )
print ( res10 ) # >>> 19.27
res11 = r . tdigest () . max ( "racer_ages" )
print ( res11 ) # >>> 85.71
res12 = r . tdigest () . reset ( "racer_ages" )
print ( res12 ) # >>> OK
Copied!
Node.js
import assert from 'assert' ;
import { createClient } from 'redis' ;
const client = createClient ();
await client . connect ();
const res1 = await client . tDigest . create ( 'bikes:sales' , 100 );
console . log ( res1 ); // >>> OK
const res2 = await client . tDigest . add ( 'bikes:sales' , [ 21 ]);
console . log ( res2 ); // >>> OK
const res3 = await client . tDigest . add ( 'bikes:sales' , [ 150 , 95 , 75 , 34 ]);
console . log ( res3 ); // >>> OK
const res4 = await client . tDigest . create ( 'racer_ages' );
console . log ( res4 ); // >>> OK
const res5 = await client . tDigest . add ( 'racer_ages' , [
45.88 , 44.2 , 58.03 , 19.76 , 39.84 , 69.28 , 50.97 , 25.41 , 19.27 , 85.71 , 42.63
]);
console . log ( res5 ); // >>> OK
const res6 = await client . tDigest . rank ( 'racer_ages' , [ 50 ]);
console . log ( res6 ); // >>> [7]
const res7 = await client . tDigest . rank ( 'racer_ages' , [ 50 , 40 ]);
console . log ( res7 ); // >>> [7, 4]
const res8 = await client . tDigest . quantile ( 'racer_ages' , [ 0.5 ]);
console . log ( res8 ); // >>> [44.2]
const res9 = await client . tDigest . byRank ( 'racer_ages' , [ 4 ]);
console . log ( res9 ); // >>> [42.63]
const res10 = await client . tDigest . min ( 'racer_ages' );
console . log ( res10 ); // >>> 19.27
const res11 = await client . tDigest . max ( 'racer_ages' );
console . log ( res11 ); // >>> 85.71
const res12 = await client . tDigest . reset ( 'racer_ages' );
console . log ( res12 ); // >>> OK
Copied!
Java
package io.redis.examples ;
public class TDigestExample {
public void run (){
UnifiedJedis unifiedJedis = new UnifiedJedis ( "redis://127.0.0.1:6379" );
String res1 = unifiedJedis . tdigestCreate ( "bikes:sales" , 100 );
System . out . println ( res1 ); // >>> True
String res2 = unifiedJedis . tdigestAdd ( "bikes:sales" , 21 );
System . out . println ( res2 ); // >>> OK
String res3 = unifiedJedis . tdigestAdd ( "bikes:sales" , 150 , 95 , 75 , 34 );
System . out . println ( res3 ); // >>> OK
String res4 = unifiedJedis . tdigestCreate ( "racer_ages" );
System . out . println ( res4 ); // >>> True
String res5 = unifiedJedis . tdigestAdd ( "racer_ages" , 45.88 ,
44.2 ,
58.03 ,
19.76 ,
39.84 ,
69.28 ,
50.97 ,
25.41 ,
19.27 ,
85.71 ,
42.63 );
System . out . println ( res5 ); // >>> OK
List < Long > res6 = unifiedJedis . tdigestRank ( "racer_ages" , 50 );
System . out . println ( res6 ); // >>> [7]
List < Long > res7 = unifiedJedis . tdigestRank ( "racer_ages" , 50 , 40 );
System . out . println ( res7 ); // >>> [7, 4]
List < Double > res8 = unifiedJedis . tdigestQuantile ( "racer_ages" , 0.5 );
System . out . println ( res8 ); // >>> [44.2]
List < Double > res9 = unifiedJedis . tdigestByRank ( "racer_ages" , 4 );
System . out . println ( res9 ); // >>> [42.63]
double res10 = unifiedJedis . tdigestMin ( "racer_ages" );
System . out . println ( res10 ); // >>> 19.27
double res11 = unifiedJedis . tdigestMax ( "racer_ages" );
System . out . println ( res11 ); // >>> 85.71
String res12 = unifiedJedis . tdigestReset ( "racer_ages" );
System . out . println ( res12 ); // >>> OK
}
}
Copied!
C#
using NRedisStack.RedisStackCommands ;
using NRedisStack.Tests ;
using StackExchange.Redis ;
public class Tdigest_tutorial
{
[SkipIfRedis(Is.OSSCluster)]
public void run ()
{
var muxer = ConnectionMultiplexer . Connect ( "localhost:6379" );
var db = muxer . GetDatabase ();
bool res1 = db . TDIGEST (). Create ( "bikes:sales" , 100 );
Console . WriteLine ( res1 ); // >>> True
bool res2 = db . TDIGEST (). Add ( "bikes:sales" , 21 );
Console . WriteLine ( res2 ); // >>> True
bool res3 = db . TDIGEST (). Add ( "bikes:sales" , 150 , 95 , 75 , 34 );
Console . WriteLine ( res3 ); // >>> true
// Tests for 'tdig_start' step.
bool res4 = db . TDIGEST (). Create ( "racer_ages" );
Console . WriteLine ( res4 ); // >>> True
bool res5 = db . TDIGEST (). Add ( "racer_ages" ,
45.88 ,
44.2 ,
58.03 ,
19.76 ,
39.84 ,
69.28 ,
50.97 ,
25.41 ,
19.27 ,
85.71 ,
42.63
);
Console . WriteLine ( res5 ); // >>> True
long [] res6 = db . TDIGEST (). Rank ( "racer_ages" , 50 );
Console . WriteLine ( string . Join ( ", " , res6 )); // >>> 7
long [] res7 = db . TDIGEST (). Rank ( "racer_ages" , 50 , 40 );
Console . WriteLine ( string . Join ( ", " , res7 )); // >>> 7, 4
// Tests for 'tdig_cdf' step.
double [] res8 = db . TDIGEST (). Quantile ( "racer_ages" , 0.5 ); ;
Console . WriteLine ( string . Join ( ", " , res8 )); // >>> 44.2
double [] res9 = db . TDIGEST (). ByRank ( "racer_ages" , 4 );
Console . WriteLine ( string . Join ( ", " , res9 )); // >>> 42.63
// Tests for 'tdig_quant' step.
double res10 = db . TDIGEST (). Min ( "racer_ages" );
Console . WriteLine ( res10 ); // >>> 19.27
double res11 = db . TDIGEST (). Max ( "racer_ages" );
Console . WriteLine ( res11 ); // >>> 85.71
// Tests for 'tdig_min' step.
bool res12 = db . TDIGEST (). Reset ( "racer_ages" );
Console . WriteLine ( res12 ); // >>> True
// Tests for 'tdig_reset' step.
}
}
Copied!
Both return nan
when the sketch is empty.
Both commands return accurate results and are equivalent to TDIGEST.BYRANK racer_ages 0
and TDIGEST.BYREVRANK racer_ages 0
, respectively.
Use TDIGEST.INFO racer_ages
to retrieve some additional information about the sketch.
Resetting a sketch
>_ Redis CLI
> TDIGEST.RESET racer_ages
OK
Copied!
Python
"""
Code samples for t-digest pages:
https://redis.io/docs/latest/develop/data-types/probabilistic/t-digest/
"""
import redis
r = redis . Redis ( decode_responses = True )
res1 = r . tdigest () . create ( "bikes:sales" , 100 )
print ( res1 ) # >>> True
res2 = r . tdigest () . add ( "bikes:sales" , [ 21 ])
print ( res2 ) # >>> OK
res3 = r . tdigest () . add ( "bikes:sales" , [ 150 , 95 , 75 , 34 ])
print ( res3 ) # >>> OK
res4 = r . tdigest () . create ( "racer_ages" )
print ( res4 ) # >>> True
res5 = r . tdigest () . add (
"racer_ages" ,
[
45.88 ,
44.2 ,
58.03 ,
19.76 ,
39.84 ,
69.28 ,
50.97 ,
25.41 ,
19.27 ,
85.71 ,
42.63 ,
],
)
print ( res5 ) # >>> OK
res6 = r . tdigest () . rank ( "racer_ages" , 50 )
print ( res6 ) # >>> [7]
res7 = r . tdigest () . rank ( "racer_ages" , 50 , 40 )
print ( res7 ) # >>> [7, 4]
res8 = r . tdigest () . quantile ( "racer_ages" , 0.5 )
print ( res8 ) # >>> [44.2]
res9 = r . tdigest () . byrank ( "racer_ages" , 4 )
print ( res9 ) # >>> [42.63]
res10 = r . tdigest () . min ( "racer_ages" )
print ( res10 ) # >>> 19.27
res11 = r . tdigest () . max ( "racer_ages" )
print ( res11 ) # >>> 85.71
res12 = r . tdigest () . reset ( "racer_ages" )
print ( res12 ) # >>> OK
Copied!
Node.js
import assert from 'assert' ;
import { createClient } from 'redis' ;
const client = createClient ();
await client . connect ();
const res1 = await client . tDigest . create ( 'bikes:sales' , 100 );
console . log ( res1 ); // >>> OK
const res2 = await client . tDigest . add ( 'bikes:sales' , [ 21 ]);
console . log ( res2 ); // >>> OK
const res3 = await client . tDigest . add ( 'bikes:sales' , [ 150 , 95 , 75 , 34 ]);
console . log ( res3 ); // >>> OK
const res4 = await client . tDigest . create ( 'racer_ages' );
console . log ( res4 ); // >>> OK
const res5 = await client . tDigest . add ( 'racer_ages' , [
45.88 , 44.2 , 58.03 , 19.76 , 39.84 , 69.28 , 50.97 , 25.41 , 19.27 , 85.71 , 42.63
]);
console . log ( res5 ); // >>> OK
const res6 = await client . tDigest . rank ( 'racer_ages' , [ 50 ]);
console . log ( res6 ); // >>> [7]
const res7 = await client . tDigest . rank ( 'racer_ages' , [ 50 , 40 ]);
console . log ( res7 ); // >>> [7, 4]
const res8 = await client . tDigest . quantile ( 'racer_ages' , [ 0.5 ]);
console . log ( res8 ); // >>> [44.2]
const res9 = await client . tDigest . byRank ( 'racer_ages' , [ 4 ]);
console . log ( res9 ); // >>> [42.63]
const res10 = await client . tDigest . min ( 'racer_ages' );
console . log ( res10 ); // >>> 19.27
const res11 = await client . tDigest . max ( 'racer_ages' );
console . log ( res11 ); // >>> 85.71
const res12 = await client . tDigest . reset ( 'racer_ages' );
console . log ( res12 ); // >>> OK
Copied!
Java
package io.redis.examples ;
public class TDigestExample {
public void run (){
UnifiedJedis unifiedJedis = new UnifiedJedis ( "redis://127.0.0.1:6379" );
String res1 = unifiedJedis . tdigestCreate ( "bikes:sales" , 100 );
System . out . println ( res1 ); // >>> True
String res2 = unifiedJedis . tdigestAdd ( "bikes:sales" , 21 );
System . out . println ( res2 ); // >>> OK
String res3 = unifiedJedis . tdigestAdd ( "bikes:sales" , 150 , 95 , 75 , 34 );
System . out . println ( res3 ); // >>> OK
String res4 = unifiedJedis . tdigestCreate ( "racer_ages" );
System . out . println ( res4 ); // >>> True
String res5 = unifiedJedis . tdigestAdd ( "racer_ages" , 45.88 ,
44.2 ,
58.03 ,
19.76 ,
39.84 ,
69.28 ,
50.97 ,
25.41 ,
19.27 ,
85.71 ,
42.63 );
System . out . println ( res5 ); // >>> OK
List < Long > res6 = unifiedJedis . tdigestRank ( "racer_ages" , 50 );
System . out . println ( res6 ); // >>> [7]
List < Long > res7 = unifiedJedis . tdigestRank ( "racer_ages" , 50 , 40 );
System . out . println ( res7 ); // >>> [7, 4]
List < Double > res8 = unifiedJedis . tdigestQuantile ( "racer_ages" , 0.5 );
System . out . println ( res8 ); // >>> [44.2]
List < Double > res9 = unifiedJedis . tdigestByRank ( "racer_ages" , 4 );
System . out . println ( res9 ); // >>> [42.63]
double res10 = unifiedJedis . tdigestMin ( "racer_ages" );
System . out . println ( res10 ); // >>> 19.27
double res11 = unifiedJedis . tdigestMax ( "racer_ages" );
System . out . println ( res11 ); // >>> 85.71
String res12 = unifiedJedis . tdigestReset ( "racer_ages" );
System . out . println ( res12 ); // >>> OK
}
}
Copied!
C#
using NRedisStack.RedisStackCommands ;
using NRedisStack.Tests ;
using StackExchange.Redis ;
public class Tdigest_tutorial
{
[SkipIfRedis(Is.OSSCluster)]
public void run ()
{
var muxer = ConnectionMultiplexer . Connect ( "localhost:6379" );
var db = muxer . GetDatabase ();
bool res1 = db . TDIGEST (). Create ( "bikes:sales" , 100 );
Console . WriteLine ( res1 ); // >>> True
bool res2 = db . TDIGEST (). Add ( "bikes:sales" , 21 );
Console . WriteLine ( res2 ); // >>> True
bool res3 = db . TDIGEST (). Add ( "bikes:sales" , 150 , 95 , 75 , 34 );
Console . WriteLine ( res3 ); // >>> true
// Tests for 'tdig_start' step.
bool res4 = db . TDIGEST (). Create ( "racer_ages" );
Console . WriteLine ( res4 ); // >>> True
bool res5 = db . TDIGEST (). Add ( "racer_ages" ,
45.88 ,
44.2 ,
58.03 ,
19.76 ,
39.84 ,
69.28 ,
50.97 ,
25.41 ,
19.27 ,
85.71 ,
42.63
);
Console . WriteLine ( res5 ); // >>> True
long [] res6 = db . TDIGEST (). Rank ( "racer_ages" , 50 );
Console . WriteLine ( string . Join ( ", " , res6 )); // >>> 7
long [] res7 = db . TDIGEST (). Rank ( "racer_ages" , 50 , 40 );
Console . WriteLine ( string . Join ( ", " , res7 )); // >>> 7, 4
// Tests for 'tdig_cdf' step.
double [] res8 = db . TDIGEST (). Quantile ( "racer_ages" , 0.5 ); ;
Console . WriteLine ( string . Join ( ", " , res8 )); // >>> 44.2
double [] res9 = db . TDIGEST (). ByRank ( "racer_ages" , 4 );
Console . WriteLine ( string . Join ( ", " , res9 )); // >>> 42.63
// Tests for 'tdig_quant' step.
double res10 = db . TDIGEST (). Min ( "racer_ages" );
Console . WriteLine ( res10 ); // >>> 19.27
double res11 = db . TDIGEST (). Max ( "racer_ages" );
Console . WriteLine ( res11 ); // >>> 85.71
// Tests for 'tdig_min' step.
bool res12 = db . TDIGEST (). Reset ( "racer_ages" );
Console . WriteLine ( res12 ); // >>> True
// Tests for 'tdig_reset' step.
}
}
Copied!
Academic sources
References