如何在CakePHP查找查询中正确嵌套自定义字段?

时间:2014-01-28 15:46:13

标签: php mysql cakephp cakephp-2.4 cakephp-model

我们经常处理的代码如下所示:

return $this->find(
'all', [
  'fields' => ['DISTINCT Tag.tag', 'COUNT(Tag.tag) as count'],
  'group' => ['Tag.tag'],
  'order' => ['count' => 'DESC']
]);

此查询引导我们输出以下内容:

[0] => Array
    (
        [Tag] => Array
            (
                [tag] => walls.io
            )

        [0] => Array
            (
                [count] => 15
            )

    )

如您所见,查询以“某种程度上错误”的嵌套方式返回结果。遗憾的是,“count”字段被放入伪[0] - 阵列中。

IIRC,CakePHP在内部使用像Tag__field这样的语法来正确嵌套虚拟字段。

将代码更改为Model __-语法时,问题保持不变:

return $this->find(
'all', [
  'fields' => ['DISTINCT Tag.tag', 'COUNT(Tag.tag) as Tag__count'],
  'group' => ['Tag.tag'],
  'order' => ['COUNT(Tag.tag)' => 'DESC']
]);

输出:

[0] => Array
    (
        [Tag] => Array
            (
                [tag] => walls.io
            )

        [0] => Array
            (
                [Tag__count] => 15
            )

    )

解决方法1:array_map

CakePHP专业人士:有没有比在select语句后手动映射数组更好/更优雅的解决方案?

$tags = array_map(function($tag) {
  $tag['Tag']['count'] = $tag[0]['count'];
  unset($tag[0]);

  return $tag;
}, $tags);

解决方法2:虚拟字段

如上所述,使用虚拟字段可以解决此问题:

$this->virtualFields = ['count' => 'COUNT(Tag.Tag)'];
return $this->find(
'all', [
  'group' => ['Tag.tag'],
  'order' => [$this->getVirtualField('count') => 'DESC']
]);

不幸的是,使用此解决方案,根本无法指定任何字段。只有完全离开“fields”-key,数组的嵌套才能按预期工作。当选择$ fields = ['Tag.tag',$ this-> getVirtualField('count')]时,嵌套再次出错。

CakePHP专业人员:您是否知道一种方法,即使您指定了自己的字段,嵌套也是正确的?

1 个答案:

答案 0 :(得分:2)

查看CakePHP代码,这种方法不存在。

查看文件lib/Cake/Model/Datasource/Database/Mysql.php

找到名为Mysql::resultSet( $results )的方法; (约line 240)。

该方法将结果映射到数组。要确定列是否是表的一部分,它使用PDOStatement::getColumnMeta()。对于您的“虚拟列”,该方法将返回一个空表,因此CakePHP代码将单独放置它,请参阅else分支

$this->map[$index++] = array(0, $column['name'], $type);

为了避免使用其他分支,您必须使用虚拟字段,但之后会遇到您注意到的其他问题。

所以你留下了array_map解决方案,或者你可以尝试重载那个Mysql类,并添加你的自定义逻辑来确定列适合的位置。