Datatables+PHP:多个表上的服务器端处理
如何让Datatables服务器端处理脚本处理自定义查询?我需要从多个表中选择列,并让Datatables呈现它们 Datatables.net使用PHP在服务器端处理SSP的概述如下: 我发现了这一点,但原始海报从未提供他的解决方案。我没有足够的声誉要求他提供更多的细节 这是我未使用Datatable的SSP的原始SQLDatatables+PHP:多个表上的服务器端处理,php,mysql,datatables,server-side,Php,Mysql,Datatables,Server Side,如何让Datatables服务器端处理脚本处理自定义查询?我需要从多个表中选择列,并让Datatables呈现它们 Datatables.net使用PHP在服务器端处理SSP的概述如下: 我发现了这一点,但原始海报从未提供他的解决方案。我没有足够的声誉要求他提供更多的细节 这是我未使用Datatable的SSP的原始SQL SELECT tbl_houses.style, tbl_houses.roomCount, tbl_residents.firstName, tbl_residents.l
SELECT tbl_houses.style, tbl_houses.roomCount, tbl_residents.firstName, tbl_residents.lastName
FROM tbl_houses, tbl_residents
WHERE tbl_houses.houseID = tbl_residents.residentID
/*
* # Equivalent query using JOIN suggested by @KumarRakesh
* # Note: JOIN ... ON is a synonym for INNER JOIN ... ON
* # Using JOIN conforms to syntax spec'd by ANSI-92 https://stackoverflow.com/a/894855/946957
*
* SELECT tbl_houses.style, tbl_houses.roomCount, tbl_residents.firstName, tbl_residents.lastName
* FROM tbl_houses
* JOIN tbl_residents ON tbl_houses.houseID = tbl_residents.residentID
*/
如何使用SSP让Datatables运行上述查询
它似乎只接受1个表,不接受自定义筛选,即WHERE子句
但是,它支持使用WHERE进行过滤。我想我需要修改ssp.class.php以强制我的WHERE子句
更新
找到了解决办法。我有空时会发布。看起来DataTables中的脚本确实不是为您的特定用例设计的。但是有一种方法允许自定义where子句,并且通过读取的源代码,我认为通过使用where方法,这种配置应该适合您。JOIN方法在这里不起作用 长话短说:将服务器_processing.php编辑为:
<?php
// DB table to use
$table = 'tbl_houses, tbl_residents';
// First table's primary key
$primaryKey = 'tbl_houses.id';
$columns = [
[ 'db' => 'tbl_houses.style'],
[ 'db' => 'bl_houses.roomCount'],
[ 'db' => 'tbl_residents.firstName'],
[ 'db' => 'tbl_residents.lastName']
);
// connection details
$sql_details = [
];
$whereAll = 'tbl_houses.houseID = tbl_residents.residentID';
require( 'ssp.class.php' );
echo json_encode(
SSP::complex( $_GET, $sql_details, $table, $primaryKey, $columns , null, $whereAll);
);
复杂方法接受您的自定义WHERE子句。但棘手的是使用两个表。这就是脚本的设计目的。我看了一下它是如何构建最终sql查询的,似乎您可以在配置中使用这个table_name.field_name符号,以及$table和$primaryKey变量的table_name、table_name符号
如前所述,DataTables脚本并不打算使用2个表。我不知道DataTables的所有功能是否都能与之配合使用;DR:我最终使用了由Emran Ul Hadi实现的对原始Datatables ssp.class.php的修改:
他的修改接受JOIN、WHERE、GROUP BY和列别名。虽然该文件已经一年多没有更新了,但它仍然适用于DataTables 1.12.x。我对他的版本做了一些修改,增加了它的健壮性,并用更清晰的示例改进了文档
当我有更多的时间时,我会在这里发布我的mods/更新。最终,我希望提交一个请求,将我的更新信息放入他的存储库。解决方案
类ssp.Class.php不支持联接和子查询,但有一个解决方法。诀窍是使用子查询,如下面的$table定义中所示。用子查询中的实际表名替换表
$table = <<<EOT
(
SELECT
a.id,
a.name,
a.father_id,
b.name AS father_name
FROM table a
LEFT JOIN table b ON a.father_id = b.id
) temp
EOT;
$primaryKey = 'id';
$columns = array(
array( 'db' => 'id', 'dt' => 0 ),
array( 'db' => 'name', 'dt' => 1 ),
array( 'db' => 'father_id', 'dt' => 2 ),
array( 'db' => 'father_name', 'dt' => 3 )
);
$sql_details = array(
'user' => '',
'pass' => '',
'db' => '',
'host' => ''
);
require( 'ssp.class.php' );
echo json_encode(
SSP::simple( $_GET, $sql_details, $table, $primaryKey, $columns )
);
您还需要编辑ssp.class.php并将FROM`$table`的所有实例替换为FROM$table,以删除反勾号
确保所有列名都是唯一的,否则将用作指定别名
笔记
还有一个存储库包含支持联接的增强的ssp.class.php
链接
有关更多信息,请参阅。原始数据表ssp.class.php的My fork: 您可以别名表和列,使用where,使用[LEFT,internal,RIGHT,CROSS]与表连接。 示例服务器_processing.php:
<?php
//For one table:
//$table = "main_table_name";
//Or for multitable queries:
$table = array(
array('table'=>'main_table_name', 'as'=>'mt'),
array('table'=>'join1_table_name', 'as'=>'jt1', 'join_type'=>'INNER', 'join_on'=>'mt.field = jt1.field'),
array('table'=>'join2_table_name', 'as'=>'jt2', 'join_type'=>'LEFT', 'join_on'=>'mt.field1 = jt2.field')
);
//Columns definition with alias
$columns = array(
array( 'db' => 'mt.field1', 'dt' => 0 ),
array( 'db' => 'mt.field2', 'dt' => 1 ),
array( 'db' => 'mt.name', 'as'=>'field3', 'dt' => 2 ),
array( 'db' => 'jt1.field1', 'as'=>'field4', 'dt' => 3 ),
array( 'db' => 'jt2.field5', 'as'=>'field5', 'dt' => 4 )
);
$primaryKey = 'mt.id'; //Primary key, for check records count
$sql_details = array(
'user' => $db_user,
'pass' => $db_password,
'db' => $db_name,
'host' => $db_host
);
$where = "mt.field1 > 10"; //You SQL where condition
require('ssp.class.php' ); //File from github.com/emayskiy/Datatables-SSP-MultiTables
echo json_encode(
// All params same as in original class SSP
SSP::complex( $_GET, $sql_details, $table, $primaryKey, $columns , '', $where)
);
?>
很抱歉迟到了,但是我通过构建自己的服务器端文件解决了这个问题,这样我就可以自定义查询和json输出 文件服务器端自定义:
public function produtosEstoque($arr)
{
//COLUNAS
$column = array('prodNome', 'prodPreco', 'prodQtdEst', 'categDescricao');
//SQL
$query2 = "SELECT a.prodID, a.prodNome, a.prodMedida, a.prodPrecoAnt, a.prodPreco, a.prodCategoriaID, a.prodOferta,a.prodQtdEst, a.prodQtdMEst, b.categDescricao FROM produtos a INNER JOIN categoria b ON a.prodCategoriaID = b.categID WHERE a.prodEstatusID = 6 ";
//SEARCH
if ($arr['search']['value']) {
$query2 .= "AND prodNome LIKE '%".$arr['search']['value']."%' ";
}
//ORDER
if (isset($arr['order'])) {
$query2 .= 'ORDER BY ' . $column[$arr['order']['0']['column']] . ' ' . $arr['order']['0']['dir'] . ' ';
} else {
$query2 .= 'ORDER BY a.prodID DESC ';
}
//LIMIT
if ($arr["length"] != -1) {
$query3 = 'LIMIT ' . $arr['start'] . ', ' . $arr['length'];
}
try {
//TOTAL DE REGISTROS NA TABELA
$query1 = "SELECT * FROM produtos WHERE prodEstatusID = 6";
$stm1 = $this->pdo->prepare($query1);
$stm1->execute();
$contReg = $stm1->rowCount($stm1);
$stm = $this->pdo->prepare($query2);
$stm->execute();
$number_filter_row = $stm->rowCount($stm);
$stm = $this->pdo->prepare($query2 . $query3);
$stm->execute();
$list = $stm->fetchAll(PDO::FETCH_OBJ);
$data = [];
foreach ($list as $row) {
$data[] = array('prod_nome' => $row->prodNome . ", " . $row->prodMedida, 'prod_preco' => "R$ ".$this->convInReal($row->prodPreco)." ".$this->formatProdOff($row->prodOferta), 'prod_estoque' => $row->prodQtdEst . " - M: " . $row->prodQtdMEst, 'prod_categoria' => $row->categDescricao);
}
$dat = array('draw' => intval($arr["draw"]),
'recordsTotal' => $contReg,
'recordsFiltered' => $number_filter_row,
'data' => $data
);
return json_encode($dat, JSON_UNESCAPED_UNICODE);
} catch (PDOException $erro) {
$data = array('msgEr' => 'ERR_002_EI', 'erroLine' => $erro->getLine());
return json_encode($data);
}
}
在这种情况下,您是否使用了联接查询?您也可以这样做。。通过选择tbl_house.style、tbl_house.roomCount、tbl_residents.firstName、tbl_residents.lastName,从tbl_house中选择加入tbl_house.houseID=tbl_resident上的tbl_resident。residentID@KumarRakesh我试图弄清楚如何将自定义SQL查询塞进Datatables服务器端处理SSP脚本中。我的问题不是如何构造SQL查询,而是如何在服务器端将其合并到Datatables的SSP框架中。我想我必须将自定义过滤器(即WHERE子句)硬连接到sss.class.php文件中。我开发了一个基于PDO适配器的服务器端数据表类,自6年以来,我几乎在所有项目中都完美地使用了它,在看到您的问题后,我刚刚在Github中托管了这个,这里有一个链接,如果它能帮助您自由提问,我会尽力回答,它支持任何加入,groupby等。感谢您的深入研究!我执行了你的建议,你是对的!datatables ssp.class.php不支持开箱即用的多个表。也就是说,$table='tbl_houses,tbl_residents'生成一个未找到的基表或视图:1146错误。然而,我确实找到了一个解决方案,并将在我有空的时候发布。您是否尝试过修改它以支持UNION ALL?@Milson一直想向Github发布一份草稿,并向Emran发出请求。我的MOD主要是简化用于调用ssp函数/过程的语法和文档的清晰度。添加联合功能应该很简单。看一看Emran为启用JOIN所做的更改,可以了解如何实现其他SQL函数
public function produtosEstoque($arr)
{
//COLUNAS
$column = array('prodNome', 'prodPreco', 'prodQtdEst', 'categDescricao');
//SQL
$query2 = "SELECT a.prodID, a.prodNome, a.prodMedida, a.prodPrecoAnt, a.prodPreco, a.prodCategoriaID, a.prodOferta,a.prodQtdEst, a.prodQtdMEst, b.categDescricao FROM produtos a INNER JOIN categoria b ON a.prodCategoriaID = b.categID WHERE a.prodEstatusID = 6 ";
//SEARCH
if ($arr['search']['value']) {
$query2 .= "AND prodNome LIKE '%".$arr['search']['value']."%' ";
}
//ORDER
if (isset($arr['order'])) {
$query2 .= 'ORDER BY ' . $column[$arr['order']['0']['column']] . ' ' . $arr['order']['0']['dir'] . ' ';
} else {
$query2 .= 'ORDER BY a.prodID DESC ';
}
//LIMIT
if ($arr["length"] != -1) {
$query3 = 'LIMIT ' . $arr['start'] . ', ' . $arr['length'];
}
try {
//TOTAL DE REGISTROS NA TABELA
$query1 = "SELECT * FROM produtos WHERE prodEstatusID = 6";
$stm1 = $this->pdo->prepare($query1);
$stm1->execute();
$contReg = $stm1->rowCount($stm1);
$stm = $this->pdo->prepare($query2);
$stm->execute();
$number_filter_row = $stm->rowCount($stm);
$stm = $this->pdo->prepare($query2 . $query3);
$stm->execute();
$list = $stm->fetchAll(PDO::FETCH_OBJ);
$data = [];
foreach ($list as $row) {
$data[] = array('prod_nome' => $row->prodNome . ", " . $row->prodMedida, 'prod_preco' => "R$ ".$this->convInReal($row->prodPreco)." ".$this->formatProdOff($row->prodOferta), 'prod_estoque' => $row->prodQtdEst . " - M: " . $row->prodQtdMEst, 'prod_categoria' => $row->categDescricao);
}
$dat = array('draw' => intval($arr["draw"]),
'recordsTotal' => $contReg,
'recordsFiltered' => $number_filter_row,
'data' => $data
);
return json_encode($dat, JSON_UNESCAPED_UNICODE);
} catch (PDOException $erro) {
$data = array('msgEr' => 'ERR_002_EI', 'erroLine' => $erro->getLine());
return json_encode($data);
}
}