#pragma once #include "IArraySink.h" #include #include #include #include #include #include #include #include #include namespace DB::GatherUtils { template struct NumericArraySource; struct GenericArraySource; template struct NullableArraySource; template struct NumericValueSource; struct GenericValueSource; template struct NullableValueSource; template struct NumericArraySink : public ArraySinkImpl> { using ColVecType = ColumnVectorOrDecimal; using CompatibleArraySource = NumericArraySource; using CompatibleValueSource = NumericValueSource; typename ColVecType::Container & elements; typename ColumnArray::Offsets & offsets; size_t row_num = 0; ColumnArray::Offset current_offset = 0; NumericArraySink(IColumn & elements_, ColumnArray::Offsets & offsets_, size_t column_size) : elements(assert_cast(elements_).getData()), offsets(offsets_) { offsets.resize_exact(column_size); } void next() { offsets[row_num] = current_offset; ++row_num; } bool isEnd() const { return row_num == offsets.size(); } size_t rowNum() const { return row_num; } void reserve(size_t num_elements) { elements.reserve_exact(num_elements); } }; struct StringSink { typename ColumnString::Chars & elements; typename ColumnString::Offsets & offsets; size_t row_num = 0; ColumnString::Offset current_offset = 0; StringSink(ColumnString & col, size_t column_size) : elements(col.getChars()), offsets(col.getOffsets()) { offsets.resize_exact(column_size); } void ALWAYS_INLINE next() { elements.push_back(0); ++current_offset; offsets[row_num] = current_offset; ++row_num; } bool isEnd() const { return row_num == offsets.size(); } size_t rowNum() const { return row_num; } void reserve(size_t num_elements) { elements.reserve_exact(num_elements); } }; struct FixedStringSink { typename ColumnString::Chars & elements; size_t string_size; size_t row_num = 0; size_t total_rows; ColumnString::Offset current_offset = 0; FixedStringSink(ColumnFixedString & col, size_t column_size) : elements(col.getChars()), string_size(col.getN()), total_rows(column_size) { elements.resize_exact(column_size * string_size); } void next() { current_offset += string_size; ++row_num; } bool isEnd() const { return row_num == total_rows; } size_t rowNum() const { return row_num; } void reserve(size_t num_elements) { elements.reserve_exact(num_elements); } }; struct GenericArraySink : public ArraySinkImpl { using CompatibleArraySource = GenericArraySource; using CompatibleValueSource = GenericValueSource; IColumn & elements; ColumnArray::Offsets & offsets; size_t row_num = 0; ColumnArray::Offset current_offset = 0; GenericArraySink(IColumn & elements_, ColumnArray::Offsets & offsets_, size_t column_size) : elements(elements_), offsets(offsets_) { offsets.resize_exact(column_size); } void next() { offsets[row_num] = current_offset; ++row_num; } bool isEnd() const { return row_num == offsets.size(); } size_t rowNum() const { return row_num; } void reserve(size_t num_elements) { elements.reserve(num_elements); } }; template struct NullableArraySink : public ArraySink { using CompatibleArraySource = NullableArraySource; using CompatibleValueSource = NullableValueSource; NullMap & null_map; NullableArraySink(IColumn & elements_, ColumnArray::Offsets & offsets_, size_t column_size) : ArraySink(assert_cast(elements_).getNestedColumn(), offsets_, column_size) , null_map(assert_cast(elements_).getNullMapData()) { } void accept(ArraySinkVisitor & visitor) override { visitor.visit(*this); } void reserve(size_t num_elements) { ArraySink::reserve(num_elements); null_map.reserve_exact(num_elements); } }; }