3

注意:我最初在一小时前问了这个问题,但直到最近才意识到我犯了一个重大的复制和粘贴错误。一个如此重要,以至于删除旧帖子并重新开始更容易。对于那个很抱歉。

在 CakePHP 框架中,更新模型后,我转储了 SQL 查询。COUNT(*)被调用了两次,没有明显的原因。

所以我有两个模型,$Foo$Bar. 为简单起见,我没有定义它们之间的任何$belongsTo关系$hasMany。问题只涉及 $Foo,但以防万一我也包含了 $Bar 的代码。

$data = array(
    'something' => 12,
    'something_else' => $this->Bar->field('id', $conditions),
);

$this->Foo->id = $this->Foo->field($data);
$this->Foo->save($data);

$Bar.id根据一组条件(这里不相关)获得该字段。我使用该 id 来确定$data应该 INSERT 或 UPDATE 到$Foo. 如果$Foo有一个满足$data的要求的 id 将返回该值;如果不是false将被返回。根据 CakePHP 的架构,save()如果有有效的 id 就会更新;否则它将插入。我很确定我没有做任何不寻常的事情。

这是我在 SQL 转储中看到的内容:

SELECT `Foo`.`id` FROM `foos` AS `Foo` WHERE `something` = 12 AND `something_else` = 1 LIMIT 1
SELECT COUNT(*) AS `count` FROM `foos` AS `Foo` WHERE `Foo`.`id` = 1
SELECT COUNT(*) AS `count` FROM `foos` AS `Foo` WHERE `Foo`.`id` = 1
UPDATE `foos` SET `something` = 12, something_else` = 1 WHERE `foos`.`id` = 1'

对于我的生活,我根本无法弄清楚为什么COUNT()需要它,更不用说为什么它被调用两次了。有人知道发生了什么吗?谢谢您的帮助。

4

2 回答 2

3

我没有使用过 CakePHP,但是查看这里的源代码,您会在字段函数中看到它调用了 find 方法。

因此,这个调用

$data = array(
    'something' => 12,
    'something_else' => $this->Bar->field('id', $conditions),
);

生成:

SELECT `Foo`.`id` FROM `foos` AS `Foo` WHERE `something` = 12 AND `something_else` = 1     LIMIT 1

然后设置字段调用

$this->Foo->id = $this->Foo->field($data);

通过内部 find 函数调用生成它:

SELECT COUNT(*) AS `count` FROM `foos` AS `Foo` WHERE `Foo`.`id` = 1

Save 函数还会调用 find 来查看保存前记录是否存在,从而生成第二个调用:

SELECT COUNT(*) AS `count` FROM `foos` AS `Foo` WHERE `Foo`.`id` = 1

最后的查询是明显的保存本身。

请自行确认我是否正确阅读了它,但这似乎就是这样做的。

于 2012-06-21T02:32:41.033 回答
0

无法告诉您发生了什么(因为我认为没有足够的代码),但可以为您提供解决方法。

使用“INSERT INTO table(id, field, field2) VALUES(id, value, value2) ON DUPLICATE KEY UPDATE field=value, field2=value2" 语法进行查询。

这使 SQL 可以检查重复项,并且还将消除您检查行是否存在和尝试插入之间的那几秒钟(以防万一另一个进程在您检查后插入)。为此,字段“id”必须是主键或“唯一”索引,以便它知道触发更新。

于 2012-06-21T00:53:21.200 回答