PHP Conference Japan 2024

SQLite3::createAggregate

(PHP 5 >= 5.3.0, PHP 7, PHP 8)

SQLite3::createAggregate注册一个 PHP 函数作为 SQL 聚合函数使用

描述

public SQLite3::createAggregate(
    字符串 $name,
    可调用 $stepCallback,
    可调用 $finalCallback,
    整数 $argCount = -1
): 布尔值

注册一个 PHP 函数或用户定义函数,作为 SQL 聚合函数在 SQL 语句中使用。

参数

name

要创建或重新定义的 SQL 聚合的名称。

stepCallback

对于结果集的每一行调用的回调函数。您的 PHP 函数应累积结果并将其存储在聚合上下文中。

此函数需要定义为

step(
    混合 $context,
    整数 $rownumber,
    混合 $value,
    混合 ...$values
): 混合
context

对于第一行,null;在后续行中,它将具有先前从 step 函数返回的值;您应该使用它来维护聚合状态。

rownumber

当前行号。

value

传递给聚合的第一个参数。

values

传递给聚合的其他参数。

此函数的返回值将用作 step 或 finalize 函数下次调用的 context 参数。

finalCallback

回调函数,用于聚合每一行的“分步”数据。一旦所有行都处理完毕,就会调用此函数,然后它应该从聚合上下文中获取数据并返回结果。此回调函数应返回 SQLite 能够理解的类型(即 标量类型)。

此函数需要定义为

fini(混合 $context, 整数 $rownumber): 混合
context

保存 step 函数最后一次调用的返回值。

rownumber

始终为 0

此函数的返回值将用作聚合的返回值。

argCount

SQL 聚合接受的参数数量。如果此参数为负,则 SQL 聚合可以接受任意数量的参数。

返回值

如果成功创建聚合,则返回 true,否则返回 false

示例

示例 #1 max_length 聚合函数示例

<?php
$data
= array(
'one',
'two',
'three',
'four',
'five',
'six',
'seven',
'eight',
'nine',
'ten',
);
$db = new SQLite3(':memory:');
$db->exec("CREATE TABLE strings(a)");
$insert = $db->prepare('INSERT INTO strings VALUES (?)');
foreach (
$data as $str) {
$insert->bindValue(1, $str);
$insert->execute();
}
$insert = null;

function
max_len_step($context, $rownumber, $string)
{
if (
strlen($string) > $context) {
$context = strlen($string);
}
return
$context;
}

function
max_len_finalize($context, $rownumber)
{
return
$context === null ? 0 : $context;
}

$db->createAggregate('max_len', 'max_len_step', 'max_len_finalize');

var_dump($db->querySingle('SELECT max_len(a) from strings'));
?>

以上示例将输出

int(5)

在此示例中,我们正在创建一个聚合函数,该函数将计算表中某一列中最长字符串的长度。对于每一行,都会调用 max_len_step 函数并传递 $context 参数。context 参数就像任何其他 PHP 变量一样,可以设置为保存数组甚至对象值。在此示例中,我们只是简单地使用它来保存到目前为止我们看到的最大长度;如果 $string 的长度大于当前最大值,我们将更新上下文以保存此新的最大长度。

处理完所有行后,SQLite 调用 max_len_finalize 函数来确定聚合结果。在这里,我们可以根据 $context 中找到的数据执行某种计算。但在我们的简单示例中,我们已经随着查询的进行计算了结果,因此我们只需要返回上下文值。

提示

不建议您在上下文中存储值的副本,然后在最后处理它们,因为这会导致 SQLite 使用大量内存来处理查询——只需考虑一下如果一百万行存储在内存中,每行包含一个 32 字节的字符串,您需要多少内存。

提示

您可以使用 SQLite3::createAggregate() 覆盖 SQLite 本地 SQL 函数。

添加注释

用户贡献的注释 2 条注释

6
boris dot dd at gmail dot com
7 年前
<?php
class Test extends SQLite3
{
public function
__construct($file)
{
parent::__construct($file);
$this->createAggregate('groupConcat', [$this, 'concatStep'], [$this, 'concatFinal']);
}
public function
concatStep(&$context, $rowId, $string, $delimiter)
{
if (!isset(
$context)) {
$context = [
'delimiter' => $delimiter,
'data' => []
];
}
$context['data'][] = $string;
return
$context;
}
public function
concatFinal(&$context)
{
return
implode($context['delimiter'], $context['data']);
}
}
$SQLite = new Test('/tmp/test.sqlite');
$SQLite->exec("create table `test` (`id` TEXT, `color` TEXT, `size` TEXT)");
$SQLite->exec("insert into `test` (`id`, `color`, `size`) values ('1', 'red', 'M')");
$SQLite->exec("insert into `test` (`id`, `color`, `size`) values ('1', 'green', 'M')");
$SQLite->exec("insert into `test` (`id`, `color`, `size`) values ('1', 'blue', 'S')");
$Result = $SQLite->query("select `size`, groupConcat(`color`, ';') as `color` from `test` group by `size`");
while (
$row = $Result->fetchArray(SQLITE3_ASSOC)) {
print_r($row);
}
/*
Array
(
[size] => M
[color] => red;green
)
Array
(
[size] => S
[color] => blue
)
*/
-3
[email protected]
9年前
缺少示例,对吧?
让我们尝试为 SQLite3 提供类似 MySQL 的功能
- REGEXP 运算符,
- MD5 函数,以及
- GROUP_CONCAT 聚合函数

$db = new SQLite3($filename);
$db->createFunction('regexp', function ($a,$b) { return preg_match("/$a/i", $b); });
$db->createFunction('md5', function ($a) { return md5($a); });
$db->createAggregate ('group_concat',
function(&$context, $rownumber, $str) { $context[]=$str; return $context; },
function(&$context) {return implode(",", (array) $context); });
To Top