精确匹配查询

执行简单的精确匹配查询

精确匹配查询可以选择所有在指定字段上与特定值相匹配的文档。

你可以对多种字段类型执行精确匹配查询,具体的查询语法取决于字段类型。

本文中的示例使用以下 schema:

字段名 字段类型
description TEXT
condition TAG
price NUMERIC

你可以在 快速入门指南 中了解更多关于创建索引和加载示例数据的细节。

数值字段

在数值字段上执行精确匹配查询时,需要构造一个起始值和结束值相同的范围查询:

  1. FT.SEARCH index "@field:[value value]"
  2. FT.SEARCH index "@field:[value]" DIALECT 2 # 需要 v2.10
  3. FT.SEARCH index "@field==value" DIALECT 2 # 需要 v2.10

如在 范围查询文章 中描述的,你还可以使用 FILTER 参数:

  1. FT.SEARCH index "*" FILTER field start end

以下示例展示了如何查询价格恰好为 270 美元的自行车:

_ Redis CLI

  1. > FT.SEARCH idx:bicycle "@price:[270 270]"
  2. 1) (integer) 1
  3. 2) "bicycle:0"
  4. 3) 1) "$"
  5. 2) "{\"pickup_zone\":\"POLYGON((-74.0610 40.7578, ..."
  6. > FT.SEARCH idx:bicycle "@price:[270]" # 需要 v2.10
  7. 1) (integer) 1
  8. 2) "bicycle:0"
  9. 3) 1) "$"
  10. 2) "{\"pickup_zone\":\"POLYGON((-74.0610 40.7578, ..."
  11. > FT.SEARCH idx:bicycle "@price==270" # 需要 v2.10
  12. 1) (integer) 1
  13. 2) "bicycle:0"
  14. 3) 1) "$"
  15. 2) "{\"pickup_zone\":\"POLYGON((-74.0610 40.7578, ..."
  16. > FT.SEARCH idx:bicycle "*" FILTER price 270 270
  17. 1) (integer) 1
  18. 2) "bicycle:0"
  19. 3) 1) "$"
  20. 2) "{\"pickup_zone\":\"POLYGON((-74.0610 40.7578, ..."

提示:厌倦了使用 redis-cli?试试 Redis Insight —— 专为 Redis 开发者设计的图形化界面。

获取 Redis Insight

Python

  1. import json
  2. import redis
  3. from redis.commands.json.path import Path
  4. from redis.commands.search.field import TextField, NumericField, TagField
  5. from redis.commands.search.indexDefinition import IndexDefinition, IndexType
  6. from redis.commands.search.query import NumericFilter, Query
  7. r = redis.Redis(decode_responses=True)
  8. # create index
  9. schema = (
  10. TextField("$.description", as_name="description"),
  11. NumericField("$.price", as_name="price"),
  12. TagField("$.condition", as_name="condition"),
  13. )
  14. index = r.ft("idx:bicycle")
  15. index.create_index(
  16. schema,
  17. definition=IndexDefinition(prefix=["bicycle:"], index_type=IndexType.JSON),
  18. )
  19. # load data
  20. with open("data/query_em.json") as f:
  21. bicycles = json.load(f)
  22. pipeline = r.pipeline(transaction=False)
  23. for bid, bicycle in enumerate(bicycles):
  24. pipeline.json().set(f'bicycle:{bid}', Path.root_path(), bicycle)
  25. pipeline.execute()
  26. res = index.search(Query("@price:[270 270]"))
  27. print(res.total)
  28. # >>> 1
  29. try:
  30. res = index.search(Query("@price:[270]")) # not yet supported in redis-py
  31. print(res.total)
  32. # >>> 1
  33. assert res.total == 1
  34. except:
  35. print("'@price:[270]' syntax not yet supported.")
  36. try:
  37. res = index.search(Query("@price==270")) # not yet supported in redis-py
  38. print(res.total)
  39. # >>> 1
  40. assert res.total == 1
  41. except:
  42. print("'@price==270' syntax not yet supported.")
  43. query = Query("*").add_filter(NumericFilter("price", 270, 270))
  44. res = index.search(query)
  45. print(res.total)
  46. # >>> 1
  47. res = index.search(Query("@condition:{new}"))
  48. print(res.total)
  49. # >>> 5
  50. schema = (
  51. TagField("$.email", as_name="email")
  52. )
  53. idx_email = r.ft("idx:email")
  54. idx_email.create_index(
  55. schema,
  56. definition=IndexDefinition(prefix=["key:"], index_type=IndexType.JSON),
  57. )
  58. r.json().set('key:1', Path.root_path(), '{"email": "test@redis.com"}')
  59. try:
  60. res = idx_email.search(Query("test@redis.com").dialect(2))
  61. print(res)
  62. except:
  63. print("'test@redis.com' syntax not yet supported.")
  64. res = index.search(Query("@description:\"rough terrain\""))
  65. print(res.total)
  66. # >>> 1 (Result{1 total, docs: [Document {'id': 'bicycle:8'...)

