Skip to content
Snippets Groups Projects
average.c 2.39 KiB
Newer Older
Bruce Cowan's avatar
Bruce Cowan committed
/* average.c
 *
 * Copyright 2018 Bruce Cowan <bruce@bcowan.me.uk>
 *
 * This program is free software: you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * You should have received a copy of the GNU General Public License
 * along with this program.  If not, see <http://www.gnu.org/licenses/>.
 *
 * SPDX-License-Identifier: GPL-3.0-or-later
 */

Bruce Cowan's avatar
Bruce Cowan committed
#include <stdlib.h>
Bruce Cowan's avatar
Bruce Cowan committed
#include <math.h>

#include <glib.h>
Bruce Cowan's avatar
Bruce Cowan committed

Bruce Cowan's avatar
Bruce Cowan committed
#include <array.h>
Bruce Cowan's avatar
Bruce Cowan committed

Bruce Cowan's avatar
Bruce Cowan committed
static gdouble
get_total (const Array *array)
{
    double total = 0;
    size_t length = array_get_length (array);

    for (size_t i = 0; i < length; i++)
Bruce Cowan's avatar
Bruce Cowan committed
        total += (double) GPOINTER_TO_INT (array_get (array, i));
Bruce Cowan's avatar
Bruce Cowan committed
static gdouble
get_average (const Array *array)
{
    return get_total (array) / array_get_length (array);
}

static Array *
create_array (size_t length)
Bruce Cowan's avatar
Bruce Cowan committed
{
    Array *array = array_new (NULL);
Bruce Cowan's avatar
Bruce Cowan committed

    for (size_t i = 1; i < length; i++)
        array_add (array, GINT_TO_POINTER (i));
Bruce Cowan's avatar
Bruce Cowan committed

    return array;
Bruce Cowan's avatar
Bruce Cowan committed
static void
expected_values (Array  *array,
                 double *total,
                 double *average)
Bruce Cowan's avatar
Bruce Cowan committed
{
Bruce Cowan's avatar
Bruce Cowan committed
    size_t n = array_get_length (array);
    gdouble sum = (n * (n+1)) / 2;

    if (total)
        *total = sum;

    if (average)
        *average = sum / n;
Bruce Cowan's avatar
Bruce Cowan committed
}

Bruce Cowan's avatar
Bruce Cowan committed
static void
test_total (void)
Bruce Cowan's avatar
Bruce Cowan committed
{
    Array *array = create_array (100);
Bruce Cowan's avatar
Bruce Cowan committed

    gdouble expected_total;
    gdouble total = get_total (array);
    expected_values (array, &expected_total, NULL);
Bruce Cowan's avatar
Bruce Cowan committed

Bruce Cowan's avatar
Bruce Cowan committed
    g_assert_cmpfloat (total, ==, expected_total);

    array_unref (array);
Bruce Cowan's avatar
Bruce Cowan committed
}

Bruce Cowan's avatar
Bruce Cowan committed
static void
test_average (void)
    Array *array = create_array (100);
Bruce Cowan's avatar
Bruce Cowan committed

    gdouble expected_average;
    gdouble average = get_average (array);
    expected_values (array, NULL, &expected_average);
Bruce Cowan's avatar
Bruce Cowan committed

    g_assert_cmpfloat (average, ==, expected_average);

    array_unref (array);
Bruce Cowan's avatar
Bruce Cowan committed
int
Bruce Cowan's avatar
Bruce Cowan committed
main (int    argc,
      char **argv)
Bruce Cowan's avatar
Bruce Cowan committed
{
Bruce Cowan's avatar
Bruce Cowan committed
    g_test_init (&argc, &argv, NULL);

    g_test_add_func ("/array/total", test_total);
    g_test_add_func ("/array/average", test_average);
Bruce Cowan's avatar
Bruce Cowan committed

    return g_test_run ();
Bruce Cowan's avatar
Bruce Cowan committed
}