成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

使用sphinx search打造你自己的中文搜索引擎

trigkit4 / 1866人閱讀

摘要:本文以為例來介紹如何打造自己的搜索引擎。其官方網(wǎng)站是可以說支持包括英文中文等所有語言的搜索。默認(rèn)把中文按字拆分的,但這樣就會(huì)產(chǎn)生搜索出不相干的內(nèi)容來。因此,有人就給打了中文分詞的補(bǔ)丁。

Google搜索引擎建立至今已經(jīng)快20年了,之后全球各類大大小小類似的搜索引擎也陸續(xù)出現(xiàn)、消亡。國(guó)內(nèi)目前以百度為大,搜狗、360、必應(yīng)等也勢(shì)在必爭(zhēng)。搜索引擎技術(shù)也發(fā)展的相當(dāng)成熟,同時(shí)也就出現(xiàn)了很多開源的搜索引擎系統(tǒng)。比如,Solr、Lucene、Elasticsearch、Sphinx等。

本文以sphinx search為例來介紹如何打造自己的搜索引擎。該搜索引擎的架構(gòu)大致如下:


Sphinx search

Sphinx search 是俄羅斯人用C++寫的,速度很快,可以非常容易的與SQL數(shù)據(jù)庫(kù)和腳本語言集成,內(nèi)置MySQL和PostgreSQL 數(shù)據(jù)庫(kù)數(shù)據(jù)源的支持。其官方網(wǎng)站是: http://sphinxsearch.com/

可以說Sphinx支持包括英文、中文等所有語言的搜索。英文是以空格、標(biāo)點(diǎn)符號(hào)來分割單詞的,很容易切分。而中文詞匯之間是沒有空格的,很難區(qū)分,所以才有了自然語言處理中的“中文分詞”技術(shù)的研究。Sphinx默認(rèn)把中文按字拆分的,但這樣就會(huì)產(chǎn)生搜索出不相干的內(nèi)容來。比如,搜索“中國(guó)”,它會(huì)把同時(shí)包含“中”和“國(guó)”但不包含“中國(guó)”的文檔搜出來。因此,有人就給Sphinx打了中文分詞的補(bǔ)丁。

如果沒有搞錯(cuò)的話,最早添加中文分詞的是Coreseek,好像也是中文圈用得最廣的支持中文分詞的Sphinx,其它還有sphinx-for-chinese。然而這二者基于的Sphinx版本都太低了,有好多年沒有更新。其中存在的一些Sphinx的bug也沒有解決。

github上有一個(gè)基于Sphinx 2.2.9版本的代碼庫(kù)添加了中文分詞: https://github.com/eric1688/s... 經(jīng)測(cè)試,該版本穩(wěn)定性和速度都要好于coreseek。當(dāng)然它依然支持英文等其它語言的搜索,只是對(duì)中文搜索更加準(zhǔn)確了。

Sphinx 安裝

git clone https://github.com/eric1688/sphinx 
cd sphinx

#編譯(假設(shè)安裝到/usr/local/sphinx目錄,下文同) 
./configure --prefix=/usr/local/sphinx
# 說明: --prefix 指定安裝路徑 --with-mysql 編譯mysql支持 --with-pgsql 編譯pgsql支持
make 
sudo make install

安裝好后,在/usr/local/sphinx目錄下有以下幾個(gè)子目錄:
etc/ sphinx配置文件,不同的索引可以寫不同的配置文件
bin/ sphinx程序,其中有建立索引的程序:indexer, 搜索守護(hù)進(jìn)程:searchd
var/ 一般用了放置indexer索引好的文件

Sphinx索引的建立

MySQL數(shù)據(jù)庫(kù)表結(jié)構(gòu)
從上面的架構(gòu)圖可以看出來,我們要搜索的數(shù)據(jù)都存放在MySQL數(shù)據(jù)庫(kù)中。假設(shè)我們的數(shù)據(jù)庫(kù)名稱叫blog_data,其中有個(gè)表叫article,表結(jié)構(gòu)如下:

字段名 說明
id 文章唯一id(主鍵)
title 文章標(biāo)題
content 文章內(nèi)容
created_time 文章創(chuàng)建時(shí)間
該article表可以是你本身網(wǎng)站的文本內(nèi)容存放的表格,也可以是你的網(wǎng)絡(luò)爬蟲抓取到的數(shù)據(jù)存儲(chǔ)表。

還有建立另外一個(gè)表sph_counter用來存儲(chǔ)indexer已經(jīng)索引的最大doc id

字段名 說明
counter_id 標(biāo)記是對(duì)哪個(gè)表做記錄
max_doc_id 被索引表的最大ID
note 注釋,可以是表名
update_at 更新時(shí)間
建立索引配置文件:
新建或修改/usr/local/sphinx/etc/blog.conf 配置文件:

