#pragma once #include #include #include #include #include namespace DB { template struct AggregateFunctionSumTraits; template <> struct AggregateFunctionSumTraits { static DataTypePtr getReturnType() { return new DataTypeVarUInt; } static void write(UInt64 x, WriteBuffer & buf) { writeVarUInt(x, buf); } static void read(UInt64 & x, ReadBuffer & buf) { readVarUInt(x, buf); } }; template <> struct AggregateFunctionSumTraits { static DataTypePtr getReturnType() { return new DataTypeVarInt; } static void write(Int64 x, WriteBuffer & buf) { writeVarInt(x, buf); } static void read(Int64 & x, ReadBuffer & buf) { readVarInt(x, buf); } }; template <> struct AggregateFunctionSumTraits { static DataTypePtr getReturnType() { return new DataTypeFloat64; } static void write(Float64 x, WriteBuffer & buf) { writeFloatBinary(x, buf); } static void read(Float64 & x, ReadBuffer & buf) { readFloatBinary(x, buf); } }; /// Считает сумму чисел. Параметром шаблона может быть UInt64, Int64 или Float64. template class AggregateFunctionSum : public IUnaryAggregateFunction { private: T sum; public: AggregateFunctionSum() : sum(0) {} String getName() const { return "sum"; } String getTypeID() const { return "sum_" + TypeName::get(); } AggregateFunctionPlainPtr cloneEmpty() const { return new AggregateFunctionSum; } DataTypePtr getReturnType() const { return AggregateFunctionSumTraits::getReturnType(); } void setArgument(const DataTypePtr & argument) { if (!argument->isNumeric()) throw Exception("Illegal type " + argument->getName() + " of argument for aggregate function " + getName(), ErrorCodes::ILLEGAL_TYPE_OF_ARGUMENT); } void addOne(const Field & value) { sum += boost::get(value); } void merge(const IAggregateFunction & rhs) { sum += static_cast &>(rhs).sum; } void serialize(WriteBuffer & buf) const { AggregateFunctionSumTraits::write(sum, buf); } void deserializeMerge(ReadBuffer & buf) { T tmp; AggregateFunctionSumTraits::read(tmp, buf); sum += tmp; } Field getResult() const { return sum; } }; }