mirror of
https://github.com/ClickHouse/ClickHouse.git
synced 2024-11-19 22:22:00 +00:00
26ab5dd7a7
This is the first step of allowing heterogeneous cells in hash tables. performance test results are ``` 1. HashMap<UInt16, UInt8, TrivialHash, HashTableFixedGrower<16>>; 2. NewLookupMap<UInt16, UInt8> ResolutionWidth 30000 1 .................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................223550276.46 ResolutionWidth 30000 2 .................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................248772721.24 Best: 2 - 24877272124 ResolutionWidth 100000 1 ..........................................................................................................................................................................................................................................................238498413.99 ResolutionWidth 100000 2 ..........................................................................................................................................................................................................................................................261808889.98 Best: 2 - 26180888998 ResolutionWidth 300000 1 ...................................................................................239307348.81 ResolutionWidth 300000 2 ...................................................................................257592761.30 Best: 2 - 25759276130 ResolutionWidth 1000000 1 .........................240144759.26 ResolutionWidth 1000000 2 .........................257093531.91 Best: 2 - 25709353191 ResolutionWidth 5000000 1 .....241573260.35 ResolutionWidth 5000000 2 .....259314162.79 Best: 2 - 25931416279 ResolutionDepth 30000 1 .................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................217108119.84 ResolutionDepth 30000 2 .................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................................249459504.41 Best: 2 - 24945950441 ResolutionDepth 100000 1 ..........................................................................................................................................................................................................................................................229065162.17 ResolutionDepth 100000 2 ..........................................................................................................................................................................................................................................................253769105.64 Best: 2 - 25376910564 ResolutionDepth 300000 1 ...................................................................................233079225.18 ResolutionDepth 300000 2 ...................................................................................256316273.78 Best: 2 - 25631627378 ResolutionDepth 1000000 1 .........................234184633.51 ResolutionDepth 1000000 2 .........................261100491.57 Best: 2 - 26110049157 ResolutionDepth 5000000 1 .....233118795.66 ResolutionDepth 5000000 2 .....252436160.41 Best: 2 - 25243616041 ```
333 lines
9.2 KiB
C++
333 lines
9.2 KiB
C++
#pragma once
|
|
|
|
#include <Common/HashTable/SmallTable.h>
|
|
#include <Common/HashTable/HashSet.h>
|
|
#include <Common/HyperLogLogCounter.h>
|
|
#include <Common/MemoryTracker.h>
|
|
#include <Core/Defines.h>
|
|
|
|
|
|
namespace DB
|
|
{
|
|
|
|
namespace details
|
|
{
|
|
|
|
enum class ContainerType : UInt8 { SMALL = 1, MEDIUM = 2, LARGE = 3 };
|
|
|
|
static inline ContainerType max(const ContainerType & lhs, const ContainerType & rhs)
|
|
{
|
|
UInt8 res = std::max(static_cast<UInt8>(lhs), static_cast<UInt8>(rhs));
|
|
return static_cast<ContainerType>(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<Key>,
|
|
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<Key, small_set_size_max>;
|
|
using Medium = HashContainer;
|
|
using Large = HyperLogLogCounter<K, Hash, HashValueType, DenominatorType, BiasEstimator, mode>;
|
|
|
|
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<Medium>().insert(value);
|
|
}
|
|
}
|
|
}
|
|
else if (container_type == details::ContainerType::MEDIUM)
|
|
{
|
|
auto & container = getContainer<Medium>();
|
|
if (container.size() < medium_set_size_max)
|
|
container.insert(value);
|
|
else
|
|
{
|
|
toLarge();
|
|
getContainer<Large>().insert(value);
|
|
}
|
|
}
|
|
else if (container_type == details::ContainerType::LARGE)
|
|
getContainer<Large>().insert(value);
|
|
}
|
|
|
|
UInt32 size() const
|
|
{
|
|
auto container_type = getContainerType();
|
|
|
|
if (container_type == details::ContainerType::SMALL)
|
|
return small.size();
|
|
else if (container_type == details::ContainerType::MEDIUM)
|
|
return getContainer<Medium>().size();
|
|
else if (container_type == details::ContainerType::LARGE)
|
|
return getContainer<Large>().size();
|
|
else
|
|
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<Medium>())
|
|
insert(x.getValue());
|
|
}
|
|
else if (rhs.getContainerType() == details::ContainerType::LARGE)
|
|
getContainer<Large>().merge(rhs.getContainer<Large>());
|
|
}
|
|
|
|
/// You can only call for an empty object.
|
|
void read(DB::ReadBuffer & in)
|
|
{
|
|
UInt8 v;
|
|
readBinary(v, in);
|
|
auto container_type = static_cast<details::ContainerType>(v);
|
|
|
|
if (container_type == details::ContainerType::SMALL)
|
|
small.read(in);
|
|
else if (container_type == details::ContainerType::MEDIUM)
|
|
{
|
|
toMedium();
|
|
getContainer<Medium>().read(in);
|
|
}
|
|
else if (container_type == details::ContainerType::LARGE)
|
|
{
|
|
toLarge();
|
|
getContainer<Large>().read(in);
|
|
}
|
|
}
|
|
|
|
void readAndMerge(DB::ReadBuffer & in)
|
|
{
|
|
auto container_type = getContainerType();
|
|
|
|
/// If readAndMerge is called with an empty state, just deserialize
|
|
/// the state is specified as a parameter.
|
|
if ((container_type == details::ContainerType::SMALL) && small.empty())
|
|
{
|
|
read(in);
|
|
return;
|
|
}
|
|
|
|
UInt8 v;
|
|
readBinary(v, in);
|
|
auto rhs_container_type = static_cast<details::ContainerType>(v);
|
|
|
|
auto max_container_type = details::max(container_type, rhs_container_type);
|
|
|
|
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_container_type == details::ContainerType::SMALL)
|
|
{
|
|
typename Small::Reader reader(in);
|
|
while (reader.next())
|
|
insert(reader.get());
|
|
}
|
|
else if (rhs_container_type == details::ContainerType::MEDIUM)
|
|
{
|
|
typename Medium::Reader reader(in);
|
|
while (reader.next())
|
|
insert(reader.get());
|
|
}
|
|
else if (rhs_container_type == details::ContainerType::LARGE)
|
|
getContainer<Large>().readAndMerge(in);
|
|
}
|
|
|
|
void write(DB::WriteBuffer & out) const
|
|
{
|
|
auto container_type = getContainerType();
|
|
writeBinary(static_cast<UInt8>(container_type), out);
|
|
|
|
if (container_type == details::ContainerType::SMALL)
|
|
small.write(out);
|
|
else if (container_type == details::ContainerType::MEDIUM)
|
|
getContainer<Medium>().write(out);
|
|
else if (container_type == details::ContainerType::LARGE)
|
|
getContainer<Large>().write(out);
|
|
}
|
|
|
|
private:
|
|
void toMedium()
|
|
{
|
|
if (getContainerType() != details::ContainerType::SMALL)
|
|
throw Poco::Exception("Internal error", ErrorCodes::LOGICAL_ERROR);
|
|
|
|
CurrentMemoryTracker::alloc(sizeof(Medium));
|
|
auto tmp_medium = std::make_unique<Medium>();
|
|
|
|
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);
|
|
|
|
CurrentMemoryTracker::alloc(sizeof(Large));
|
|
auto tmp_large = std::make_unique<Large>();
|
|
|
|
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<Medium>())
|
|
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;
|
|
|
|
CurrentMemoryTracker::free(sizeof(Medium));
|
|
}
|
|
else if (container_type == details::ContainerType::LARGE)
|
|
{
|
|
delete large;
|
|
large = nullptr;
|
|
|
|
CurrentMemoryTracker::free(sizeof(Large));
|
|
}
|
|
}
|
|
|
|
template <typename T>
|
|
inline T & getContainer()
|
|
{
|
|
return *reinterpret_cast<T *>(address & mask);
|
|
}
|
|
|
|
template <typename T>
|
|
inline const T & getContainer() const
|
|
{
|
|
return *reinterpret_cast<T *>(address & mask);
|
|
}
|
|
|
|
void setContainerType(details::ContainerType t)
|
|
{
|
|
address &= mask;
|
|
address |= static_cast<UInt8>(t);
|
|
}
|
|
|
|
inline details::ContainerType getContainerType() const
|
|
{
|
|
return static_cast<details::ContainerType>(address & ~mask);
|
|
}
|
|
|
|
void clearContainerType()
|
|
{
|
|
address &= mask;
|
|
}
|
|
|
|
private:
|
|
Small small;
|
|
union
|
|
{
|
|
Medium * medium;
|
|
Large * large;
|
|
UInt64 address = 0;
|
|
};
|
|
static const UInt64 mask = 0xFFFFFFFFFFFFFFFC;
|
|
static const UInt32 medium_set_size_max = 1UL << medium_set_power2_max;
|
|
};
|
|
|
|
}
|