Перейти к основному содержанию
ClickHouse поддерживает импорт данных из CSV и экспорт данных в CSV. Поскольку CSV-файлы могут различаться по особенностям формата, включая строки заголовков, пользовательские разделители и символы экранирования, ClickHouse предоставляет форматы и настройки для эффективной обработки каждого такого случая.

Импорт данных из CSV-файла

Перед импортом данных создадим таблицу с подходящей структурой:
CREATE TABLE sometable
(
    `path` String,
    `month` Date,
    `hits` UInt32
)
ENGINE = MergeTree
ORDER BY tuple(month, path)
Чтобы импортировать данные из CSV-файла в таблицу sometable, можно направить файл напрямую в clickhouse-client:
clickhouse-client -q "INSERT INTO sometable FORMAT CSV" < data_small.csv
Обратите внимание, что мы используем FORMAT CSV, чтобы ClickHouse понимал, что данные поступают в формате CSV. Либо можно загрузить данные из локального файла с помощью конструкции FROM INFILE:
INSERT INTO sometable
FROM INFILE 'data_small.csv'
FORMAT CSV
Здесь мы используем выражение FORMAT CSV, чтобы ClickHouse понимал формат файла. Мы также можем загружать данные напрямую из URL-адресов с помощью функции url() или из файлов S3 с помощью функции s3().
Мы можем не указывать формат явно для file() и INFILE/OUTFILE. В этом случае ClickHouse автоматически определит формат по расширению файла.

CSV-файлы с заголовками

Предположим, что в нашем CSV-файле есть заголовки:
head data-small-headers.csv
"path","month","hits"
"Akiba_Hebrew_Academy","2017-08-01",241
"Aegithina_tiphia","2018-02-01",34
Чтобы импортировать данные из этого файла, можно использовать формат CSVWithNames:
clickhouse-client -q "INSERT INTO sometable FORMAT CSVWithNames" < data_small_headers.csv
В этом случае ClickHouse пропускает первую строку при импорте данных из файла.
Начиная с версии 23.1, ClickHouse автоматически распознаёт заголовки в CSV-файлах при использовании формата CSV, поэтому использовать CSVWithNames или CSVWithNamesAndTypes не нужно.

CSV-файлы с пользовательскими разделителями

Если в CSV-файле в качестве разделителя используется не запятая, можно воспользоваться параметром format_csv_delimiter, чтобы указать нужный символ:
SET format_csv_delimiter = ';'
Теперь при импорте из CSV-файла в качестве разделителя будет использоваться символ ;, а не запятая.

Пропуск строк в CSV-файле

Иногда при импорте данных из CSV-файла требуется пропустить определённое количество строк. Это можно сделать с помощью настройки input_format_csv_skip_first_lines:
SET input_format_csv_skip_first_lines = 10
В этом случае мы пропустим первые десять строк CSV-файла:
SELECT count(*) FROM file('data-small.csv', CSV)
┌─count()─┐
│     990 │
└─────────┘
В файле 1k строк, но ClickHouse загрузил только 990, так как мы указали пропустить первые 10.
При использовании функции file() в ClickHouse Cloud вам нужно запускать команды в клиенте ClickHouse на машине, где находится файл. Другой вариант — использовать clickhouse-local для локального изучения файлов.

Обработка значений NULL в CSV-файлах

Значения NULL могут кодироваться по-разному в зависимости от приложения, в котором был создан файл. По умолчанию ClickHouse использует \N в качестве значения NULL в CSV. Но это можно изменить с помощью параметра format_csv_null_representation. Предположим, у нас есть следующий CSV-файл:
> cat nulls.csv
Donald,90
Joe,Nothing
Nothing,70
Если мы загрузим данные из этого файла, ClickHouse будет интерпретировать Nothing как String (что верно):
SELECT * FROM file('nulls.csv')
┌─c1──────┬─c2──────┐
│ Donald  │ 90      │
│ Joe     │ Nothing │
│ Nothing │ 70      │
└─────────┴─────────┘
Если мы хотим, чтобы ClickHouse считал Nothing значением NULL, это можно указать с помощью следующего параметра:
SET format_csv_null_representation = 'Nothing'
Теперь NULL находится именно там, где мы и ожидали его увидеть:
SELECT * FROM file('nulls.csv')
┌─c1─────┬─c2───┐
│ Donald │ 90   │
│ Joe    │ ᴺᵁᴸᴸ │
│ ᴺᵁᴸᴸ   │ 70   │
└────────┴──────┘

Файлы TSV (с разделителем табуляции)

