首先实际状态:有一个带表单的ZF2应用程序。该表单包含一些自动填充字段(在前端使用jQuery Autocomplete实现)。
它背后的SQL语句如下所示:
SELECT name FROM countries WHERE countries.name LIKE %en%
-> should find "Arg[en]tina", "Arm[en]ia", "B[en]in", "Turkm[en]istan" etc.
or
SELECT name FROM countries WHERE countries.continent_id = 2
-> should find "Afghanistan", "Armenia", "Azerbaijan" etc. (2 = Asia)
or
SELECT name FROM countries WHERE countries.continent_id = 2 AND countries.name LIKE %en%
-> should find "Arm[en]ia", "Turkm[en]istan" etc. (2 = Asia)
当然,它会导致问题,数据库会受到许多小型自动填充请求的恐吓。缓存应该有所帮助 - 我已经开始实现基于Zend\Cache\Storage\Adapter\Apcu
的缓存机制。但后来我看到了下一个麻烦:使用像APCu这样的公共缓存我无法动态过滤结果。所以这样的缓存似乎不适用于自动完成的情况。
我很确定,这是一个常见的问题,并且已经有了解决方案。
如何在ZF2应用程序中实现自动完成功能的缓存机制?
答案 0 :(得分:0)
这里与ZF2无关。这完全是关于自定义搜索服务设计及其面临的挑战。
如果没有适当的缓存层和/或全文搜索引擎,构建自动完成实现,这对应用程序来说将是自杀。您可以在很短的时间内轻松完成数万次不必要的重复查询。
在"理想"世界上,良好的自动完成实施利用了全文搜索引擎,如Elasticsearch或Apache Solr。并分别使用他们的Completion Suggester和Suggester组件。
无论如何,只使用对象缓存和数据库仍然可以实现简单的自动完成功能。只有你需要的是一个帮助方法来创建一个正确的缓存键"为每个字母组合。例如:
function createKeyByQuery($str)
{
return 'autocomplete-prefix-'.(string) $str;
}
并在您的suggest()
方法中:
public function suggest($keyword)
{
$key = $this->createKeyByQuery($keyword);
if($this->cache->hasItem($key)) {
return $this->cache->getItem($key);
}
// fetch form the database here
$data = $this->db->query();
$this->cache->setItem($key, $data);
return $data;
}
如果过滤器的数量不是太多,请将它们作为键的一部分。在这种情况下,建议方法的签名将是:
public function suggest($keyword, array $filters = []);
和密钥生成器需要更新:
function createKeyByQuery($str, array $filters = [])
{
return 'autocomplete-prefix-' . (string) $str . md5(serialize($filters));
}
此解决方案可能不适用于复杂/域相关数据,因为它具有相当大的失效挑战。例如。如何找到保存"阿根廷"在有效载荷?
由于您只处理国家/地区列表作为过滤器,因此应解决此问题。
对于england
关键字和两个不同的过滤器,共有10个过滤器选项,将有10x2x7 = 140个不同的缓存键。对于具有5个选项的单个过滤器,5x1x7 = 37个不同的键。
APCu是此实施的不错选择。
希望它有所帮助。