PHP+redis实现超迷你全文检索

2014年10月31日 11:45:39

情景: 我们平台有好多游戏, 运营的同事在查询某一款游戏的时候, 目前使用的是html的select下拉列表的展现形式, 运营的同事得一个个去找,然后选中,耗时又费眼

效果: 输入"三国"或者"国三", 将自动列出所有包含"三国"的游戏名字, 输入不限顺序; 例如输入"杀三国",仍然会将"三国杀"这款游戏找出来

实现: 我用redis的集合+PHP的array_intersect()和mb系列函数, 实现了一个超迷你的全文检索功能

原理: (大道不过两三言,说穿不值一文钱,哈哈)

1, 将所有的游戏名字读出来,拆分成单个汉字

2, 将这些汉字作为redis集合的键,写入redis,每个集合里的值是所有那些游戏名字中包含此汉字的游戏的id

3, 当用户输入文字的时候通过ajax异步请求,将用户输入传给PHP

4, 将输入的文字拆分成单个汉字, 分别找到这些汉字在redis中的集合值

5, 取出来,求交集,就找到了同时包含这几个汉字的游戏的id

6, 最后到数据库里查出来相应的游戏信息即可

缺点: 删除数据不方便

 PHP写入redis和检索的代码:

 1     //自动补全
 2     //不限输入汉字的前后顺序: 输入"国三杀" => 输出 "三国杀"
 3     function getAutoComplate()
 4     {
 5         //$word = $this->input->post('word');
 6         $word = '三国';
 7         if (empty($word)) {
 8             exit('0');
 9         }
10         $intWordLength = mb_strlen($word, 'UTF-8');
11 
12         $this->load->library('iredis');
13         if (1 == $intWordLength) {
14             $arrGid = $this->iredis->getAutoComplate($word);
15         } else {
16             $arrGid = array();
17             for ($i=0; $i < $intWordLength; $i++) {
18                 $strOne = mb_substr($word, $i, 1, 'UTF-8');
19                 $arrGidTmp = $this->iredis->getAutoComplate($strOne);
20                 $arrGid = empty($arrGid) ? $arrGidTmp : array_intersect($arrGid, $arrGidTmp); //求交集,因为传入的参数个数不确定,因此不能直接求交集
21             }
22         }
23 
24         $arrGame = $this->gamemodel->getGameNameForAutoComplate($arrGid);
25         // var_dump($arrGame);exit;
26         $jsonGame = json_encode($arrGame);
27         exit($jsonGame);
28     }
29 
30     //自动补全, 建立索引
31     function setAutoComplate()
32     {
33         $arrGame = $this->gamemodel->getAllGameNameForAutoComplate();
34         $arrIndex = array();
35         foreach ($arrGame as $gid => $gname) {
36             $intGnameLength = mb_strlen($gname, 'UTF-8');
37             for ($i=0; $i < $intGnameLength; $i++) {
38                 $strOne = mb_substr($gname, $i, 1, 'UTF-8');
39                 $arrIndex[$strOne][] = $gid;
40             }
41         }
42         
43         $this->load->library('iredis');
44         foreach ($arrIndex as $word => $arrGid) {
45             foreach ($arrGid as $gid) {
46                 $this->iredis->setAutoComplate($word, $gid);
47             }
48         }
49         
50     }

 操作redis的方法

 1     //自动补全功能
 2     public function setAutoComplate($key, $value)
 3     {
 4         $youxikey = 'youxi_'.$key;
 5         $this->sAdd($youxikey, $value);
 6     }
 7 
 8     //自动补全功能
 9     public function getAutoComplate($key)
10     {
11         $youxikey = 'youxi_'.$key;
12         return $this->sMembers($youxikey);
13     }

 

 

Finger PHP 框架

posted @ 2014-10-31 12:00  myD  阅读(5547)  评论(0编辑  收藏  举报