source blog_main
{ 
    type          = mysql
    sql_host      = localhost
    sql_user      = reader
    sql_pass      = readerpassword
    sql_db        = blog_data
    sql_port      = 3306
    sql_query_pre = SET NAMES utf8mb4
    sql_query_pre = REPLACE INTO sph_counter SELECT 1, MAX(id), "article", NOW() FROM article

    sql_query     = SELECT id, title, content,  
                    UNIX_TIMESTAMP(created_time) AS ctime, 
                    FROM article 
                    WHERE id <= (SELECT max_doc_id from sph_counter WHERE counter_id=1)
    sql_attr_timestamp = ctime  #從SQL讀取到的值必須為整數(shù),作為時(shí)間屬性

}

index blog_main
{
    source       = blog_main #對(duì)應(yīng)的source名稱
    path         = /user/local/sphinx/var/data/blog_main
    docinfo      = extern
    mlock        = 0
    morphology   = none
    min_word_len = 1 
    html_strip   = 0 

    charset_type     = utf-8
    chinese_dictionary = /user/local/sphinx/etc/xdict  #中文分詞的詞典
    ngram_len = 0 
    stopwords        = /user/local/sphinx/etc/stop_words.utf8
}

#全局index定義
indexer
{
    mem_limit = 512M
}

#searchd服務(wù)定義
searchd
{
    listen          = 9900
    listen          = 9306:mysql41  # 實(shí)時(shí)索引監(jiān)聽的端口
    read_timeout    = 5 
    max_children    = 90
    max_matches     = 100000
    max_packet_size = 32M 
    read_buffer     = 1M
    subtree_docs_cache = 8M
    subtree_hits_cache = 16M 
    #workers        = threads?
    dist_threads    = 2 
    seamless_rotate = 0 
    preopen_indexes = 0 
    unlink_old      = 1 
    pid_file  = /usr/local/sphinx/var/log/blog_searchd_mysql.pid
    log       = /usr/local/sphinx/var/log/blog_searchd_mysql.log
    query_log = /usr/local/sphinx/var/log/blog_query_mysql.log
}

編輯好以上配置文件,就可以開始建立索引了:

cd /usr/local/sphinx/bin
./indexer -c ../etc/blog.conf
# 如果已經(jīng)有searchd在運(yùn)行了,就要加 --roate 來進(jìn)行索引

索引建立后,就會(huì)在var/data/下面有名稱前綴為blog_main.XXX的索引文件生成。

建立實(shí)時(shí)索引
上面的配置文件是建立一個(gè)靜態(tài)索引,把當(dāng)時(shí)數(shù)據(jù)庫(kù)里面的所有數(shù)據(jù)進(jìn)行索引。但是,你的數(shù)據(jù)庫(kù)往往是不斷增加新數(shù)據(jù)的。為了及時(shí)索引并搜索到最新加入的數(shù)據(jù),就需要配置實(shí)時(shí)索引了。

index rt_weixin                                                                                                     {
    type = rt
    path = /usr/local/sphinx/var/data/rt_blog
    rt_field = title
    rt_field = content

    rt_attr_timestamp = pubtime
    ngram_chars = U+3000..U+2FA1F #為了支持中文
    ngram_len = 1
}

該倉(cāng)庫(kù)代碼的作者可能是忘了給實(shí)時(shí)索引加中文分詞,如果不配置ngram_chars 參數(shù)就不能搜到中文,添加后搜索是按單字匹配的,可見作者確實(shí)是忘了給實(shí)時(shí)索引部分加中文分詞。

添加以上實(shí)時(shí)索引后并不能搜索到實(shí)時(shí)數(shù)據(jù)。實(shí)時(shí)索引的更新/添加只能通過SphinxQL(一個(gè)類似MySQL的協(xié)議),所以還要寫一個(gè)Python腳本,從數(shù)據(jù)庫(kù)讀取最新的數(shù)據(jù)并通過SphinxQL更新到實(shí)時(shí)索引。

import MySQLdb
# 連接實(shí)時(shí)索引
db_rt = MySQLdb.connect(
    "127.0.0.1",
    "nodb",  # 對(duì)于實(shí)時(shí)索引來說,db,user,password都是不需要的,隨便寫。
    "noname",
    "nopass",
    port=9306,  # 實(shí)時(shí)索引監(jiān)聽的端口
)  

# 向?qū)崟r(shí)索引更新數(shù)據(jù)的函數(shù)
def into_rt(index_name, item):
    cursor = db_rt.cursor()
    fields = item.keys()
    values = item.values()
    fieldstr = ",".join(fields)
    valstr = ",".join([""%s""] * len(item))
    for i in xrange(len(values)):
        if isinstance(values[i], unicode):
            values[i] = values[i].encode("utf8")
        elif isinstance(values[i], datetime):
            try:
                values[i] = int(time.mktime(values[i].timetuple()))
            except:
                traceback.print_exc()
                print values[i]
                values[i] = int(time.time())
    sql = "INSERT INTO %s (%s) VALUES(%s)" % (index_name, fieldstr, valstr)
    # print sql
    sql = sql % tuple(values)
    try:
        cursor.execute(sql)
        db_rt.commit()
    except Exception, e:
        if e[0] == 1064:
            # ignore duplicated id error
            pass
        else:
            traceback.print_exc()
            raise "realtime index error"
    finally:
        cursor.close()

