搜索历史基本原理实现即时自动补全联想搜索技巧

作者:全村最野的狗 时间:2023-05-31 22:02:04 

实现搜索历史-[即时自动补全&联想搜索]

无论是新闻、内容、还是电商平台,联想输入已经成为搜索功能的标配,早已不是什么新鲜事物。我们随便打开一个搜索引擎或者是电商平台,当我们在输入框输入拼音或者文字时就会看到输入框下方弹出有意义的搜索建议,提示我们是不是想要输入“以下”内容,帮助我们补齐输入或是修正错误的输入,优化我们的搜索体验。

搜索历史基本原理实现即时自动补全联想搜索技巧

在上图示例中,我们可以看到,输入关键字 联想搜索,Google 搜索会联想到联想搜索elasticsearch联想搜索,好处就是,我们无须输入完整的关键字即可轻松完成针对这些 topics 的搜索。

今天我们实现的功能和联想搜索有一点差别,我们是根据用户隔离,基于个人搜索历史的联想搜索

如何实现基于个人搜索历史的联想推荐

一个好的自动补全器必须是快速的,并且在用户键入下一个字符后立即更新联想词列表。自动补全器的核心是一个函数,它接受输入的前缀,并搜索以给定前缀开头的词汇或语句列表。通常来说,只需要返回少量的数目即可。

架构图

搜索历史基本原理实现即时自动补全联想搜索技巧

词汇表实现

实现方式有很多种,例如前缀树实现,有限状态自动机(DFA)实现等等。

这里采用Redis ZSET数据结构快速实现。

  • Redis 有序集合和集合一样也是 string 类型元素的集合,且不允许重复的成员

  • 不同的是每个元素都会关联一个 double 类型的分数。redis 正是通过分数来为集合中的成员进行从小到大的排序。

type zset
key search-history-common
key search-history-user:1
key search-history-user:2
key search-history-user:3

搜索历史基本原理实现即时自动补全联想搜索技巧

搜索历史基本原理实现即时自动补全联想搜索技巧

备注:

  • 常用搜索词库数据统计规则:定时取出所有人词库中排名靠前n位的搜索项并放入常用搜索库中

  • 分数值 = 原有分数值*1.01+1.01 (为什么用一元函数,因为可以让常用词和不常用词更快的区分开)

  • 分数值初始值为 1

实现原理

新增关键字操作

  • 直接添加 默认score = 1

  • 添加失败查询score

  • 设置新score = score*1.1+1.1 (注意zset不能重新设置score

  • 缓存完成

搜索历史基本原理实现即时自动补全联想搜索技巧

# 计算新score
# 新score应该 = score*1.1+1.1 但是 需要用ZADD,所以需要换算,(score*1.1+1.1)-score = a
ZADD key a member
# 化简得到
ZADD key score*0.1+1.1 member
# 添加 member = 1 返回 score
# 如果存在则添加失败 返回 0
ZADD key 1 member
# 获取分,不存在返回 null
ZSCORE key member
# 对某个键加上增量
ZADD key 1 member

删除关键字操作

  • 直接删除

# 删除成功返回 1,如果一个zset下没有item, zset也会被自动删除
ZREM key member

查询推荐列表操作

  • 全量查询当前用户词汇表

  • 使用String.contains 或者其他框架过滤出推荐词

  • 返回推荐列表到前端

搜索历史基本原理实现即时自动补全联想搜索技巧

# 全量查询 zset key(从小到大)
ZRANGE key 0 -1
member2
member
# 全量查询 zset key(从大到小)
ZRANGE key 0 -1 WITHSCORES
member2
2
member
6

来源:https://juejin.cn/post/7199269310296096828

标签:搜索历史,自动补全,联想搜索
0
投稿

猜你喜欢

  • Go数组与切片轻松掌握

    2024-02-12 19:55:36
  • 利用python判断字母大小写的几种方法小结

    2022-05-10 16:41:49
  • 淘宝网获亚洲最佳在线客户体验大奖

    2009-03-31 12:55:00
  • 5行Python代码实现电脑永不息屏

    2023-04-04 01:26:37
  • 提升Python运行速度的5个小技巧

    2021-07-28 20:25:24
  • 在python tkinter中Canvas实现进度条显示的方法

    2023-06-04 18:12:56
  • sql查询语句之平均分、最高最低分及排序语句

    2024-01-19 06:43:34
  • python实现双链表

    2022-06-20 01:47:48
  • Vuex中如何getters动态获取state的值

    2024-05-28 15:54:23
  • 浅谈keras 模型用于预测时的注意事项

    2022-10-16 13:23:04
  • 在js(jquery)中获得文本框焦点和失去焦点的方法

    2024-04-26 17:14:20
  • PHP开启opcache提升代码性能

    2023-06-17 04:39:55
  • SQL Join的一些总结(实例)

    2012-08-21 10:19:29
  • Python环境搭建过程从安装到Hello World

    2023-03-03 07:41:36
  • python Pandas之DataFrame索引及选取数据

    2023-01-01 02:27:10
  • logrus日志自定义格式操作

    2024-04-26 17:20:24
  • 运行asp.net程序 报错:磁盘空间不足

    2011-11-03 17:16:22
  • python 实现list或string按指定分段

    2023-10-30 02:04:20
  • 用electron 打包发布集成vue2.0项目的操作过程

    2024-05-09 15:22:59
  • python中的路径拼接问题

    2021-02-06 11:28:56
  • asp之家 网络编程 m.aspxhome.com