如何使用 ssp.class.php 连接两个表

Posted

技术标签:

【中文标题】如何使用 ssp.class.php 连接两个表【英文标题】:How to join two tables with ssp.class.php 【发布时间】:2015-09-09 09:01:56 【问题描述】:

我开始使用DataTables Table plug-in for jQuery 并遇到了一些问题。我正在使用来自here 的示例代码。

我的 mysql 表是这样的:

标识 |姓名 |父亲身份

father_id 是同一表中的 id 值,仅在不同行中。所以如果我想知道父亲的名字,我必须在同一张表中搜索WHERE id = father_id。但是 DataTable 是做什么的,它只是显示 MySQL 表的内容。

在我的 DataTable 中,我想显示这样的数据:

标识 |姓名 |父亲姓名 |父亲身份

因此,当 DataTable 从 MySQL 表中获取数据时,但在它创建表之前,我想更改列值,当时是 MySQL 同一行中的 father_id 值。我也想通过使用特定的father_id 搜索来添加father_name

【问题讨论】:

如何处理rdbms方面的问题?您可以选择在源中添加该字段吗?使用源表上的自联接来更改源查询会更快、更有效。 DataTables 本身在这个问题上是不可知的。需要解决的所有问题都会导致数据集返回到浏览器。 【参考方案1】:

正如PaulF 指出的,您需要使用JOIN 或子查询从同一张表中检索父亲的姓名。

我假设您正在使用 ssp.class.php 根据您提到的示例在服务器端处理您的数据。

ssp.class.php 类不支持连接和子查询,但有一种解决方法。诀窍是使用如下$table 定义中所示的子查询。在子查询中将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 以删除反引号。

确保所有列名都是唯一的,否则使用AS 分配别名。

注意事项

还有 github.com/emran/ssp 存储库,其中包含增强的 ssp.class.php 支持 JOIN。

链接

请参阅jQuery DataTables: Using WHERE, JOIN and GROUP BY with ssp.class.php 了解更多信息。

【讨论】:

感谢您提供非常好的信息性答案。但是我遇到了一个问题。现在它告诉我“SQLSTATE [42000]:语法错误或访问冲突:1059 标识符名称太长” @hockeyman,你是对的,你还需要编辑 ssp.class.php 并将所有 FROM `$table` 实例替换为 FROM $table 以删除反引号。我已经更新了我的答案,还添加了包含增强型ssp.class.php 的存储库。 现在如果father_id 没有值,它不会显示该行。如果没有father_id,我该怎么做才能只显示father_name为空的行? @hockeyman,我修复了 SQL 的一个小问题,但是即使没有找到 father_name,它也应该返回一条记录。也许 DataTables 抱怨 JSON 中有 null 值的字段,尝试使用 (CASE WHEN b.name IS NOT NULL THEN b.name ELSE '' END) AS father_name 而不是 b.name AS father_name 看看是否有帮助。 很好(Y)谢谢【参考方案2】:

最近我发现自己在使用 DataTables 并且需要更复杂的 JOIN 和 WHERE 子句,而原始 ssp.class.php 不支持。因此,我修改了原始文件并稍微更改了 API,以提供一个表格,为我提供所需的灵活性。我将“SSP::simple”和“SSP::complex”的功能组合成一个名为“SSP::process”的函数。

由于脚本篇幅较长,我把它放到了pastebin.com这里:ssp.class.php

还有一个我如何使用它的简单示例:

private function get_recent_payments() 
    global 
        $pdoHost, $pdoUser, 
        $pdoPass, $pdoDatabase;

    // SQL server connection information
    $sql_details = array(
        'user' => $pdoUser,
        'pass' => $pdoPass,
        'db'   => $pdoDatabase,
        'host' => $pdoHost
    );

    // DataTables server-side processing
    require_once('ssp.class.php');

    $options = [
        'table' => 'payments',
        'alias' => 'l',
        'primaryKey' => 'id',
        'columns' => [
            [ 'db' => 'id',       'dt' => 0 ],
            [
                'db' => 'client_id',     
                'dt' => 1,
                'join' => [
                    'table' => 'clients',
                    'on' => 'id',
                    'select' => 'first_name',
                    'alias' => 'c',
                    'as' => 'fname',
                ]
            ],
            [
                'db' => 'client_id',     
                'dt' => 2,
                'join' => [
                    'table' => 'clients',
                    'on' => 'id',
                    'select' => 'last_name',
                    'alias' => 'c'
                ]
            ],
            [ 'db' => 'pay_date', 'dt' => 3 ]
        ],
        'where' => [
            [
                'db' => 'client_id',
                'op' => '!=',
                'value' => $_SESSION['client_id']
            ]
        ]
    ];

    $this->response(SSP::process($_REQUEST, $sql_details, $options));

选项数组的“where”和“whereResult”(详见“SSP::complex”)子句也可能有一个“别名”来引用连接表中的列。

传递给服务器的示例 SQL 查询:

SELECT l.`id`, c.`first_name` AS 'fname', c.`last_name`, l.`pay_date` 
        FROM `payments` l
        JOIN `clients` c ON (c.`id` = l.`client_id`)
        WHERE l.`client_id` != :binding_0
        ORDER BY l.`pay_date` DESC
        LIMIT 0, 5

我采用结构化数组路线,因为这使我能够构建查询,同时使用反引号和绑定语句参数保持查询的刚性。我发布这篇文章是希望其他人会发现它和我一样有用。

【讨论】:

【参考方案3】:

将表与自身连接 - 您将需要使用别名

SELECT a.id, a.name, b.name, b.id
FROM table a
join table b on (b.father_id=a.id);

【讨论】:

以上是关于如何使用 ssp.class.php 连接两个表的主要内容,如果未能解决你的问题,请参考以下文章

YADCF + Datatables 服务器端使用 Php 填充选择

laravel 的数据表服务器端 php 类

sql与数据库学习加深

sql与数据库学习加深

数据可视化之DAX篇(二十一)连接表的几个DAX函数,一次全掌握

ORACle中对于数据量庞大的表如何用delete+insert来代替update语句操作步骤或者请举例说明