Looking for a Histogram Binning algorithm for decimal data
Asked Answered
R

3

13

I need to generate bins for the purposes of calculating a histogram. Language is C#. Basically I need to take in an array of decimal numbers and generate a histogram plot out of those.

Haven't been able to find a decent library to do this outright so now I'm just looking for either a library or an algorithm to help me do the binning of the data.

So...

  • Are there any C# libraries out there that will take in an array of decimal data and output a binned histogram?
  • Is there generic algorithm for building the bins to be used in generated a histogram?
Ryurik answered 5/3, 2010 at 15:40 Comment(0)
S
17

Here is a simple bucket function I use. Sadly, .NET generics doesn't support a numerical type contraint so you will have to implement a different version of the following function for decimal, int, double, etc.

public static List<int> Bucketize(this IEnumerable<decimal> source, int totalBuckets)
{
    var min = source.Min();
    var max = source.Max();
    var buckets = new List<int>();

    var bucketSize = (max - min) / totalBuckets;
    foreach (var value in source)
    {
        int bucketIndex = 0;
        if (bucketSize > 0.0)
        {
            bucketIndex = (int)((value - min) / bucketSize);
            if (bucketIndex == totalBuckets)
            {
                bucketIndex--;
            }
        }
        buckets[bucketIndex]++;
    }
    return buckets;
}
Surface answered 5/3, 2010 at 15:53 Comment(4)
I had written an almost identical algorithm using the SAS language and was going to have to have my dev translate it into C#. Thanks for this.Ryurik
@Jake Pearson: If you import the namespace System.Linq, then you will not need your first foreach loop to find the min and max values. Instead just write: min = source.Min(); and max = source.Max(). Im not sure how much more effiecient it is on the cpu, but it's slightly less to read.Illinois
I might be a crazy person, but that code just doesn't work. The buckets variable never has any entries in the list and always throws an index out of range exception.Kurzawa
I had an issue initially because I was not using a sorted vector. By making an array: int[] bucketsArray = new int[totalBuckets]; and then converting the array to a list: List<int> bucketsList = new List<int>(bucketsArray); I could get around this. Thanks for the answer, it helped a lot!Paramedic
S
8

I got odd results using @JakePearson accepted answer. It has to do with an edge case.

Here is the code I used to test his method. I changed the extension method ever so slightly, returning an int[] and accepting double instead of decimal.

public partial class Form1 : Form
{
    public Form1()
    {
        InitializeComponent();

        Random rand = new Random(1325165);

        int maxValue = 100;
        int numberOfBuckets = 100;

        List<double> values = new List<double>();
        for (int i = 0; i < 10000000; i++)
        {
            double value = rand.NextDouble() * (maxValue+1);               
            values.Add(value);
        }

        int[] bins = values.Bucketize(numberOfBuckets);

        PointPairList points = new PointPairList();
        for (int i = 0; i < numberOfBuckets; i++)
        {
            points.Add(i, bins[i]);
        }

        zedGraphControl1.GraphPane.AddBar("Random Points", points,Color.Black);
        zedGraphControl1.GraphPane.YAxis.Title.Text = "Count";
        zedGraphControl1.GraphPane.XAxis.Title.Text = "Value";


        zedGraphControl1.AxisChange();
        zedGraphControl1.Refresh();

    }
}

public static class Extension
{
    public static int[] Bucketize(this IEnumerable<double> source, int totalBuckets)
    {
        var min = source.Min();
        var max = source.Max();
        var buckets = new int[totalBuckets];

        var bucketSize = (max - min) / totalBuckets;
        foreach (var value in source)
        {
            int bucketIndex = 0;
            if (bucketSize > 0.0)
            {
                bucketIndex = (int)((value - min) / bucketSize);
                if (bucketIndex == totalBuckets)
                {
                    bucketIndex--;
                }
            }
            buckets[bucketIndex]++;
        }
        return buckets;
    }
}

Everything works well when using 10,000,000 random double values between 0 and 100 (exclusive). Each bucket has roughly the same number of values, which makes sense given that Random returns a normal distribution.

Good Result

But when I changed the value generation line from

double value = rand.NextDouble() * (maxValue+1);              

to

double value = rand.Next(0, maxValue + 1);

and you get the following result, which double counts the last bucket.

Odd Result

It appears that when a value is same as one of the boundaries of a bucket, the code as it is written puts the value in the incorrect bucket. This artifact doesn't appear to happen with random double values as the chance of a random number being equal to a boundary of a bucket is rare and wouldn't be obvious.

The way I corrected this is to define what side of the bucket boundary is inclusive vs. exclusive.

Think of

0< x <=1 1< x <=2 ... 99< x <=100

vs.

0<= x <1 1<= x <2 ... 99<= x <100

You cannot have both boundaries inclusive, as the method wouldn't know which bucket to put it in if you have a value that is exactly equal to a boundary.

    public enum BucketizeDirectionEnum
    {
        LowerBoundInclusive,
        UpperBoundInclusive
    }

    public static int[] Bucketize(this IList<double> source, int totalBuckets, BucketizeDirectionEnum inclusivity = BucketizeDirectionEnum.UpperBoundInclusive)
    {
        var min = source.Min();
        var max = source.Max();
        var buckets = new int[totalBuckets];
        var bucketSize = (max - min) / totalBuckets;

        if (inclusivity == BucketizeDirectionEnum.LowerBoundInclusive)
        {
            foreach (var value in source)
            {
                int bucketIndex = (int)((value - min) / bucketSize);
                if (bucketIndex == totalBuckets)
                    continue;
                buckets[bucketIndex]++;
            }
        }
        else
        {
            foreach (var value in source)
            {
                int bucketIndex = (int)Math.Ceiling((value - min) / bucketSize) - 1;
                if (bucketIndex < 0)
                    continue;
                buckets[bucketIndex]++;
            }
        }

        return buckets;
    }

The only issue now is if the input dataset has a lot of min and max values, the binning method will exclude many of those values and the resulting graph will misrepresent the dataset.

Squilgee answered 28/5, 2014 at 17:46 Comment(0)
Y
1

I was able to get this to work. I had issues with the buckets. I created an array of max bucket sizes. It works like this - take the values that are greater than the min bucket size and <= max bucket size.

For example if your bucket range is 0.98 to 1.00, you want values > 0.98 and <= 1.00

The top array creates a array of max bucket size. Then you just iterate and compare. In my case there were only 15 buckets. On modern machines the additional overhead is trivial.

For i As Double = minValue To maxValue Step binSize
    maxbinValues(n) = i
    n = n + 1
    newBuckets(n) = 0
Next



        For Each value In source
            If value < minValue Then Continue For
            If value > maxValue Then Continue For
            Dim foundbucketindex As Integer = 0
            ' new code
            For i As Integer = 0 To n
                If value <= maxbinValues(i) Then
                    newBuckets(i) = newBuckets(i) + 1
                    foundbucketindex = i
                    Exit For
                End If
            Next
        Next
Yarborough answered 26/1, 2024 at 23:52 Comment(0)

© 2022 - 2025 — McMap. All rights reserved.