Django 博客實現(xiàn)簡單的全文搜索的示例代碼
作者:HelloGitHub-追夢人物
文中所涉及的示例代碼,已同步更新到 HelloGitHub-Team 倉庫
搜索是一個復(fù)雜的功能,但對于一些簡單的搜索任務(wù),我們可以使用 Django Model 層提供的一些內(nèi)置方法來完成。現(xiàn)在我們來為我們的博客提供一個簡單的搜索功能。
概述
博客文章通常包含標(biāo)題和正文兩個部分。當(dāng)用戶輸入某個關(guān)鍵詞進(jìn)行搜索后,我們希望為用戶顯示標(biāo)題和正文中含有被搜索關(guān)鍵詞的全部文章。整個搜索的過程如下:
- 用戶在搜素框中輸入搜索關(guān)鍵詞,假設(shè)為 “django”,然后用戶點擊了搜索按鈕提交其輸入的結(jié)果到服務(wù)器。
- 服務(wù)器接收到用戶輸入的搜索關(guān)鍵詞 “django” 后去數(shù)據(jù)庫查找文章標(biāo)題和正文中含有該關(guān)鍵詞的全部文章。
- 服務(wù)器將查詢結(jié)果返回給用戶。
整個過程就是這樣,下面來看看 Django 如何用實現(xiàn)這些過程。
將關(guān)鍵詞提交給服務(wù)器
先來回顧一下我們的 Django 博客的 Post(文章)模型:
blog/models.py
class Post(models.Model): # 標(biāo)題 title = models.CharField("標(biāo)題", max_length=70) # 正文 body = models.TextField("正文") # 其他屬性... def __str__(self): return self.title
先看到第 1 步,用戶在搜索框輸入搜索關(guān)鍵詞,因此我們要在博客上為用戶提供一個搜索表單,HTML 表單代碼大概像這樣:
templates/base.html
<form role="search" method="get" id="searchform" action="{% url 'blog:search' %}"> <input type="search" name="q" placeholder="搜索" required> <button type="submit"><span class="ion-ios-search-strong"></span></button> </form>
特別注意這里 <input type="search" name="q" placeholder="搜索" required> 中的 name 屬性,當(dāng)用戶在這個 input 中輸入搜索內(nèi)容并提交表單后,鍵入的數(shù)據(jù)會以鍵值對的形式提交服務(wù)器,這個鍵的名字就是通過 name 屬性指定的。這樣服務(wù)器就可以根據(jù) name 的值來取得用戶輸入的內(nèi)容。
用戶輸入了搜索關(guān)鍵詞并點擊了搜索按鈕后,數(shù)據(jù)就被發(fā)送給了 Django 后臺服務(wù)器。表單的 action 屬性的值為 {% url 'blog:search' %}(雖然我們還沒有寫這個視圖函數(shù)),表明用戶提交的結(jié)果將被發(fā)送給 blog 應(yīng)用下 search 視圖函數(shù)對應(yīng)的 URL。
查找含有搜索關(guān)鍵詞的文章
搜索的功能將由 search 視圖函數(shù)提供,代碼寫在 blog/views.py 里:
blog/views.py
from django.contrib import messages def search(request): q = request.GET.get('q') if not q: error_msg = "請輸入搜索關(guān)鍵詞" messages.add_message(request, messages.ERROR, error_msg, extra_tags='danger') return redirect('blog:index') post_list = Post.objects.filter(Q(title__icontains=q) | Q(body__icontains=q)) return render(request, 'blog/index.html', {'post_list': post_list})
首先我們使用 request.GET.get('q') 獲取到用戶提交的搜索關(guān)鍵詞。用戶通過表單 get 方法提交的數(shù)據(jù) Django 為我們保存在 request.GET 里,這是一個類似于 Python 字典的對象,所以我們使用 get 方法從字典里取出鍵 q 對應(yīng)的值,即用戶的搜索關(guān)鍵詞。這里字典的鍵之所以叫 q 是因為我們的表單中搜索框 input 的 name 屬性的值是 q,如果修改了 name 屬性的值,那么這個鍵的名稱也要相應(yīng)修改。
接下來我們做了一個小小的校驗,如果用戶沒有輸入搜索關(guān)鍵詞而提交了表單,我們就無需執(zhí)行查詢,我們給給用戶發(fā)一條錯誤提醒消息,這里使用了 django messages 應(yīng)用,這在 交流的橋梁:評論功能 中講過。然后將用戶重定向到首頁。這里的 redirect 函數(shù)也在那篇教程中講過。
如果用戶輸入了搜索關(guān)鍵詞,我們就通過 filter 方法從數(shù)據(jù)庫里過濾出符合條件的所有文章。這里的過濾條件是 title__icontains=q,即 title 中包含(contains)關(guān)鍵字 q,前綴 i 表示不區(qū)分大小寫。這里 icontains 是查詢表達(dá)式(Field lookups),我們在之前也使用過其他類似的查詢表達(dá)式,其用法是在模型需要篩選的屬性后面跟上兩個下劃線。Django 內(nèi)置了很多查詢表達(dá)式,建議過一遍 Django 官方留個印象,了解每個表達(dá)式的作用,以后碰到相關(guān)的需求就可以快速定位到文檔查詢其用途 Field lookups。
此外我們這里從 from django.db.models 中引入了一個新的東西:Q 對象。Q 對象用于包裝查詢表達(dá)式,其作用是為了提供復(fù)雜的查詢邏輯。例如這里 Q(title__icontains=q) | Q(body__icontains=q) 表示標(biāo)題(title)含有關(guān)鍵詞 q 或者正文(body)含有關(guān)鍵詞 q ,或邏輯使用 | 符號。如果不用 Q 對象,就只能寫成 title__icontains=q, body__icontains=q,這就變成標(biāo)題(title)含有關(guān)鍵詞 q 且正文(body)含有關(guān)鍵詞 q,就達(dá)不到我們想要的目的。
綁定 URL
有了視圖函數(shù)后記得把視圖函數(shù)映射到相應(yīng)了 URL,如下。
blog/urls.py
urlpatterns = [ # 其他 url 配置 path('search/', views.search, name='search'), ]
大功告成,在導(dǎo)航欄嘗試輸入一些關(guān)鍵詞,看看效果吧!
當(dāng)然這樣的搜索功能是非常簡略的,難以滿足一些復(fù)雜的搜索需求。編寫一個搜索引擎是一個大工程,好在 django-haystack 這款第三方 app 為我們完成了全部工作。使用它我們可以實現(xiàn)更加復(fù)雜的搜索功能,比如全文檢索、按搜索相關(guān)度排序、關(guān)鍵字高亮等等類似于百度搜索的功能,功能十分強(qiáng)大。當(dāng)然其使用也會復(fù)雜一些,下一篇教程將向大家介紹 django-haystack 結(jié)合 Elasticsearch 搜索引擎的使用方法。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- 開啟Django博客的RSS功能的實現(xiàn)方法
- 基于Django統(tǒng)計博客文章閱讀量
- Python使用Django實現(xiàn)博客系統(tǒng)完整版
- 使用python和Django完成博客數(shù)據(jù)庫的遷移方法
- 利用Celery實現(xiàn)Django博客PV統(tǒng)計功能詳解
- Python采用Django開發(fā)自己的博客系統(tǒng)
- Python Django搭建網(wǎng)站流程圖解
- Django如何實現(xiàn)網(wǎng)站注冊用戶郵箱驗證功能
- python3.6+django2.0+mysql搭建網(wǎng)站過程詳解
- 用Django寫天氣預(yù)報查詢網(wǎng)站
- Python+Django搭建自己的blog網(wǎng)站
- 利用django創(chuàng)建一個簡易的博客網(wǎng)站的示例
相關(guān)文章
Python結(jié)合Selenium簡單實現(xiàn)Web自動化測試
這篇文章是入門級別的應(yīng)用Python + Selenium進(jìn)行自動化測試,包括環(huán)境搭建及簡單的實例,具有一定的參考價值,感興趣的小伙伴們可以參考一下2021-09-09Python matplotlib畫圖時圖例說明(legend)放到圖像外側(cè)詳解
這篇文章主要介紹了Python matplotlib畫圖時圖例說明(legend)放到圖像外側(cè)詳解,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-05-05TensorFlow中tf.batch_matmul()的用法
這篇文章主要介紹了TensorFlow中tf.batch_matmul()的用法,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-06-06