跳转到主要内容
在 ClickHouse Cloud 上使用 Memory 表引擎时,数据不会在所有节点之间复制 (这是设计使然) 。为保证所有查询都路由到同一节点,并让 Memory 表引擎按预期工作,可以采用以下任一方式:
  • 在同一会话中执行所有操作
  • 使用采用 TCP 或原生接口 (这会启用对粘性连接的支持) 的客户端,例如 clickhouse-client
Memory 引擎以未压缩形式将数据存储在 RAM 中。数据会按读取时接收到的原样进行存储。换句话说,从这张表读取数据完全没有开销。 并发数据访问会被同步处理。锁持有时间很短:读写操作彼此不会阻塞。 不支持索引。读取可并行进行。 对于简单查询,由于无需从磁盘读取、解压或反序列化数据,可达到最高吞吐量 (超过 10 GB/秒) 。 (需要说明的是,在很多情况下,MergeTree 引擎的吞吐量也几乎同样高。) 服务器重启后,表中的数据会消失,表会变为空。 通常,使用这种表引擎并无充分理由。不过,它可用于测试,以及在行数相对较少 (最多约 100,000,000 行) 但需要最高速度的任务中使用。 系统会将 Memory 引擎用于带有外部查询数据的临时表 (参见“处理查询的外部数据”一节) ,以及实现 GLOBAL IN (参见“IN 运算符”一节) 。 可以指定上下限来限制 Memory 引擎表的大小,从而使其能够充当环形缓冲区 (参见 引擎参数) 。

引擎参数

  • min_bytes_to_keep — 当内存表设置了大小上限时,需保留的最小字节数。
    • 默认值:0
    • 需要 max_bytes_to_keep
  • max_bytes_to_keep — 内存表中保留的最大字节数;每次插入时都会删除最旧的行 (即环形缓冲区) 。如果添加大块时,需删除的最旧一批行会使结果低于 min_bytes_to_keep 限制,则最大字节数可能会超过 указанное 限制。
    • 默认值:0
  • min_rows_to_keep — 当内存表设置了大小上限时,需保留的最小行数。
    • 默认值:0
    • 需要 max_rows_to_keep
  • max_rows_to_keep — 内存表中保留的最大行数;每次插入时都会删除最旧的行 (即环形缓冲区) 。如果添加大块时,需删除的最旧一批行会使结果低于 min_rows_to_keep 限制,则最大行数可能会超过 указанное 限制。
    • 默认值:0
  • compress - 是否压缩内存中的数据。
    • 默认值:false

用法

初始化设置
CREATE TABLE memory (i UInt32) ENGINE = Memory SETTINGS min_rows_to_keep = 100, max_rows_to_keep = 1000;
修改设置
ALTER TABLE memory MODIFY SETTING min_rows_to_keep = 100, max_rows_to_keep = 1000;
注意: bytesrows 这两个上限参数可以同时设置,不过最终会遵循 maxmin 中较小的限制值。

示例

CREATE TABLE memory (i UInt32) ENGINE = Memory SETTINGS min_bytes_to_keep = 4096, max_bytes_to_keep = 16384;

/* 1. 测试最旧的块因未达到最小阈值而不会被删除 - 3000 行 */
INSERT INTO memory SELECT * FROM numbers(0, 1600); -- 8'192 bytes

/* 2. 添加不会被删除的块 */
INSERT INTO memory SELECT * FROM numbers(1000, 100); -- 1'024 bytes

/* 3. 测试最旧的块被删除 - 9216 字节 - 1100 */
INSERT INTO memory SELECT * FROM numbers(9000, 1000); -- 8'192 bytes

/* 4. 检查超大块覆盖所有内容 */
INSERT INTO memory SELECT * FROM numbers(9000, 10000); -- 65'536 bytes

SELECT total_bytes, total_rows FROM system.tables WHERE name = 'memory' AND database = currentDatabase();
┌─total_bytes─┬─total_rows─┐
│       65536 │      10000 │
└─────────────┴────────────┘
此外,对于行:
CREATE TABLE memory (i UInt32) ENGINE = Memory SETTINGS min_rows_to_keep = 4000, max_rows_to_keep = 10000;

/* 1. 测试最旧的块因未达到最小阈值而不被删除 - 3000 行 */
INSERT INTO memory SELECT * FROM numbers(0, 1600); -- 1'600 行

/* 2. 添加不会被删除的块 */
INSERT INTO memory SELECT * FROM numbers(1000, 100); -- 100 行

/* 3. 测试最旧的块被删除 - 9216 字节 - 1100 */
INSERT INTO memory SELECT * FROM numbers(9000, 1000); -- 1'000 行

/* 4. 检查超大块覆盖所有内容 */
INSERT INTO memory SELECT * FROM numbers(9000, 10000); -- 10'000 行

SELECT total_bytes, total_rows FROM system.tables WHERE name = 'memory' AND database = currentDatabase();
┌─total_bytes─┬─total_rows─┐
│       65536 │      10000 │
└─────────────┴────────────┘
最后修改于 2026年6月10日