| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122 |
- //
- // Aggregation.cs
- //
- // Author:
- // Juraj Skripsky ([email protected])
- //
- // (C) 2004 HotFeet GmbH (http://www.hotfeet.ch)
- //
- using System;
- using System.Collections;
- using System.Data;
- namespace Mono.Data.SqlExpressions {
- public enum AggregationFunction {
- Count, Sum, Min, Max, Avg, StDev, Var
- }
- public class Aggregation : IExpression {
- bool cacheResults;
- DataRow[] rows;
- ColumnReference column;
- AggregationFunction function;
- int count;
- IConvertible result;
-
- public Aggregation (bool cacheResults, DataRow[] rows, AggregationFunction function, ColumnReference column)
- {
- this.cacheResults = cacheResults;
- this.rows = rows;
- this.column = column;
- this.function = function;
- this.result = null;
- }
-
- public object Eval (DataRow row)
- {
- //TODO: implement a better caching strategy and a mechanism for cache invalidation.
- //for now only aggregation over the table owning 'row' (e.g. 'sum(parts)'
- //in constrast to 'sum(parent.parts)' and 'sum(child.parts)') is cached.
- if (cacheResults && result != null && column.ReferencedTable == ReferencedTable.Self)
- return result;
-
- count = 0;
- result = null;
-
- object[] values;
- if (rows == null)
- values = column.GetValues (column.GetReferencedRows (row));
- else
- values = column.GetValues (rows);
-
- foreach (object val in values) {
- if (val == null)
- continue;
-
- count++;
- Aggregate ((IConvertible)val);
- }
-
- switch (function) {
- case AggregationFunction.StDev:
- case AggregationFunction.Var:
- result = CalcStatisticalFunction (values);
- break;
-
- case AggregationFunction.Avg:
- result = Numeric.Divide (result, count);
- break;
-
- case AggregationFunction.Count:
- result = count;
- break;
- }
-
- if (result == null)
- result = 0;
-
- return result;
- }
-
- private void Aggregate (IConvertible val)
- {
- switch (function) {
- case AggregationFunction.Min:
- result = (result != null ? Numeric.Min (result, val) : val);
- return;
-
- case AggregationFunction.Max:
- result = (result != null ? Numeric.Max (result, val) : val);
- return;
- case AggregationFunction.Sum:
- case AggregationFunction.Avg:
- case AggregationFunction.StDev:
- case AggregationFunction.Var:
- result = (result != null ? Numeric.Add (result, val) : val);
- return;
- }
- }
-
- private IConvertible CalcStatisticalFunction (object[] values)
- {
- double average = (double)Convert.ChangeType(result, TypeCode.Double) / count;
- double res = 0.0;
-
- foreach (object val in values) {
- if (val == null)
- continue;
-
- double diff = average - (double)Convert.ChangeType(val, TypeCode.Double);
- res += System.Math.Pow (diff, 2);
- }
- res /= (count - 1);
-
- if (function == AggregationFunction.StDev)
- res = System.Math.Sqrt (res);
-
- return res;
- }
- }
- }
|