ClickHouse/dbms/src/DataStreams/LimitBlockInputStream.cpp
2017-01-02 23:12:12 +03:00

66 lines
1.5 KiB
C++

#include <algorithm>
#include <DB/DataStreams/LimitBlockInputStream.h>
namespace DB
{
LimitBlockInputStream::LimitBlockInputStream(BlockInputStreamPtr input_, size_t limit_, size_t offset_, bool always_read_till_end_)
: limit(limit_), offset(offset_), always_read_till_end(always_read_till_end_)
{
children.push_back(input_);
}
Block LimitBlockInputStream::readImpl()
{
Block res;
size_t rows = 0;
/// pos - сколько строк было прочитано, включая последний прочитанный блок
if (pos >= offset + limit)
{
if (!always_read_till_end)
return res;
else
{
while (children.back()->read())
;
return res;
}
}
do
{
res = children.back()->read();
if (!res)
return res;
rows = res.rows();
pos += rows;
} while (pos <= offset);
/// отдать целый блок
if (pos >= offset + rows && pos <= offset + limit)
return res;
/// отдать кусок блока
size_t start = std::max(
static_cast<Int64>(0),
static_cast<Int64>(offset) - static_cast<Int64>(pos) + static_cast<Int64>(rows));
size_t length = std::min(
static_cast<Int64>(limit), std::min(
static_cast<Int64>(pos) - static_cast<Int64>(offset),
static_cast<Int64>(limit) + static_cast<Int64>(offset) - static_cast<Int64>(pos) + static_cast<Int64>(rows)));
for (size_t i = 0; i < res.columns(); ++i)
res.safeGetByPosition(i).column = res.safeGetByPosition(i).column->cut(start, length);
return res;
}
}