以上是及時(shí)建立實(shí)時(shí)索引的python程序的主要部分??梢园阉O(shè)置成后臺(tái)一直運(yùn)行的守護(hù)程序,也可以在crontab里面配置每隔幾分鐘運(yùn)行一次。

索引的更新
靜態(tài)的主索引如果只建立一次,實(shí)時(shí)索引的數(shù)據(jù)量會(huì)越積越多,對(duì)實(shí)時(shí)索引的搜索帶來很大壓力,所以我們要定時(shí)重新建立主索引,清理實(shí)時(shí)索引。
清理實(shí)時(shí)索引的程序可以參考上面建立實(shí)時(shí)索引的python程序。

crontab 設(shè)置每天凌晨1點(diǎn)運(yùn)行 indexer
crontab 設(shè)置indexer運(yùn)行完畢后清理實(shí)時(shí)索引,并從新的max_doc_id開始建立實(shí)時(shí)索引
以上就是建立一個(gè)自己的搜索引擎的過程。更多配置細(xì)節(jié)可到官方網(wǎng)站參考文檔。

文章來源于:猿人學(xué)網(wǎng)站的python教程。

版權(quán)申明:若沒有特殊說明,文章皆是猿人學(xué)原創(chuàng),沒有猿人學(xué)授權(quán),請(qǐng)勿以任何形式轉(zhuǎn)載。

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/43724.html

相關(guān)文章

  • 給全文搜索引擎Manticore (Sphinx) search 增加中文分詞

    摘要:最初使用的是,一個(gè)國(guó)人在基礎(chǔ)上添加了分詞的搜索引擎,可惜后來不再更新,的版本太低,也會(huì)出現(xiàn)后來也使用最新的,它可以支持幾乎所有語言,通過其內(nèi)置的對(duì)中文進(jìn)行索引和搜索。基于以上弊端,為中日韓文本加入分詞的是很有必要的。 Sphinx search 是一款非常棒的開源全文搜索引擎,它使用C++開發(fā),索引和搜索的速度非???,我使用sphinx的時(shí)間也有好多年了。最初使用的是coreseek,...

    WalkerXu 評(píng)論0 收藏0
  • Centos 編譯安裝sphinx-0.9.9全文檢索

    摘要:簡(jiǎn)介是開源的搜索引擎,它支持英文的全文檢索。但是往往我們要求的是中文索引,怎么做呢國(guó)人提供了一個(gè)可供企業(yè)使用的,基于的中文全文檢索引擎。 Sphinx 簡(jiǎn)介 Sphinx是開源的搜索引擎,它支持英文的全文檢索。所以如果單獨(dú)搭建Sphinx,你就已經(jīng)可以使用全文索引了。但是往往我們要求的是中文索引,怎么做呢?國(guó)人提供了一個(gè)可供企業(yè)使用的,基于Sphinx的中文全文檢索引擎。也就是說Cor...

    LeoHsiun 評(píng)論0 收藏0
  • Centos 編譯安裝sphinx-0.9.9全文檢索

    摘要:簡(jiǎn)介是開源的搜索引擎,它支持英文的全文檢索。但是往往我們要求的是中文索引,怎么做呢國(guó)人提供了一個(gè)可供企業(yè)使用的,基于的中文全文檢索引擎。 Sphinx 簡(jiǎn)介 Sphinx是開源的搜索引擎,它支持英文的全文檢索。所以如果單獨(dú)搭建Sphinx,你就已經(jīng)可以使用全文索引了。但是往往我們要求的是中文索引,怎么做呢?國(guó)人提供了一個(gè)可供企業(yè)使用的,基于Sphinx的中文全文檢索引擎。也就是說Cor...

    馬龍駒 評(píng)論0 收藏0
  • Centos 編譯安裝sphinx-0.9.9全文檢索

    摘要:簡(jiǎn)介是開源的搜索引擎,它支持英文的全文檢索。但是往往我們要求的是中文索引,怎么做呢國(guó)人提供了一個(gè)可供企業(yè)使用的,基于的中文全文檢索引擎。 Sphinx 簡(jiǎn)介 Sphinx是開源的搜索引擎,它支持英文的全文檢索。所以如果單獨(dú)搭建Sphinx,你就已經(jīng)可以使用全文索引了。但是往往我們要求的是中文索引,怎么做呢?國(guó)人提供了一個(gè)可供企業(yè)使用的,基于Sphinx的中文全文檢索引擎。也就是說Cor...

    fjcgreat 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

閱讀需要支付1元查看
<