PHP實現文本快速查找 - 二分查找


PHP實現文本快速查找 - 二分查找法

起因

先說說事情的起因,最近在分析數據時經常遇到一種場景,代碼需要頻繁的讀某一張數據庫的表,比如根據地區ID獲取地區名稱、根據網站分類ID獲取分類名稱、根據關鍵詞ID獲取關鍵詞等。雖然以上需求都可以在原始建表時,通過冗余數據來解決。但仍有部分業務存的只是關聯表的ID,數據分析時需要頻繁的查表。

所讀的表存在共同的特點

  • 數據幾乎不會變更
  • 數據量適中,從一萬到100多萬,如果全加載到內存也不太合適。

糾結的地方

在做數據分析時,需要十分頻繁的讀這些表,每秒有可能需要讀上萬次。其實內部的數據庫集群完全可以勝任,但會對線上業務稍有影響。(你懂得,小公司不可能為離線分析做一套完整的數據存儲服務。大部分數據分析還要借助線上的數據集群)

優化方案的思考

有沒有一種方式可以不增加線上的壓力,同時提供更高效的查詢方式?想過redis,但最終選擇用文本存儲。因為數據分析是一個獨立的需求,不希望與現有的redis集群或者其它存儲服務有交集。還有一個原因是每次分析的中間結果,對下一次分析並沒有很大的實質作用,並不需要把結果持久存儲,而且占的內存也會較多。最終使用文本存儲,然后用二分來查找。特點,1,存儲非常快,雖然redis等nosql服務雖然已經非常快,但仍無法與文本存儲相提並論;2,查找的時候使用二分查找,百萬條記錄查詢也可在0.1ms內完成(使用線上的普通硬盤,如果是ssd盤會更快)。

實現步驟

  • 將數據庫中需要的字段導出到文本

     方法:使用mysql的phpmyadmin工具,執行sql語句查出主建id和相應字段
    如以上的關鍵詞表: select kid, keyword from keyword
    然后使用phpmyadmin的導出工具,可以快速把結果導出到文本中
    操作截圖:
    

    image


image

  • 將導出的文本(已經按id進行過排序)轉換格式重新存儲
  • 程序讀取轉換后的格式

文本存儲格式

說明 :需求中,文本每行有兩列,第一列是主建ID(數字),第二列為文本。整個文本已經按第一列有序排列,兩列之間用tab鍵分隔。 之前有看過ip.dat的存儲,本次仿照其存儲格式:將文本中的內容每行轉換為固定長度后,存儲到新的文件。搜索時,使用文件操作函數fopen,fseek,fgets等函數按字節讀取內容,並以二分查找法快速定位需要的內容。

代碼實現部分

  • 通用類,類似需求只需要提供符合標准的文本(每行兩列,第一列為查找的ID,第二列為文本。同時文本已經按第一列有序排序)
  • 生成以上所提到的存儲格式
  • 提供根據id查詢接口

代碼片斷

  • 重新生成新的存儲格式

     //讀源文件,寫入到新的索引文件
    $readfd = fopen($this->filename, 'rb');
    $writefd = fopen($this->formatFile.'_tmp', 'wb+');
    if ($readfd === false || $writefd === false) {
    return false;
    }
    echo "\n start reformat file $this->filename ..";
    while (!feof($readfd)) {
    $line = fgets($readfd, 8192);
    fwrite($writefd, pack("a".$this->maxLength, $line));
    }
    echo "\n reformat ok\n";
    fclose($readfd);
    fclose($writefd);
    rename($this->formatFile.'_tmp', $this->formatFile);
    
  • 二分查找的代碼片斷

     /**
    * 在索引文件中進行二分查找
    * @param int $id 進行二分查找的id
    * @return [type] [description]
    */
    public function search($key)
    {
    $filesize = filesize($this->formatFile);
    $fd = fopen($this->formatFile, "rb");
    $left = 0; //行號
    $right = ($filesize / $this->maxLength) - 1; 
    while ($left <= $right) {
    $middle = intval(($right + $left)/2);
    fseek($fd, ($middle) * $this->maxLength);
    $info = unpack("a*", fread($fd, $this->maxLength))['1'];
    $lineinfo = explode("\t", $info, 2);
    if ($lineinfo['0'] > $key) {
    $right = $middle - 1;
    } elseif ($lineinfo['0'] < $key) {
    $left = $middle + 1;
    } else {
    return $lineinfo['1'];
    }
    }
    return false;
    }
    
  • 整個類庫代碼一共91行,具體可查看github的demo代碼 ,相關鏈接

運行截圖

image

以上拿100萬的關鍵詞進行測試,根據關鍵詞id快速查找關鍵詞,平均速度可以達到0.1毫秒。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM