情景:
我们平台有好多游戏,运营的同事在查询某一款游戏的时候,目前使用的是html的select下拉列表的展现形式,运营的同事得一个个去找,然后选中,耗时又费眼效果:
输入"三国"或者"国三",将自动列出所有包含"三国"的游戏名字,输入不限顺序; 例如输入"杀三国",仍然会将"三国杀"这款游戏找出来实现:
我用redis的集合+PHP的array_intersect()和mb系列函数,实现了一个超迷你的全文检索功能原理:
(大道不过两三言,说穿不值一文钱,哈哈)1、将所有的游戏名字读出来,拆分成单个汉字
2、 将这些汉字作为redis集合的键,写入redis,每个集合里的值是所有那些游戏名字中包含此汉字的游戏的id
3、当用户输入文字的时候通过ajax异步请求,将用户输入传给PHP
4、将输入的文字拆分成单个汉字,分别找到这些汉字在redis中的集合值
5、取出来,求交集,就找到了同时包含这几个汉字的游戏的id
6、最后到数据库里查出来相应的游戏信息即可
缺点:
删除数据不方便 输出 "三国杀"
function getAutoComplate()
{
//$word = $this->input->post('word');
$word = '三国';
if (empty($word)) {
exit('0');
}
$intWordLength = mb_strlen($word,'UTF-8');
$this->load->library('iredis');
if (1 == $intWordLength) {
$arrGid = $this->iredis->getAutoComplate($word);
} else {
$arrGid = array();
for ($i=0; $i < $intWordLength; $i++) {
$strOne = mb_substr($word,$i,1,'UTF-8');
$arrGidTmp = $this->iredis->getAutoComplate($strOne);
$arrGid = empty($arrGid) ? $arrGidTmp : array_intersect($arrGid,$arrGidTmp); //求交集,因为传入的参数个数不确定,因此不能直接求交集
}
}
$arrGame = $this->gamemodel->getGameNameForAutoComplate($arrGid);
// var_dump($arrGame);exit;
$jsonGame = json_encode($arrGame);
exit($jsonGame);
}
//自动补全,建立索引
function setAutoComplate()
{
$arrGame = $this->gamemodel->getAllGameNameForAutoComplate();
$arrIndex = array();
foreach ($arrGame as $gid => $gname) {
$intGnameLength = mb_strlen($gname,'UTF-8');
for ($i=0; $i < $intGnameLength; $i++) {
$strOne = mb_substr($gname,'UTF-8');
$arrIndex[$strOne][] = $gid;
}
}
$this->load->library('iredis');
foreach ($arrIndex as $word => $arrGid) {
foreach ($arrGid as $gid) {
$this->iredis->setAutoComplate($word,$gid);
}
}
}
操作redis的方法