Django+haystack+whoosh+jieba全文檢索實(shí)現(xiàn)

簡(jiǎn)介

全文檢索主要用在大數(shù)據(jù)量時(shí)多字段模糊檢索上能較大的提高檢索效率。django實(shí)現(xiàn)全文檢索功能主要靠haystack框架盛卡,而用的最多的全文檢索引擎就是whoosh,jieba主要用于中文分詞筑凫,whoosh自帶的分詞是英文的滑沧。要實(shí)現(xiàn)以上組合的全文檢索,首先要安裝這些模塊:

pip install django-haystack
pip install whoosh
pip install jieba

配置haystack框架和whoosh引擎

安裝好以上模塊后巍实,接下來要到項(xiàng)目的settings.py中添加haystack應(yīng)用滓技,配置whoosh搜索引擎。

#settings.py
INSTALLED_APPS = [
    'django.contrib.admin',
    'django.contrib.auth',
    'django.contrib.contenttypes',
    'django.contrib.sessions',
    'django.contrib.messages',
    'django.contrib.staticfiles',
    'haystack',#全文檢索框架
    'book',
    'user',
    'recommend',
    'library',
    'comment',
]

HAYSTACK_CONNECTIONS = {
    'default': {
        #使用whoosh引擎
        'ENGINE': 'haystack.backends.whoosh_cn_backend.WhooshEngine',
        #索引文件路徑
        'PATH': os.path.join(BASE_DIR, 'whoosh_index'),
    }
}

接下來到項(xiàng)目的urls.py中添加全文檢索的路由棚潦。

#項(xiàng)目的urls.py
urlpatterns = [
    path('admin/', admin.site.urls),
    path('',include('book.urls')),
    path('book/', include('book.urls')),
    path('user/', include('user.urls')),
    path('recommend/', include('recommend.urls')),
    path('library/', include('library.urls')),
    path('comment/', include('comment.urls')),
    path('search/', include('haystack.urls')),#全文檢索路由
]

然后在要做全文檢索的appbook下創(chuàng)建search_indexes.py文件令漂,該文件是固定命名,內(nèi)容如下:

from haystack import indexes#導(dǎo)入索引
from .models import Book_info#導(dǎo)入模型
#Book_infoIndex是固定格式命名,Book_info是你models.py中的類名
class Book_infoIndex(indexes.SearchIndex, indexes.Indexable):
    text = indexes.CharField(document=True, use_template=True)
 
    def get_model(self):
        return Book_info
 
    def index_queryset(self, using=None):
        return self.get_model().objects.all()

然后到項(xiàng)目下的templates文件夾中依次創(chuàng)建search/indexes/book/目錄叠必,book是你需要使用全文檢索的app名荚孵,根據(jù)實(shí)際情況更改為自己的app名,book前面的目錄名是固定寫法纬朝,不能更改收叶。
在book目錄下以"類名_text.txt"格式創(chuàng)建搜索引擎的索引文件,如"book_info_text.txt"共苛,book_info是bookapp下的models.py中的一個(gè)類class Book_info(models.Model):判没。
創(chuàng)建好txt文件后,在文件中輸入要索引的字段隅茎。

{{ object.book_name }}
{{ object.book_author }}
{{ object.book_press }}

上面的內(nèi)容我設(shè)置了Book_info數(shù)據(jù)庫(kù)中的book_name澄峰、book_authorbook_press三個(gè)字段的全文索引患膛。

中文分詞設(shè)置

接下來設(shè)置中文分詞摊阀,在系統(tǒng)根目錄中查找haystack(windows系統(tǒng)查找對(duì)象是我的電腦,linux系統(tǒng)使用find / -name haystack)踪蹬,找到C:\Users\cg\AppData\Local\Programs\Python\Python37-32\Lib\site-packages\haystack\backendspython安裝包下的這個(gè)文件夾胞此,在該文件夾下創(chuàng)建ChineseAnalyzer.py,內(nèi)容如下:

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()

復(fù)制whoosh_backend.py跃捣,將名稱改為whoosh_cn_backend.py漱牵,打開該文件,引入中文分析類:

from .ChineseAnalyzer import ChineseAnalyzer

查找文件中analyzer=StemmingAnalyzer()疚漆,將其改為analyzer=ChineseAnalyzer()

完成以上的配置就可以建立索引文件了酣胀,在項(xiàng)目終端下輸入命令重建索引:

python manage.py rebuild_index

創(chuàng)建好索引文件后通過python manage.py update_index來更新索引文件。

全文索引的使用

更改原來的檢索模板文件:

<form action="/search/" method="GET">
       <div class="input-group mb-3">
              <input type="text" name="q" class="form-control" autocomplete="off" required placeholder="可檢索字段-書名/作者/出版社">
              <div class="input-group-append">
                    <button class="btn btn-info" type="submit">搜索</button>  
              </div>
       </div>
</form>

