haystack+whoosh

haystack: 全文检索框架(对检索引擎包装下,抹掉差异 提供统一接口,方便码农)。该框架支持Elasticsearch,Whoosh, XapianSolr搜索引擎。
Elasticsearch:据说是java写的。Elastic(有弹性的,灵活 可伸展的),大概说明其 操作的灵活性。
Xapian: apian蜜蜂,X apian,大概使用蜜蜂的嗅探能力,来说明其搜索范围之广。

whoosh:检索引擎。(单词原意:飞快的移动。大概是说 其检索的速度很快,咻的一下 就找到了)

检索引擎(whoosh)的作用: 对 表中的某些字段 进行 关键词分析,在 关键词 和 表中其它记录s 之间 建立联系(索引表)。


1. 安装django-haystack, whoosh, jieba(切换到项目目录下)
$ cd ~/PycharmProjects/dailyfresh
$ pip install whoosh 
$ pip install django-haystack
$ pip install jieba 
2. 在Django项目的settings.py里面的INSTALLED_APPS添加Haystack
add 'haystack' to INSTALLED_APPS
3. 修改settings.py文件, 配置引擎

配置haystack使用什么引擎,以及配置检索文件的存放路径

# 配置 全文检索框架(haystack)使用 检索引擎(whoosh)
HAYSTACK_CONNECTIONS = {
    'default': {
        # 使用whoosh引擎
        # haystack 通过 下面路径 中的 WhooshEngine类,来使用whoosh检索引擎
        # WhooshEngine的路径 venv/lib/site-packages/haystack/backends/whoosh_backend.py WhooshEngine类
        'ENGINE': 'haystack.backends.whoosh_backend.WhooshEngine',
        # 索引文件的 存放路径,所有的 索引文件 都存放在 该目录下。生成索引文件时,自动 在目录(BASE_DIR)下 创建目录(whoosh_index)
        'PATH': os.path.join(BASE_DIR, 'whoosh_index'),
    }
}

# 检索引擎(whoosh)的作用: 对 表中的某些字段 进行 关键词分析,在 关键词 和 表中其它记录s 之间 建立联系(索引表)
# 表中 字段内容 发生变化时,索引 也应发生变化 来适应 字段内容的变化。

# 当表中数据 发生变化(添加 删除 修改)时,自动生成 新的索引(替换 旧的索引)
HAYSTACK_SIGNAL_PROCESSOR = 'haystack.signals.RealtimeSignalProcessor'
4. 对某个django的app创建索引

goods应用 目录下面建立search_indexes.py文件,文件名不能更改。(goods是你某个app的名称,可替换为 具体的app名称)
search_indexes.py文件中,只需更改3处地方:
1)导入 需要建立索引的模型类(GoodsSKU)
2)索引类的名称(命名规则:模型类+Index)
3)get_model返回的 模型类(改为 你想建立索引的模型类)

# 定义索引类

from haystack import indexes

# 导入模型类
from apps.goods.models import GoodsSKU



# 对某个类的 某些数据 建立索引
# 索引类 类名 的格式:模型类名+Index
class GoodsSKUIndex(indexes.SearchIndex, indexes.Indexable):
    # text: 索引字段。document=True: 指定 该text为 索引字段。
    # use_template=True 指定 对表中的哪些字段进行关键词分析 建立索引文件。对字段的说明,是 放在一个文件中的。
    text = indexes.CharField(document=True, use_template=True)

    def get_model(self):
        # 返回 需要检索的 模型类
        return GoodsSKU

    # 返回 数据。对该数据 建立索引。
    def index_queryset(self, using=None):
        return self.get_model().objects.all()
5. 对哪些字段建立索引

创建目录templates/search/indexes/goods/,在templates/search/indexes/goods/目录下, 新建文件goodssku_text.txt。(goods为 APP名。goodssku为模型名,并且 全部 改为小写。模型名的 后缀_text.txt,不能更改)

# object代表 GoodsSKU模型名,
# 因为文件名已经指定了 模型类的路径和名称
# name, desc, goods, detail为字段名
# 根据表中的哪些字段 建立 索引数据

{{ object.name }}   # 根据商品的名称name 建立索引
{{ object.desc }}   #          简介desc 建立索引
{{ object.goods.detail }}   # 根据商品的详情detail 建立索引(详情在spu表中)
6. 命令生成索引文件

在项目目录下,执行命令:

$ python manage.py rebuild_index

如果成功生成索引文件,会在项目的根目录下,出现下图中的目录和文件。

成功生成索引文件 图

7. 添加 搜索框的表单,并设置