Формат данных с разделителем табуляции широко используется для обмена данными. Чтобы загрузить данные из файла в формате TSV в ClickHouse, используется формат TabSeparated:
clickhouse-client -q "INSERT INTO sometable FORMAT TabSeparated" < data_small.tsv
Существует также формат TabSeparatedWithNames, который позволяет работать с файлами в формате TSV с заголовками. И, как и в случае с CSV, можно пропустить первые X строк с помощью параметра input_format_tsv_skip_first_lines.

Неэкранированный TSV

Иногда файлы в формате TSV сохраняются без экранирования символов табуляции и переводов строки. Для обработки таких файлов следует использовать TabSeparatedRaw.

Экспорт в CSV

Любой формат из предыдущих примеров также можно использовать для экспорта данных. Чтобы экспортировать данные из таблицы (или из запроса) в формат CSV, используется то же предложение FORMAT:
SELECT *
FROM sometable
LIMIT 5
FORMAT CSV
"Akiba_Hebrew_Academy","2017-08-01",241
"Aegithina_tiphia","2018-02-01",34
"1971-72_Utah_Stars_season","2016-10-01",1
"2015_UEFA_European_Under-21_Championship_qualification_Group_8","2015-12-01",73
"2016_Greater_Western_Sydney_Giants_season","2017-05-01",86
Чтобы добавить строку заголовков в CSV-файл, используйте формат CSVWithNames:
SELECT *
FROM sometable
LIMIT 5
FORMAT CSVWithNames
"path","month","hits"
"Akiba_Hebrew_Academy","2017-08-01",241
"Aegithina_tiphia","2018-02-01",34
"1971-72_Utah_Stars_season","2016-10-01",1
"2015_UEFA_European_Under-21_Championship_qualification_Group_8","2015-12-01",73
"2016_Greater_Western_Sydney_Giants_season","2017-05-01",86

Сохранение экспортированных данных в CSV-файл

Чтобы сохранить экспортированные данные в файл, можно использовать конструкцию INTO…OUTFILE:
SELECT *
FROM sometable
INTO OUTFILE 'out.csv'
FORMAT CSVWithNames
36838935 rows in set. Elapsed: 1.304 sec. Processed 36.84 million rows, 1.42 GB (28.24 million rows/s., 1.09 GB/s.)
Обратите внимание, что ClickHouse потребовалось ~1 секунды, чтобы сохранить 36 млн строк в CSV-файл.

Экспорт CSV с пользовательскими разделителями

Если нужно использовать разделители, отличные от запятой, можно воспользоваться параметром format_csv_delimiter:
SET format_csv_delimiter = '|'
Теперь ClickHouse будет использовать | как разделитель в формате CSV:
SELECT *
FROM sometable
LIMIT 5
FORMAT CSV
"Akiba_Hebrew_Academy"|"2017-08-01"|241
"Aegithina_tiphia"|"2018-02-01"|34
"1971-72_Utah_Stars_season"|"2016-10-01"|1
"2015_UEFA_European_Under-21_Championship_qualification_Group_8"|"2015-12-01"|73
"2016_Greater_Western_Sydney_Giants_season"|"2017-05-01"|86

Экспорт CSV для Windows

Если нужно, чтобы CSV-файл корректно работал в среде Windows, стоит включить параметр output_format_csv_crlf_end_of_line. В этом случае в качестве перевода строки будет использоваться \r\n вместо \n:
SET output_format_csv_crlf_end_of_line = 1;

Определение схемы для CSV-файлов

Во многих случаях приходится работать с неизвестными CSV-файлами, поэтому нужно определить, какие типы использовать для столбцов. ClickHouse по умолчанию пытается определить форматы данных на основе анализа указанного CSV-файла. Это называется “определением схемы”. Определённые типы данных можно просмотреть с помощью оператора DESCRIBE в сочетании с функцией file():
DESCRIBE file('data-small.csv', CSV)
┌─name─┬─type─────────────┬─default_type─┬─default_expression─┬─comment─┬─codec_expression─┬─ttl_expression─┐
│ c1   │ Nullable(String) │              │                    │         │                  │                │
│ c2   │ Nullable(Date)   │              │                    │         │                  │                │
│ c3   │ Nullable(Int64)  │              │                    │         │                  │                │
└──────┴──────────────────┴──────────────┴────────────────────┴─────────┴──────────────────┴────────────────┘
Здесь ClickHouse смог эффективно определить типы столбцов в нашем CSV-файле. Если мы не хотим, чтобы ClickHouse делал это автоматически, эту функцию можно отключить с помощью следующей опции:
SET input_format_csv_use_best_effort_in_schema_inference = 0
В этом случае типы всех столбцов будут считаться String.

