ClickHouse/dbms/src/Columns/ColumnNullable.cpp

354 lines
9.7 KiB
C++
Raw Normal View History

#include <DB/Columns/ColumnNullable.h>
namespace DB
{
namespace ErrorCodes
{
extern const int LOGICAL_ERROR;
}
2016-08-16 22:40:05 +00:00
ColumnNullable::ColumnNullable(ColumnPtr nested_column_, ColumnPtr null_map_)
: nested_column{nested_column_}, null_map{null_map_}
{
2016-08-12 15:22:28 +00:00
if (nested_column->isNullable())
throw Exception{"A nullable column cannot contain another nullable column", ErrorCodes::LOGICAL_ERROR};
}
ColumnPtr ColumnNullable::convertToFullColumnIfConst() const
{
ColumnPtr new_col_holder;
2016-08-12 15:22:28 +00:00
if (auto full_col = nested_column->convertToFullColumnIfConst())
2016-08-16 22:40:05 +00:00
new_col_holder = std::make_shared<ColumnNullable>(full_col, null_map);
return new_col_holder;
}
2016-07-11 10:09:16 +00:00
void ColumnNullable::updateHashWithValue(size_t n, SipHash & hash) const
{
2016-08-16 12:34:54 +00:00
const auto & arr = getNullMapContent().getData();
hash.update(reinterpret_cast<const char *>(&arr[n]), sizeof(arr[0]));
if (arr[n] == 0)
2016-08-12 15:22:28 +00:00
nested_column->updateHashWithValue(n, hash);
2016-07-11 10:09:16 +00:00
}
ColumnPtr ColumnNullable::cloneResized(size_t size) const
{
2016-08-16 22:40:05 +00:00
ColumnPtr new_nested_col = nested_column->cloneResized(size);
ColumnPtr new_null_map = getNullMapContent().cloneResized(size);
return std::make_shared<ColumnNullable>(new_nested_col, new_null_map);
}
Field ColumnNullable::operator[](size_t n) const
{
if (isNullAt(n))
return Field{};
else
{
const IColumn & col = *nested_column;
return col[n];
}
}
void ColumnNullable::get(size_t n, Field & res) const
{
if (isNullAt(n))
res = Field{};
else
2016-08-12 15:22:28 +00:00
nested_column->get(n, res);
}
StringRef ColumnNullable::getDataAt(size_t n) const
{
throw Exception{"Method getDataAt is not supported for " + getName(), ErrorCodes::NOT_IMPLEMENTED};
}
void ColumnNullable::insertData(const char * pos, size_t length)
{
throw Exception{"Method insertData is not supported for " + getName(), ErrorCodes::NOT_IMPLEMENTED};
}
StringRef ColumnNullable::serializeValueIntoArena(size_t n, Arena & arena, char const *& begin) const
{
2016-08-16 12:53:22 +00:00
const auto & arr = getNullMapContent().getData();
static constexpr auto s = sizeof(arr[0]);
auto pos = arena.allocContinue(s, begin);
memcpy(pos, &arr[n], s);
size_t nested_size = 0;
2016-08-16 12:53:22 +00:00
if (arr[n] == 0)
nested_size = nested_column->serializeValueIntoArena(n, arena, begin).size;
return StringRef{begin, s + nested_size};
}
const char * ColumnNullable::deserializeAndInsertFromArena(const char * pos)
{
2016-08-16 12:53:22 +00:00
UInt8 val = *reinterpret_cast<const UInt8 *>(pos);
pos += sizeof(val);
2016-08-16 12:53:22 +00:00
getNullMapContent().insert(val);
if (val == 0)
pos = nested_column->deserializeAndInsertFromArena(pos);
2016-08-16 12:53:22 +00:00
else
nested_column->insertDefault();
return pos;
}
void ColumnNullable::insertRangeFrom(const IColumn & src, size_t start, size_t length)
{
2016-08-16 11:26:17 +00:00
const ColumnNullable & nullable_col = static_cast<const ColumnNullable &>(src);
getNullMapContent().insertRangeFrom(*nullable_col.null_map, start, length);
nested_column->insertRangeFrom(*nullable_col.nested_column, start, length);
}
void ColumnNullable::insert(const Field & x)
{
if (x.isNull())
{
2016-08-12 15:22:28 +00:00
nested_column->insertDefault();
getNullMapContent().insert(1);
}
else
{
2016-08-12 15:22:28 +00:00
nested_column->insert(x);
getNullMapContent().insert(0);
}
}
void ColumnNullable::insertDefault()
{
2016-08-12 15:22:28 +00:00
nested_column->insertDefault();
2016-12-30 00:31:38 +00:00
getNullMapContent().insert(1);
}
void ColumnNullable::popBack(size_t n)
{
2016-08-12 15:22:28 +00:00
nested_column->popBack(n);
getNullMapContent().popBack(n);
}
ColumnPtr ColumnNullable::filter(const Filter & filt, ssize_t result_size_hint) const
{
2016-08-16 22:40:05 +00:00
ColumnPtr filtered_data = nested_column->filter(filt, result_size_hint);
ColumnPtr filtered_null_map = getNullMapContent().filter(filt, result_size_hint);
return std::make_shared<ColumnNullable>(filtered_data, filtered_null_map);
}
ColumnPtr ColumnNullable::permute(const Permutation & perm, size_t limit) const
{
2016-08-16 22:40:05 +00:00
ColumnPtr permuted_data = nested_column->permute(perm, limit);
ColumnPtr permuted_null_map = getNullMapContent().permute(perm, limit);
return std::make_shared<ColumnNullable>(permuted_data, permuted_null_map);
}
2016-08-16 11:26:17 +00:00
int ColumnNullable::compareAt(size_t n, size_t m, const IColumn & rhs_, int null_direction_hint) const
{
2016-08-17 17:20:21 +00:00
/// NULL values share the properties of NaN values.
/// Here the last parameter of compareAt is called null_direction_hint
/// instead of the usual nan_direction_hint and is used to implement
/// the ordering specified by either NULLS FIRST or NULLS LAST in the
/// ORDER BY construction.
const ColumnNullable & nullable_rhs = static_cast<const ColumnNullable &>(rhs_);
bool lval_is_null = isNullAt(n);
bool rval_is_null = nullable_rhs.isNullAt(m);
2016-08-17 17:20:21 +00:00
if (unlikely(lval_is_null || rval_is_null))
{
if (lval_is_null && rval_is_null)
return 0;
else
return lval_is_null ? null_direction_hint : -null_direction_hint;
}
const IColumn & nested_rhs = *(nullable_rhs.getNestedColumn());
2016-08-16 11:26:17 +00:00
return nested_column->compareAt(n, m, nested_rhs, null_direction_hint);
}
void ColumnNullable::getPermutation(bool reverse, size_t limit, Permutation & res) const
{
2016-08-12 15:22:28 +00:00
nested_column->getPermutation(reverse, limit, res);
size_t s = res.size();
/// Since we have created a permutation "res" that sorts a subset of the column values
/// and some of these values may actually be nulls, there is no guarantee that
/// these null values are well positioned. So we create a permutation "p" which
/// operates on the result of "res" by moving all the null values to the required
/// direction and leaving the order of the remaining elements unchanged.
/// Create the permutation p.
Permutation p;
p.resize(s);
size_t pos_left = 0;
size_t pos_right = s - 1;
if (reverse)
{
/// Move the null elements to the right.
for (size_t i = 0; i < s; ++i)
{
if (isNullAt(res[i]))
{
p[i] = pos_right;
--pos_right;
}
else
{
p[i] = pos_left;
++pos_left;
}
}
}
else
{
/// Move the null elements to the left.
for (size_t i = 0; i < s; ++i)
{
size_t j = s - i - 1;
if (isNullAt(res[j]))
{
p[j] = pos_left;
++pos_left;
}
else
{
p[j] = pos_right;
--pos_right;
}
}
}
/// Combine the permutations res and p.
Permutation res2;
res2.resize(s);
for (size_t i = 0; i < s; ++i)
res2[i] = res[p[i]];
res = std::move(res2);
}
void ColumnNullable::reserve(size_t n)
{
2016-08-12 15:22:28 +00:00
nested_column->reserve(n);
getNullMapContent().reserve(n);
}
size_t ColumnNullable::byteSize() const
{
2016-08-12 15:22:28 +00:00
return nested_column->byteSize() + getNullMapContent().byteSize();
}
2016-08-16 11:26:17 +00:00
namespace
{
/// The following function implements a slightly more general version
/// of getExtremes() than the implementation from ColumnVector.
/// It takes into account the possible presence of nullable values.
template <typename T>
void getExtremesFromNullableContent(const ColumnVector<T> & col, const NullValuesByteMap & null_map, Field & min, Field & max)
{
const auto & data = col.getData();
size_t size = data.size();
if (size == 0)
{
min = typename NearestFieldType<T>::Type(0);
max = typename NearestFieldType<T>::Type(0);
return;
}
size_t min_i = 0;
for (; min_i < size; ++min_i)
{
if (null_map[min_i] == 0)
break;
}
if (min_i == size)
{
min = Field{};
max = Field{};
return;
}
T cur_min = data[min_i];
T cur_max = data[min_i];
for (size_t i = min_i + 1; i < size; ++i)
{
if (null_map[i] != 0)
continue;
if (data[i] < cur_min)
cur_min = data[i];
if (data[i] > cur_max)
cur_max = data[i];
}
min = typename NearestFieldType<T>::Type(cur_min);
max = typename NearestFieldType<T>::Type(cur_max);
}
}
void ColumnNullable::getExtremes(Field & min, Field & max) const
{
2016-08-16 11:26:17 +00:00
if (const auto col = typeid_cast<const ColumnInt8 *>(nested_column.get()))
getExtremesFromNullableContent<Int8>(*col, getNullMapContent().getData(), min, max);
else if (const auto col = typeid_cast<const ColumnInt16 *>(nested_column.get()))
getExtremesFromNullableContent<Int16>(*col, getNullMapContent().getData(), min, max);
else if (const auto col = typeid_cast<const ColumnInt32 *>(nested_column.get()))
getExtremesFromNullableContent<Int32>(*col, getNullMapContent().getData(), min, max);
else if (const auto col = typeid_cast<const ColumnInt64 *>(nested_column.get()))
getExtremesFromNullableContent<Int64>(*col, getNullMapContent().getData(), min, max);
else if (const auto col = typeid_cast<const ColumnUInt8 *>(nested_column.get()))
getExtremesFromNullableContent<UInt8>(*col, getNullMapContent().getData(), min, max);
else if (const auto col = typeid_cast<const ColumnUInt16 *>(nested_column.get()))
getExtremesFromNullableContent<UInt16>(*col, getNullMapContent().getData(), min, max);
else if (const auto col = typeid_cast<const ColumnUInt32 *>(nested_column.get()))
getExtremesFromNullableContent<UInt32>(*col, getNullMapContent().getData(), min, max);
else if (const auto col = typeid_cast<const ColumnUInt64 *>(nested_column.get()))
getExtremesFromNullableContent<UInt64>(*col, getNullMapContent().getData(), min, max);
else if (const auto col = typeid_cast<const ColumnFloat32 *>(nested_column.get()))
getExtremesFromNullableContent<Float32>(*col, getNullMapContent().getData(), min, max);
else if (const auto col = typeid_cast<const ColumnFloat64 *>(nested_column.get()))
getExtremesFromNullableContent<Float64>(*col, getNullMapContent().getData(), min, max);
else
2016-08-12 15:22:28 +00:00
nested_column->getExtremes(min, max);
}
ColumnPtr ColumnNullable::replicate(const Offsets_t & offsets) const
{
2016-08-16 22:40:05 +00:00
ColumnPtr replicated_data = nested_column->replicate(offsets);
ColumnPtr replicated_null_map = getNullMapContent().replicate(offsets);
return std::make_shared<ColumnNullable>(replicated_data, replicated_null_map);
}
2016-08-16 13:31:42 +00:00
void ColumnNullable::applyNullValuesByteMap(const ColumnNullable & other)
{
NullValuesByteMap & arr1 = getNullMapContent().getData();
const NullValuesByteMap & arr2 = other.getNullMapContent().getData();
if (arr1.size() != arr2.size())
throw Exception{"Inconsistent sizes", ErrorCodes::LOGICAL_ERROR};
for (size_t i = 0; i < arr1.size(); ++i)
arr1[i] |= arr2[i];
}
}