如何在ZF2应用程序中实现自动完成的服务器端缓存?

时间:2016-10-30 19:32:42

标签: php caching autocomplete zend-framework2 apcu

首先实际状态:有一个带表单的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应用程序中实现自动完成功能的缓存机制?

1 个答案:

答案 0 :(得分:0)

这里与ZF2无关。这完全是关于自定义搜索服务设计及其面临的挑战。

如果没有适当的缓存层和/或全文搜索引擎,构建自动完成实现,这对应用程序来说将是自杀。您可以在很短的时间内轻松完成数万次不必要的重复查询。

在"理想"世界上,良好的自动完成实施利用了全文搜索引擎,如ElasticsearchApache Solr。并分别使用他们的Completion SuggesterSuggester组件。

无论如何,只使用对象缓存和数据库仍然可以实现简单的自动完成功能。只有你需要的是一个帮助方法来创建一个正确的缓存键"为每个字母组合。例如:

   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是此实施的不错选择。

希望它有所帮助。