Python Quick-Start

Node.js

  1. import assert from 'node:assert';
  2. import fs from 'node:fs';
  3. import { createClient, SchemaFieldTypes, AggregateGroupByReducers, AggregateSteps} from 'redis';
  4. const client = createClient();
  5. await client.connect().catch(console.error);
  6. // create index
  7. await client.ft.create('idx:bicycle', {
  8. '$.description': {
  9. type: SchemaFieldTypes.TEXT,
  10. AS: 'description'
  11. },
  12. '$.price': {
  13. type: SchemaFieldTypes.NUMERIC,
  14. AS: 'price'
  15. },
  16. '$.condition': {
  17. type: SchemaFieldTypes.TAG,
  18. AS: 'condition'
  19. }
  20. }, {
  21. ON: 'JSON',
  22. PREFIX: 'bicycle:'
  23. })
  24. // load data
  25. const bicycles = JSON.parse(fs.readFileSync('data/query_em.json', 'utf8'));
  26. await Promise.all(
  27. bicycles.map((bicycle, bid) => {
  28. return client.json.set(`bicycle:${bid}`, '$', bicycle);
  29. })
  30. );
  31. const res1 = await client.ft.search('idx:bicycle', '@price:[270 270]');
  32. console.log(res1.total); // >>> 1
  33. try {
  34. const res2 = await client.ft.search('idx:bicycle', '@price:[270]');
  35. console.log(res2.total); // >>> 1
  36. assert.strictEqual(res2.total, 1);
  37. } catch (err) {
  38. console.log("'@price:[270]' syntax not yet supported.");
  39. }
  40. try {
  41. const res3 = await client.ft.search('idx:bicycle', '@price==270');
  42. console.log(res3.total); // >>> 1
  43. assert.strictEqual(res3.total, 1);
  44. } catch (err) {
  45. console.log("'@price==270' syntax not yet supported.");
  46. }
  47. // FILTER is not supported
  48. // const res4 = await client.ft.search('idx:bicycle', '*', {
  49. // FILTER: {
  50. // field: 'price',
  51. // min: 270,
  52. // max: 270,
  53. // }
  54. // });
  55. // console.log(res4.total); // >>> 1
  56. const res5 = await client.ft.search('idx:bicycle', '@condition:{new}');
  57. console.log(res5.total); // >>> 5
  58. await client.ft.create('idx:email', {
  59. '$.email': {
  60. type: SchemaFieldTypes.TAG,
  61. AS: 'email'
  62. }
  63. }, {
  64. ON: 'JSON',
  65. PREFIX: 'key:'
  66. })
  67. await client.json.set('key:1', '$', { email: 'test@redis.com' });
  68. try {
  69. const res6 = await client.ft.search('idx:email', 'test@redis.com', { DIALECT: 2 });
  70. console.log(res6);
  71. } catch (err) {
  72. console.log("'test@redis.com' syntax not yet supported.");
  73. }
  74. const res7 = await client.ft.search('idx:bicycle', '@description:"rough terrain"');
  75. console.log(res7.total); // >>> 1 (Result{1 total, docs: [Document {'id': 'bicycle:8'...)