--- name: clickhouse-io description: ClickHouse 資料庫模式、查詢優化、分析以及高效能分析工作負載的資料工程最佳實踐。 --- # ClickHouse 分析模式 (ClickHouse Analytics Patterns) 用於高效能分析與資料工程的 ClickHouse 特定模式。 ## 何時啟用 - 設計 ClickHouse 資料表綱要 (MergeTree 引擎選擇) - 撰寫分析查詢 (聚合、視窗函式、Join) - 優化查詢效能 (分區裁剪、投影 Projections、物化視圖 Materialized views) - 攝取大量資料 (批次插入、Kafka 整合) - 從 PostgreSQL/MySQL 遷移至 ClickHouse 進行分析 - 實作即時儀表板或時間序列分析 ## 概述 ClickHouse 是一個用於線上分析處理 (OLAP) 的欄位導向資料庫管理系統 (DBMS)。它針對大型資料集上的快速分析查詢進行了優化。 **關鍵特性:** - 欄位導向儲存 - 資料壓縮 - 平行查詢執行 - 分散式查詢 - 即時分析 ## 資料表設計模式 ### MergeTree 引擎 (最常用) ```sql CREATE TABLE markets_analytics ( date Date, market_id String, market_name String, volume UInt64, trades UInt32, unique_traders UInt32, avg_trade_size Float64, created_at DateTime ) ENGINE = MergeTree() PARTITION BY toYYYYMM(date) ORDER BY (date, market_id) SETTINGS index_granularity = 8192; ``` ### ReplacingMergeTree (去重) ```sql -- 適用於可能存在重複的資料 (例如來自多個來源) CREATE TABLE user_events ( event_id String, user_id String, event_type String, timestamp DateTime, properties String ) ENGINE = ReplacingMergeTree() PARTITION BY toYYYYMM(timestamp) ORDER BY (user_id, event_id, timestamp) PRIMARY KEY (user_id, event_id); ``` ### AggregatingMergeTree (預聚合) ```sql -- 用於維護聚合指標 CREATE TABLE market_stats_hourly ( hour DateTime, market_id String, total_volume AggregateFunction(sum, UInt64), total_trades AggregateFunction(count, UInt32), unique_users AggregateFunction(uniq, String) ) ENGINE = AggregatingMergeTree() PARTITION BY toYYYYMM(hour) ORDER BY (hour, market_id); -- 查詢聚合資料 SELECT hour, market_id, sumMerge(total_volume) AS volume, countMerge(total_trades) AS trades, uniqMerge(unique_users) AS users FROM market_stats_hourly WHERE hour >= toStartOfHour(now() - INTERVAL 24 HOUR) GROUP BY hour, market_id ORDER BY hour DESC; ``` ## 查詢優化模式 ### 高效率過濾 ```sql -- ✅ 推薦 (GOOD):優先使用索引欄位 SELECT * FROM markets_analytics WHERE date >= '2025-01-01' AND market_id = 'market-123' AND volume > 1000 ORDER BY date DESC LIMIT 100; -- ❌ 錯誤 (BAD):先過濾非索引欄位 SELECT * FROM markets_analytics WHERE volume > 1000 AND market_name LIKE '%election%' AND date >= '2025-01-01'; ``` ### 聚合 (Aggregations) ```sql -- ✅ 推薦 (GOOD):使用 ClickHouse 特有的聚合函式 SELECT toStartOfDay(created_at) AS day, market_id, sum(volume) AS total_volume, count() AS total_trades, uniq(trader_id) AS unique_traders, avg(trade_size) AS avg_size FROM trades WHERE created_at >= today() - INTERVAL 7 DAY GROUP BY day, market_id ORDER BY day DESC, total_volume DESC; -- ✅ 使用 quantile 獲取百分位數 (比 percentile 更高效) SELECT quantile(0.50)(trade_size) AS median, quantile(0.95)(trade_size) AS p95, quantile(0.99)(trade_size) AS p99 FROM trades WHERE created_at >= now() - INTERVAL 1 HOUR; ``` ### 視窗函式 (Window Functions) ```sql -- 計算累計總量 (Running totals) SELECT date, market_id, volume, sum(volume) OVER ( PARTITION BY market_id ORDER BY date ROWS BETWEEN UNBOUNDED PRECEDING AND CURRENT ROW ) AS cumulative_volume FROM markets_analytics WHERE date >= today() - INTERVAL 30 DAY ORDER BY market_id, date; ``` ## 資料插入模式 ### 批次插入 (強烈建議) ```typescript import { ClickHouse } from 'clickhouse' const clickhouse = new ClickHouse({ url: process.env.CLICKHOUSE_URL, port: 8123, basicAuth: { username: process.env.CLICKHOUSE_USER, password: process.env.CLICKHOUSE_PASSWORD } }) // ✅ 批次插入 (高效) async function bulkInsertTrades(trades: Trade[]) { const values = trades.map(trade => `( '${trade.id}', '${trade.market_id}', '${trade.user_id}', ${trade.amount}, '${trade.timestamp.toISOString()}' )`).join(',') await clickhouse.query(` INSERT INTO trades (id, market_id, user_id, amount, timestamp) VALUES ${values} `).toPromise() } // ❌ 單筆插入 (慢) async function insertTrade(trade: Trade) { // 絕對不要在迴圈中這樣做! await clickhouse.query(` INSERT INTO trades VALUES ('${trade.id}', ...) `).toPromise() } ``` ### 串流插入 (Streaming Insert) ```typescript // 用於持續性的資料攝取 import { createWriteStream } from 'fs' import { pipeline } from 'stream/promises' async function streamInserts() { const stream = clickhouse.insert('trades').stream() for await (const batch of dataSource) { stream.write(batch) } await stream.end() } ``` ## 物化視圖 (Materialized Views) ### 即時聚合 ```sql -- 建立用於每小時統計的物化視圖 CREATE MATERIALIZED VIEW market_stats_hourly_mv TO market_stats_hourly AS SELECT toStartOfHour(timestamp) AS hour, market_id, sumState(amount) AS total_volume, countState() AS total_trades, uniqState(user_id) AS unique_users FROM trades GROUP BY hour, market_id; -- 查詢物化視圖 SELECT hour, market_id, sumMerge(total_volume) AS volume, countMerge(total_trades) AS trades, uniqMerge(unique_users) AS users FROM market_stats_hourly WHERE hour >= now() - INTERVAL 24 HOUR GROUP BY hour, market_id; ``` ## 效能監控 ### 查詢效能 ```sql -- 檢查慢查詢 SELECT query_id, user, query, query_duration_ms, read_rows, read_bytes, memory_usage FROM system.query_log WHERE type = 'QueryFinish' AND query_duration_ms > 1000 AND event_time >= now() - INTERVAL 1 HOUR ORDER BY query_duration_ms DESC LIMIT 10; ``` ### 資料表統計 ```sql -- 檢查表大小 SELECT database, table, formatReadableSize(sum(bytes)) AS size, sum(rows) AS rows, max(modification_time) AS latest_modification FROM system.parts WHERE active GROUP BY database, table ORDER BY sum(bytes) DESC; ``` ## 常見分析查詢 ### 時間序列分析 ```sql -- 每日活躍使用者 (DAU) SELECT toDate(timestamp) AS date, uniq(user_id) AS daily_active_users FROM events WHERE timestamp >= today() - INTERVAL 30 DAY GROUP BY date ORDER BY date; -- 留存分析 (Retention) SELECT signup_date, countIf(days_since_signup = 0) AS day_0, countIf(days_since_signup = 1) AS day_1, countIf(days_since_signup = 7) AS day_7, countIf(days_since_signup = 30) AS day_30 FROM ( SELECT user_id, min(toDate(timestamp)) AS signup_date, toDate(timestamp) AS activity_date, dateDiff('day', signup_date, activity_date) AS days_since_signup FROM events GROUP BY user_id, activity_date ) GROUP BY signup_date ORDER BY signup_date DESC; ``` ### 漏斗分析 (Funnel Analysis) ```sql -- 轉換漏斗 SELECT countIf(step = 'viewed_market') AS viewed, countIf(step = 'clicked_trade') AS clicked, countIf(step = 'completed_trade') AS completed, round(clicked / viewed * 100, 2) AS view_to_click_rate, round(completed / clicked * 100, 2) AS click_to_completion_rate FROM ( SELECT user_id, session_id, event_type AS step FROM events WHERE event_date = today() ) GROUP BY session_id; ``` ### 分群分析 (Cohort Analysis) ```sql -- 按註冊月份劃分的使用者分群 SELECT toStartOfMonth(signup_date) AS cohort, toStartOfMonth(activity_date) AS month, dateDiff('month', cohort, month) AS months_since_signup, count(DISTINCT user_id) AS active_users FROM ( SELECT user_id, min(toDate(timestamp)) OVER (PARTITION BY user_id) AS signup_date, toDate(timestamp) AS activity_date FROM events ) GROUP BY cohort, month, months_since_signup ORDER BY cohort, months_since_signup; ``` ## 資料管道模式 (Data Pipeline Patterns) ### ETL 模式 ```typescript // 擷取 (Extract)、轉換 (Transform)、載入 (Load) async function etlPipeline() { // 1. 從來源擷取 const rawData = await extractFromPostgres() // 2. 轉換 const transformed = rawData.map(row => ({ date: new Date(row.created_at).toISOString().split('T')[0], market_id: row.market_slug, volume: parseFloat(row.total_volume), trades: parseInt(row.trade_count) })) // 3. 載入至 ClickHouse await bulkInsertToClickHouse(transformed) } // 定期執行 setInterval(etlPipeline, 60 * 60 * 1000) // 每小時一次 ``` ### 變更資料擷取 (CDC) ```typescript // 監聽 PostgreSQL 變更並同步至 ClickHouse import { Client } from 'pg' const pgClient = new Client({ connectionString: process.env.DATABASE_URL }) pgClient.query('LISTEN market_updates') pgClient.on('notification', async (msg) => { const update = JSON.parse(msg.payload) await clickhouse.insert('market_updates', [ { market_id: update.id, event_type: update.operation, // INSERT, UPDATE, DELETE timestamp: new Date(), data: JSON.stringify(update.new_data) } ]) }) ``` ## 最佳實踐 ### 1. 分區策略 (Partitioning Strategy) - 按時間分區 (通常是按月或按日)。 - 避免分區過多 (影響效能)。 - 分區鍵使用 DATE 類型。 ### 2. 排序鍵 (Ordering Key) - 將最常被過濾的欄位排在前面。 - 考量基數 (Cardinality,基數高的排在前面)。 - 排序會影響壓縮效果。 ### 3. 資料型別 - 使用最小且合適的型別 (UInt32 vs UInt64)。 - 重複出現的字串使用 LowCardinality。 - 類別資料使用 Enum。 ### 4. 應避免的做法 - SELECT * (請指定欄位)。 - FINAL (應在查詢前進行資料合併)。 - 過多的 JOIN (分析時建議進行去正規化 Denormalize)。 - 少量頻繁的插入 (應改為批次插入)。 ### 5. 監控 - 追蹤查詢效能。 - 監控磁碟使用量。 - 檢查合併 (Merge) 操作。 - 審查慢查詢日誌。 **請記住**:ClickHouse 在分析工作負載方面表現卓越。請根據您的查詢模式設計資料表、使用批次插入,並善用物化視圖來進行即時聚合。