Экспорт и импорт CSV с явным указанием типов столбцов

ClickHouse также позволяет явно задавать типы столбцов при экспорте данных в формате CSVWithNamesAndTypes (и других форматов семейства WithNames):
SELECT *
FROM sometable
LIMIT 5
FORMAT CSVWithNamesAndTypes
"path","month","hits"
"String","Date","UInt32"
"Akiba_Hebrew_Academy","2017-08-01",241
"Aegithina_tiphia","2018-02-01",34
"1971-72_Utah_Stars_season","2016-10-01",1
"2015_UEFA_European_Under-21_Championship_qualification_Group_8","2015-12-01",73
"2016_Greater_Western_Sydney_Giants_season","2017-05-01",86
Этот формат будет содержать две строки заголовка: одну с именами столбцов, другую — с типами столбцов. Это позволит ClickHouse (и другим приложениям) определять типы столбцов при загрузке данных из таких файлов:
DESCRIBE file('data_csv_types.csv', CSVWithNamesAndTypes)
┌─name──┬─type───┬─default_type─┬─default_expression─┬─comment─┬─codec_expression─┬─ttl_expression─┐
│ path  │ String │              │                    │         │                  │                │
│ month │ Date   │              │                    │         │                  │                │
│ hits  │ UInt32 │              │                    │         │                  │                │
└───────┴────────┴──────────────┴────────────────────┴─────────┴──────────────────┴────────────────┘
Теперь ClickHouse определяет типы столбцов по (второй) строке заголовка, а не угадывает их.

Пользовательские разделители, разделители строк и правила экранирования

В сложных случаях текстовые данные могут быть оформлены весьма нестандартным образом, но при этом всё равно сохранять структуру. Для таких случаев в ClickHouse есть специальный формат CustomSeparated, который позволяет задавать пользовательские правила экранирования, разделители, разделители строк, а также начальные и конечные символы. Предположим, в файле есть следующие данные:
row('Akiba_Hebrew_Academy';'2017-08-01';241),row('Aegithina_tiphia';'2018-02-01';34),...
Мы видим, что отдельные строки заключены в row(), записи разделяются ,, а отдельные значения — символом ;. В этом случае мы можем использовать следующие настройки, чтобы читать данные из этого файла:
SET format_custom_row_before_delimiter = 'row(';
SET format_custom_row_after_delimiter = ')';
SET format_custom_field_delimiter = ';';
SET format_custom_row_between_delimiter = ',';
SET format_custom_escaping_rule = 'Quoted';
Теперь мы можем загрузить данные из нашего файла:
SELECT *
FROM file('data_small_custom.txt', CustomSeparated)
LIMIT 3
┌─c1────────────────────────┬─────────c2─┬──c3─┐
│ Akiba_Hebrew_Academy      │ 2017-08-01 │ 241 │
│ Aegithina_tiphia          │ 2018-02-01 │  34 │
│ 1971-72_Utah_Stars_season │ 2016-10-01 │   1 │
└───────────────────────────┴────────────┴─────┘
Мы также можем использовать CustomSeparatedWithNames, чтобы корректно экспортировать и импортировать строки заголовков. Ознакомьтесь с форматами regex and template, чтобы работать с ещё более сложными случаями.

Работа с большими CSV-файлами

CSV-файлы могут быть большими, и ClickHouse эффективно работает с файлами любого размера. Большие файлы обычно поставляются в сжатом виде, и ClickHouse поддерживает работу с ними без необходимости распаковки перед обработкой. При вставке можно использовать предложение COMPRESSION:
INSERT INTO sometable
FROM INFILE 'data_csv.csv.gz'
COMPRESSION 'gzip' FORMAT CSV
Если предложение COMPRESSION не указано, ClickHouse всё равно попытается определить сжатие файла по его расширению. Тот же подход можно использовать, чтобы экспортировать файлы напрямую в сжатые форматы:
SELECT *
FROM for_csv
INTO OUTFILE 'data_csv.csv.gz'
COMPRESSION 'gzip' FORMAT CSV
Это создаст файл data_csv.csv.gz в сжатом формате.

Другие форматы

ClickHouse поддерживает множество форматов — как текстовых, так и двоичных, — для самых разных сценариев и платформ. Подробнее о других форматах и способах работы с ними читайте в следующих статьях: Также ознакомьтесь с clickhouse-local — это переносимый полнофункциональный инструмент для работы с локальными и удалёнными файлами без необходимости использовать сервер ClickHouse.
Последнее изменение 10 июня 2026 г.