Cassandra 有两个主键并使用第二个

Rim*_*der 3 cassandra cassandra-3.0

我在 Cassandra 中创建了下一个表

create table yyy (
 index1 int,
 index2 int,
 name text,
 primary key ((index1, index2), name)
)
Run Code Online (Sandbox Code Playgroud)

当我只按 index1 搜索时,完美。没关系!

select * from yyy where index1 ...
Run Code Online (Sandbox Code Playgroud)

买我不能按 index2 搜索

ReadFailure:来自服务器的错误:code=1300 [Replica(s) failed to execute read] message="操作失败 - 收到 0 个响应和 1 个失败" info={'failures': 1, 'received_responses': 0, 'required_responses' : 1, '一致性': 'ONE'}

如何按index2搜索?

Jus*_*ron 7

与关系数据库相比,Cassandra 在查询数据方面的灵活性要低得多。

您所称的index1index2实际上在 Cassandra 中一起称为复合分区键。为了使用 WHERE 子句,您必须始终提供整个分区键。

例如,这些查询将成功:

SELECT * FROM yyy WHERE index1 = 1 AND index2 = 2;

SELECT * FROM yyy WHERE index1 = 1 AND index2 = 2 AND name = 'fred';

但是这些查询都会失败:

SELECT * FROM yyy WHERE index1 = 1;

SELECT * FROM yyy WHERE index2 = 2;

SELECT * FROM yyy WHERE index1 = 1 AND name = 'fred';

这是因为 Cassandra 使用分区键来确定分区将存储在哪个节点上。如果您只提供一半的密钥,Cassandra 无法判断哪个节点包含数据。

在为 Cassandra 进行数据建模时,如果需要无法由单个表提供的某些查询,您通常会非规范化表。例如,如果您需要单独查询 index2,您可以使用此架构创建第二个表:

create table zzz (
 index1 int,
 index2 int,
 name text,
 primary key ((index2), index1, name)
)
Run Code Online (Sandbox Code Playgroud)

现在你可以执行 SELECT * FROM zzz WHERE index2 = 2;

请注意,在这种情况下,您还需要确保您的index2分区不会变得太大。

还有一些其他功能允许更灵活的查询,尽管您应该在使用它们之前完全了解它们的局限性,因为如果不按预期使用它们可能会导致严重的性能或操作问题:

如果您不熟悉 Cassandra,我强烈建议您阅读 Cassandra 数据建模最佳实践,因为它不像关系数据库那样简单。有很多很好的博客和资源 - 谷歌是你的朋友!