#pragma once #include #include #include #include namespace DB { namespace ErrorCodes { extern const int LOGICAL_ERROR; } namespace details { enum class ContainerType : uint8_t { SMALL = 1, MEDIUM = 2, LARGE = 3 }; static inline ContainerType max(const ContainerType & lhs, const ContainerType & rhs) { uint8_t res = std::max(static_cast(lhs), static_cast(rhs)); return static_cast(res); } } /** For a small number of keys - an array of fixed size "on the stack". * For the average, HashSet is allocated. * For large, HyperLogLog is allocated. */ template < typename Key, typename HashContainer, UInt8 small_set_size_max, UInt8 medium_set_power2_max, UInt8 K, typename Hash = IntHash32, typename HashValueType = UInt32, typename BiasEstimator = TrivialBiasEstimator, HyperLogLogMode mode = HyperLogLogMode::FullFeatured, typename DenominatorType = double > class CombinedCardinalityEstimator { public: using Self = CombinedCardinalityEstimator < Key, HashContainer, small_set_size_max, medium_set_power2_max, K, Hash, HashValueType, BiasEstimator, mode, DenominatorType >; using value_type = Key; private: using Small = SmallSet; using Medium = HashContainer; using Large = HyperLogLogCounter; public: CombinedCardinalityEstimator() { setContainerType(details::ContainerType::SMALL); } ~CombinedCardinalityEstimator() { destroy(); } void insert(Key value) { auto container_type = getContainerType(); if (container_type == details::ContainerType::SMALL) { if (small.find(value) == small.end()) { if (!small.full()) small.insert(value); else { toMedium(); getContainer().insert(value); } } } else if (container_type == details::ContainerType::MEDIUM) { auto & container = getContainer(); if (container.size() < medium_set_size_max) container.insert(value); else { toLarge(); getContainer().insert(value); } } else if (container_type == details::ContainerType::LARGE) getContainer().insert(value); } UInt64 size() const { auto container_type = getContainerType(); if (container_type == details::ContainerType::SMALL) return small.size(); if (container_type == details::ContainerType::MEDIUM) return getContainer().size(); if (container_type == details::ContainerType::LARGE) return getContainer().size(); throw Poco::Exception("Internal error", ErrorCodes::LOGICAL_ERROR); } void merge(const Self & rhs) { auto container_type = getContainerType(); auto max_container_type = details::max(container_type, rhs.getContainerType()); if (container_type != max_container_type) { if (max_container_type == details::ContainerType::MEDIUM) toMedium(); else if (max_container_type == details::ContainerType::LARGE) toLarge(); } if (rhs.getContainerType() == details::ContainerType::SMALL) { for (const auto & x : rhs.small) insert(x.getValue()); } else if (rhs.getContainerType() == details::ContainerType::MEDIUM) { for (const auto & x : rhs.getContainer()) insert(x.getValue()); } else if (rhs.getContainerType() == details::ContainerType::LARGE) getContainer().merge(rhs.getContainer()); } /// You can only call for an empty object. void read(DB::ReadBuffer & in) { UInt8 v; readBinary(v, in); auto container_type = static_cast(v); if (container_type == details::ContainerType::SMALL) small.read(in); else if (container_type == details::ContainerType::MEDIUM) { toMedium(); getContainer().read(in); } else if (container_type == details::ContainerType::LARGE) { toLarge(); getContainer().read(in); } } void write(DB::WriteBuffer & out) const { auto container_type = getContainerType(); writeBinary(static_cast(container_type), out); if (container_type == details::ContainerType::SMALL) small.write(out); else if (container_type == details::ContainerType::MEDIUM) getContainer().write(out); else if (container_type == details::ContainerType::LARGE) getContainer().write(out); } private: void toMedium() { if (getContainerType() != details::ContainerType::SMALL) throw Poco::Exception("Internal error", ErrorCodes::LOGICAL_ERROR); auto tmp_medium = std::make_unique(); for (const auto & x : small) tmp_medium->insert(x.getValue()); medium = tmp_medium.release(); setContainerType(details::ContainerType::MEDIUM); } void toLarge() { auto container_type = getContainerType(); if ((container_type != details::ContainerType::SMALL) && (container_type != details::ContainerType::MEDIUM)) throw Poco::Exception("Internal error", ErrorCodes::LOGICAL_ERROR); auto tmp_large = std::make_unique(); if (container_type == details::ContainerType::SMALL) { for (const auto & x : small) tmp_large->insert(x.getValue()); } else if (container_type == details::ContainerType::MEDIUM) { for (const auto & x : getContainer()) tmp_large->insert(x.getValue()); destroy(); } large = tmp_large.release(); setContainerType(details::ContainerType::LARGE); } void NO_INLINE destroy() { auto container_type = getContainerType(); clearContainerType(); if (container_type == details::ContainerType::MEDIUM) { delete medium; medium = nullptr; } else if (container_type == details::ContainerType::LARGE) { delete large; large = nullptr; } } template T & getContainer() { return *reinterpret_cast(address & mask); } template const T & getContainer() const { return *reinterpret_cast(address & mask); } void setContainerType(details::ContainerType t) { address &= mask; address |= static_cast(t); } details::ContainerType getContainerType() const { return static_cast(address & ~mask); } void clearContainerType() { address &= mask; } Small small; union { Medium * medium; Large * large; UInt64 address = 0; }; static const UInt64 mask = 0xFFFFFFFFFFFFFFFC; static const UInt32 medium_set_size_max = 1ULL << medium_set_power2_max; }; }