预准备语句中的递归

时间:2010-04-26 14:31:14

标签: php mysql recursion pdo prepared-statement

我一直在使用PDO并准备我的所有陈述主要是出于安全原因。但是,我的代码的一部分 使用不同的参数多次执行相同的语句,我认为这将是准备好的语句真正发挥作用的地方。但他们实际上打破了代码...

代码的基本逻辑就是这个。

function someFunction($something) {
  global $pdo;

  $array = array();

  static $handle = null;
  if (!$handle) {
    $handle = $pdo->prepare("A STATEMENT WITH :a_param");
  }

  $handle->bindValue(":a_param", $something);
  if ($handle->execute()) {
    while ($row = $handle->fetch()) {
      $array[] = someFunction($row['blah']);
    }
  }

  return $array;
}

对我来说看起来很好,但它错过了很多行。最后我意识到语句句柄正在被改变(用不同的param执行),这意味着在while循环中对fetch的调用只会工作一次,然后函数再次调用自己,结果集也会改变。

所以我想知道以递归方式使用PDO预处理语句的最佳方法是什么。

一种方法可能是使用fetchAll(),但它在手册中说有很大的开销。这一点的重点是提高效率。

我能做的另一件事是不重用静态句柄,而是每次都重新创建一个新句柄。我相信,由于查询字符串是相同的,因此MySQL驱动程序内部无论如何都将使用预处理语句,因此在每次递归调用上创建新句柄的开销很小。就我个人而言,我认为这是有道理的。

或者有什么方法可以重写这个吗?

3 个答案:

答案 0 :(得分:3)

您无法嵌套语句句柄:您需要关闭先前打开的句柄,然后才能在单个会话中打开另一个句柄。

事实上,PDO会在您发出新的准备时自动执行。

递归调用函数时:

  • 初始句柄已分配(1)
  • 第一条记录是(1)
  • 该函数以递归方式调用。 (1)的值驻留在递归堆栈中。
  • 新句柄已分配(2)无效(1)
  • 第一条记录是(2)
  • 该函数返回
  • 您尝试获取(1)的下一条记录,但由于它无效
  • 而失败

如上所述,MySQL不支持递归,这意味着您必须使用PHPfetchAll方面进行递归。

答案 1 :(得分:0)

真正的问题是$handle是静态的。当需要在递归调用中保留状态时,静态变量对于递归是有问题的,而不仅仅是对于预准备语句。在这种情况下,递归调用执行新查询,丢弃先前的状态。如果您想要一个准备好的查询,PDO::fetchAll确实是唯一的选择。

根据语句的不同,您可能会重写它以立即返回所有结果,然后构建树。

答案 2 :(得分:0)

如果使用相同的变量(因为pdo bindValue),每次值与第一个值相同。所以这将失败:

foreach ($bind_params as $key => $value) {
    $stmt->bindParam(":$key", $value);
}

结果:

$key[0] = $value[0];
$key[1] = $value[0];
$key[2] = $value[0];
$key[3] = $value[0];

所以你想做丑陋的伎俩,然后:

        $i = 0;
        foreach ($bind_params as $key => $value) {
            $i++;
            $$i = $value;
            $stmt->bindParam(":$key", $$i);
        }

结果:

$key[0] = $value[0];
$key[1] = $value[1];
$key[2] = $value[2];
$key[3] = $value[3];