#pragma once #include #include namespace DB { /** Интерфейс для агрегатных функций, принимающих одно значение. Это почти все агрегатные функции. */ template class IUnaryAggregateFunction : public IAggregateFunctionHelper { private: Derived & getDerived() { return static_cast(*this); } const Derived & getDerived() const { return static_cast(*this); } public: void setArguments(const DataTypes & arguments) override final { if (arguments.size() != 1) throw Exception("Passed " + toString(arguments.size()) + " arguments to unary aggregate function " + this->getName(), ErrorCodes::NUMBER_OF_ARGUMENTS_DOESNT_MATCH); getDerived().setArgument(arguments[0]); } /// Добавить значение. void add(AggregateDataPtr place, const IColumn ** columns, size_t row_num) const override final { getDerived().addImpl(place, *columns[0], row_num); } static void addFree(const IAggregateFunction * that, AggregateDataPtr place, const IColumn ** columns, size_t row_num) { return static_cast(*that).addImpl(place, *columns[0], row_num); } IAggregateFunction::AddFunc getAddressOfAddFunction() const override { return &addFree; } /** Реализуйте это в классе-наследнике: * void addImpl(AggregateDataPtr place, const IColumn & column, size_t row_num) const; * void setArgument(const DataTypePtr & argument); */ }; }