templates/base.html中 添加搜索框表单
form表单的method=‘get'不能改变,搜索栏的文本框的name="q"也是固定不变的。
action是 表单的 提交地址,对搜索的处理 交由 搜索引擎 进行处理。对表单action的配置,表单地址的配置 可随意更改,但是需要与项目目录下urls.py中的 全文检索框架的url配置 一致。

    <div class="search_con fl">
        {# 下面form表单的method 是'get'不能改变,搜索栏的文本框的name="q" 也是固定不变的 #}
        {# action 表单的 提交地址,对搜索的处理 交由 搜索引擎 进行处理 #}
        <form method="get" action="/search">
            <input type="text" class="input_text fl" name="q" placeholder="搜索商品">
            <input type="submit" class="input_btn fr" name="" value="搜索">
        </form>

    </div>
8. 配置全文检索的url

在项目目录下urls.py中,添加 全文检索框架的url配置(url地址 要和 表单action地址 一致)

re_path(r'^search', include('haystack.urls')),  # 全文检索框架
9. 全文检索搜索过程

搜索到结果后,haystack会把搜索出来的结果 传递给 templates/search目录下的search.html, 传递的上下文 有:

haystack会对搜索的结果数据 进行分页,并把 当前页的page对象 传递给search.html

  • query: 搜索关键字.
  • page: 当前页的page对象。遍历page对象,获取到的是SearchResult类的 实例对象,该对象的object属性 才是模型类的对象.page对象object属性包含模型类的对象,这些模型类的对象是 搜索的结果。
  • paginator: 当前分页的pagenitor对象.
    通过HAYSTACK_SEARCH_RESULTS_PER_PAGE 可以控制 每页显示搜索结果的数量。
10. 更改whoosh引擎 使用的 关键词分析 类

1)安装jieba
在项目目录下

pip install jieba
  1. 更改whoosh默认的 关键词分析类

找到项目虚拟环境下haystack目录(venv/lib/python3.6/site-packages/haystack/backends/)。
在该目录下 新建文件ChineseAnalyzer.py, 写入 如下 内容。

import jieba
from whoosh.analysis import Tokenizer, Token

class ChineseTokenizer(Tokenizer):
    def __call__(self, value, positions=False, chars=False,
                 keeporiginal=False, removestops=True,
                 start_pos=0, start_char=0, mode='', **kwargs):
        t = Token(positions, chars, removestops=removestops, mode=mode, **kwargs)
        seglist = jieba.cut(value, cut_all=True)
        for w in seglist:
            t.original = t.text = w
            t.boost = 1.0
            if positions:
                t.pos = start_pos + value.find(w)
            if chars:
                t.startchar = start_char + value.find(w)
                t.endchar = start_char + value.find(w) + len(w)
            yield t

def ChineseAnalyzer():
    return ChineseTokenizer()

更改whoosh_backend.py中的词语分析类:

# 复制whoosh_backend.py并改名为whoosh_cn_backend.py
$ cp whoosh_backend.py whoosh_cn_backend.py

$ vi whoosh_cn_backend.py
# 添加from .ChineseAnalyzer import ChineseAnalyzer一行,导入ChineseAnalyzer方法

# 命令行模式搜索StemmingAnalyzer(), 将其修改为ChineseAnalyzer()

# :wq

更改settings.py中的 HAYSTACK_CONNECTIONS配置, 使其使用whoosh_cn_backend

# 配置 全文检索框架(haystack)使用 检索引擎(whoosh)
HAYSTACK_CONNECTIONS = {
    'default': {
        # 使用whoosh引擎
        # haystack 通过 下面路径 中的 WhooshEngine类,来使用whoosh检索引擎
        # WhooshEngine的路径 venv/lib/site-packages/haystack/backends/whoosh_backend.py WhooshEngine类
        # 'ENGINE': 'haystack.backends.whoosh_backend.WhooshEngine',
        'ENGINE': 'haystack.backends.whoosh_cn_backend.WhooshEngine',
        # 索引文件的 存放路径,所有的 索引文件 都存放在 该目录下。生成索引文件时,自动 在目录(BASE_DIR)下 创建目录(whoosh_index)
        'PATH': os.path.join(BASE_DIR, 'whoosh_index'),
    }
}

# 检索引擎(whoosh)的作用: 对 表中的某些字段 进行 关键词分析,在 关键词 和 表中其它记录s 之间 建立联系(索引表)
11. 重新生成索引文件

切换到项目目录下

$ python manage.py rebuild_index
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,335评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,895评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,766评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,918评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,042评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,169评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,219评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,976评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,393评论 1 304
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,711评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,876评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,562评论 4 336
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,193评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,903评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,142评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,699评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,764评论 2 351

推荐阅读更多精彩内容