上面的action參數(shù)對(duì)應(yīng)上文在項(xiàng)目urls.py中設(shè)置的路由娶聘,代表表單提交到全文檢索路由闻镶,input輸入中的name=q參數(shù)是haystack的固定寫法,q代表查詢的關(guān)鍵詞丸升。
用戶提交檢索后铆农,系統(tǒng)將檢索詞提交給haystack,經(jīng)過haystack查詢后狡耻,默認(rèn)結(jié)果返回到項(xiàng)目根目錄下templates/search/search.html文件墩剖,結(jié)果中主要包含以下關(guān)鍵參數(shù):
query:查詢的關(guān)鍵詞。
page:當(dāng)前頁(yè)的page對(duì)象夷狰,通過該對(duì)象獲取查詢的數(shù)據(jù)岭皂。
paginator:分頁(yè)對(duì)象。
模板中主要代碼如下:

<div class="clearfix">
    <div class="alert alert-info">
        檢索到關(guān)于:&nbsp;<b>“{{ query }}”</b>&nbsp;的圖書,當(dāng)前第&nbsp;<b>{{ page.number }} </b>&nbsp;頁(yè)
    </div>
    {# 遍歷檢索圖書結(jié)果 #}
    {% for item in page %}
    <a href="{% url 'book_detail' %}?ids={{ item.object.book_id }}" style="color:black">
        <div class="responsive">
            <div class="img">
                
                    <img src="{{ item.object.book_pic }}" alt="" width="300" height="200">
                
                <div class="desc">{{ item.object.book_name }}</div>
                <div class="desc"><span>{{ item.object.book_press }}</span></div>
            </div>
        </div>
    </a>
    {% endfor %}
</div>

{# 分頁(yè) #}
  <div class="center">
    <ul class="pagination">
      {% if page.has_previous %}
      <li class="page-item"><a class="page-link" href="?q={{query}}&amp;page={{ page.previous_page_number }}">上一頁(yè)</a></li>
      {% endif %}
      {% if page.has_next %}
      <li class="page-item"><a class="page-link" href="?q={{query}}&amp;page={{ page.next_page_number }}">下一頁(yè)</a></li>
      {% endif %}
    </ul>
  </div>

需要注意的是:通過page對(duì)象遍歷獲取的對(duì)象屬性沼头,需要在中間增加object爷绘,否則獲取不到對(duì)象的屬性书劝。檢索結(jié)果如下:

image.png

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市揉阎,隨后出現(xiàn)的幾起案子庄撮,更是在濱河造成了極大的恐慌,老刑警劉巖毙籽,帶你破解...
    沈念sama閱讀 221,548評(píng)論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件洞斯,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡坑赡,警方通過查閱死者的電腦和手機(jī)烙如,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,497評(píng)論 3 399
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來毅否,“玉大人亚铁,你說我怎么就攤上這事∶樱” “怎么了徘溢?”我有些...
    開封第一講書人閱讀 167,990評(píng)論 0 360
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)捆探。 經(jīng)常有香客問我然爆,道長(zhǎng),這世上最難降的妖魔是什么黍图? 我笑而不...
    開封第一講書人閱讀 59,618評(píng)論 1 296
  • 正文 為了忘掉前任曾雕,我火速辦了婚禮,結(jié)果婚禮上助被,老公的妹妹穿的比我還像新娘剖张。我一直安慰自己,他們只是感情好揩环,可當(dāng)我...
    茶點(diǎn)故事閱讀 68,618評(píng)論 6 397
  • 文/花漫 我一把揭開白布搔弄。 她就那樣靜靜地躺著,像睡著了一般丰滑。 火紅的嫁衣襯著肌膚如雪肯污。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 52,246評(píng)論 1 308
  • 那天吨枉,我揣著相機(jī)與錄音,去河邊找鬼哄芜。 笑死貌亭,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的认臊。 我是一名探鬼主播圃庭,決...
    沈念sama閱讀 40,819評(píng)論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了剧腻?” 一聲冷哼從身側(cè)響起拘央,我...
    開封第一講書人閱讀 39,725評(píng)論 0 276
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎书在,沒想到半個(gè)月后灰伟,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 46,268評(píng)論 1 320
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡儒旬,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,356評(píng)論 3 340
  • 正文 我和宋清朗相戀三年栏账,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片栈源。...
    茶點(diǎn)故事閱讀 40,488評(píng)論 1 352
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡挡爵,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出甚垦,到底是詐尸還是另有隱情茶鹃,我是刑警寧澤,帶...
    沈念sama閱讀 36,181評(píng)論 5 350
  • 正文 年R本政府宣布艰亮,位于F島的核電站闭翩,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏垃杖。R本人自食惡果不足惜男杈,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,862評(píng)論 3 333
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望调俘。 院中可真熱鬧伶棒,春花似錦、人聲如沸彩库。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,331評(píng)論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)骇钦。三九已至宛渐,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間眯搭,已是汗流浹背窥翩。 一陣腳步聲響...
    開封第一講書人閱讀 33,445評(píng)論 1 272
  • 我被黑心中介騙來泰國(guó)打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留鳞仙,地道東北人寇蚊。 一個(gè)月前我還...
    沈念sama閱讀 48,897評(píng)論 3 376
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像棍好,于是被迫代替她去往敵國(guó)和親仗岸。 傳聞我的和親對(duì)象是個(gè)殘疾皇子允耿,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,500評(píng)論 2 359

推薦閱讀更多精彩內(nèi)容