#pragma once #include #include #include #include #include #include #include #include #include "DictionaryStructure.h" #include "IDictionary.h" #include "IDictionarySource.h" #include "DictionaryHelpers.h" namespace DB { class FlatDictionary final : public IDictionary { public: struct Configuration { size_t initial_array_size; size_t max_array_size; bool require_nonempty; DictionaryLifetime dict_lifetime; bool use_async_executor = false; }; FlatDictionary( const StorageID & dict_id_, const DictionaryStructure & dict_struct_, DictionarySourcePtr source_ptr_, Configuration configuration_, BlockPtr update_field_loaded_block_ = nullptr); std::string getTypeName() const override { return "Flat"; } size_t getBytesAllocated() const override { return bytes_allocated; } size_t getQueryCount() const override { return query_count.load(); } double getFoundRate() const override { size_t queries = query_count.load(); if (!queries) return 0; return std::min(1.0, static_cast(found_count.load()) / queries); } double getHitRate() const override { return 1.0; } size_t getElementCount() const override { return element_count; } double getLoadFactor() const override { return static_cast(element_count) / bucket_count; } std::shared_ptr clone() const override { return std::make_shared(getDictionaryID(), dict_struct, source_ptr->clone(), configuration, update_field_loaded_block); } DictionarySourcePtr getSource() const override { return source_ptr; } const DictionaryLifetime & getLifetime() const override { return configuration.dict_lifetime; } const DictionaryStructure & getStructure() const override { return dict_struct; } bool isInjective(const std::string & attribute_name) const override { return dict_struct.getAttribute(attribute_name).injective; } DictionaryKeyType getKeyType() const override { return DictionaryKeyType::Simple; } ColumnPtr getColumn( const std::string & attribute_name, const DataTypePtr & attribute_type, const Columns & key_columns, const DataTypes & key_types, DefaultOrFilter default_or_filter) const override; ColumnUInt8::Ptr hasKeys(const Columns & key_columns, const DataTypes & key_types) const override; bool hasHierarchy() const override { return dict_struct.hierarchical_attribute_index.has_value(); } ColumnPtr getHierarchy(ColumnPtr key_column, const DataTypePtr & key_type) const override; ColumnUInt8::Ptr isInHierarchy( ColumnPtr key_column, ColumnPtr in_key_column, const DataTypePtr & key_type) const override; DictionaryHierarchicalParentToChildIndexPtr getHierarchicalIndex() const override; size_t getHierarchicalIndexBytesAllocated() const override { return hierarchical_index_bytes_allocated; } ColumnPtr getDescendants( ColumnPtr key_column, const DataTypePtr & key_type, size_t level, DictionaryHierarchicalParentToChildIndexPtr parent_to_child_index) const override; Pipe read(const Names & column_names, size_t max_block_size, size_t num_streams) const override; private: template using ContainerType = std::conditional_t, std::vector, PaddedPODArray>; using NullableSet = HashSet>; struct Attribute final { AttributeUnderlyingType type; std::optional is_nullable_set; std::variant< ContainerType, ContainerType, ContainerType, ContainerType, ContainerType, ContainerType, ContainerType, ContainerType, ContainerType, ContainerType, ContainerType, ContainerType, ContainerType, ContainerType, ContainerType, ContainerType, ContainerType, ContainerType, ContainerType, ContainerType, ContainerType, ContainerType, ContainerType, ContainerType> container; }; void createAttributes(); void blockToAttributes(const Block & block); void updateData(); void loadData(); void buildHierarchyParentToChildIndexIfNeeded(); void calculateBytesAllocated(); Attribute createAttribute(const DictionaryAttribute & attribute); template void getItemsImpl( const Attribute & attribute, const PaddedPODArray & keys, ValueSetter && set_value, DefaultValueExtractor & default_value_extractor) const; template void getItemsShortCircuitImpl( const Attribute & attribute, const PaddedPODArray & keys, ValueSetter && set_value, IColumn::Filter & default_mask) const; template void resize(Attribute & attribute, UInt64 key); void setAttributeValue(Attribute & attribute, UInt64 key, const Field & value); const DictionaryStructure dict_struct; const DictionarySourcePtr source_ptr; const Configuration configuration; std::vector attributes; std::vector loaded_keys; size_t bytes_allocated = 0; size_t hierarchical_index_bytes_allocated = 0; size_t element_count = 0; size_t bucket_count = 0; mutable std::atomic query_count{0}; mutable std::atomic found_count{0}; BlockPtr update_field_loaded_block; Arena string_arena; DictionaryHierarchicalParentToChildIndexPtr hierarhical_index; }; }