使用 Datatables v1.10.0 进行服务器端处理

Posted

技术标签:

【中文标题】使用 Datatables v1.10.0 进行服务器端处理【英文标题】:Server Side processing with Datatables v1.10.0 【发布时间】:2014-06-24 08:30:55 【问题描述】:

您好,我在让数据表的服务器端处理功能与 SQL Server 一起工作时遇到了一些问题。

我有一个测试页面,应该显示数据库表中的两列(目前)。

html

<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">

<html xmlns="http://www.w3.org/1999/xhtml">
<head>
    <title></title>
    <link rel="Stylesheet" type="text/css" href="DataTables-1.10.0/media/css/jquery.dataTables.min.css" />
</head>
<body>
<table id="example" class="display" cellspacing="0" >
<thead>
    <tr>
            <th align="center">PK</th>
            <th align="center">Network</th>               
    </tr>
</thead>

<tfoot>
    <tr>
            <th align="center">PK</th>
            <th align="center">Network</th>               
    </tr>
</tfoot>
</table>
</body>
<script type="text/javascript" src="DataTables-1.10.0/media/js/jquery.js"></script>
<script type="text/javascript" src="DataTables-1.10.0/media/js/jquery.dataTables.min.js">

</script>
<script type="text/javascript" charset="utf-8">
$(document).ready(function () 
    $('#example').dataTable(
        "processing": true,
        "bServerSide": true,
        "ajax": "php/testGetArchive.php"
    );
);
</script>

</html>

我正在使用网站上的示例代码来实现服务器端功能:

http://next.datatables.net/examples/server_side/simple.html

这是我调用的 php 页面的版本:

<?php

/*
 * DataTables example server-side processing script.
 *
 * Please note that this script is intentionally extremely simply to show how
 * server-side processing can be implemented, and probably shouldn't be used as
 * the basis for a large complex system. It is suitable for simple use cases as
 * for learning.
 *
 * See http://datatables.net/usage/server-side for full details on the server-
 * side processing requirements of DataTables.
 *
 * @license MIT - http://datatables.net/license_mit
 */

/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
 * Easy set variables
 */

// DB table to use
$table = 'tblViews';

// Table's primary key
$primaryKey = 'PK';

// Array of database columns which should be read and sent back to DataTables.
// The `db` parameter represents the column name in the database, while the `dt`
// parameter represents the DataTables column identifier. In this case simple
// indexes
$columns = array(
    array( 'db' => 'PK', 'dt' => 0 ),
    array( 'db' => 'Network',  'dt' => 1 )
);

// SQL server connection information
$sql_details = array(
    'user' => '******',
    'pass' => '******',
    'db'   => '******db',
    'host' => '******\SQLEXPRESS'
);


/* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
 * If you just want to use the basic configuration for DataTables with PHP
 * server-side, there is no need to edit below this line.
 */

require( 'ssp.class.php' );

echo json_encode(
    SSP::simple( $_GET, $sql_details, $table, $primaryKey, $columns )
);

然后调用此处找到的第二个 PHP 示例:

https://github.com/DataTables/DataTables/blob/master/examples/server_side/scripts/ssp.class.php

这是我的副本。我执行的唯一修改是删除示例所需的代码块。

<?php

/*
 * Helper functions for building a DataTables server-side processing SQL query
 *
 * The static functions in this class are just helper functions to help build
 * the SQL used in the DataTables demo server-side processing scripts. These
 * functions obviously do not represent all that can be done with server-side
 * processing, they are intentionally simple to show how it works. More complex
 * server-side processing operations will likely require a custom script.
 *
 * See http://datatables.net/usage/server-side for full details on the server-
 * side processing requirements of DataTables.
 *
 * @license MIT - http://datatables.net/license_mit
 */

class SSP 
    /**
     * Create the data output array for the DataTables rows
     *
     *  @param  array $columns Column information array
     *  @param  array $data    Data from the SQL get
     *  @return array          Formatted data in a row based format
     */
    static function data_output ( $columns, $data )
    
        $out = array();

        for ( $i=0, $ien=count($data) ; $i<$ien ; $i++ ) 
            $row = array();

            for ( $j=0, $jen=count($columns) ; $j<$jen ; $j++ ) 
                $column = $columns[$j];

                // Is there a formatter?
                if ( isset( $column['formatter'] ) ) 
                    $row[ $column['dt'] ] = $column['formatter']( $data[$i][ $column['db'] ], $data[$i] );
                
                else 
                    $row[ $column['dt'] ] = $data[$i][ $columns[$j]['db'] ];
                
            

            $out[] = $row;
        

        return $out;
    


    /**
     * Paging
     *
     * Construct the LIMIT clause for server-side processing SQL query
     *
     *  @param  array $request Data sent to server by DataTables
     *  @param  array $columns Column information array
     *  @return string SQL limit clause
     */
    static function limit ( $request, $columns )
    
        $limit = '';

        if ( isset($request['start']) && $request['length'] != -1 ) 
            $limit = "LIMIT ".intval($request['start']).", ".intval($request['length']);
        

        return $limit;
    


    /**
     * Ordering
     *
     * Construct the ORDER BY clause for server-side processing SQL query
     *
     *  @param  array $request Data sent to server by DataTables
     *  @param  array $columns Column information array
     *  @return string SQL order by clause
     */
    static function order ( $request, $columns )
    
        $order = '';

        if ( isset($request['order']) && count($request['order']) ) 
            $orderBy = array();
            $dtColumns = SSP::pluck( $columns, 'dt' );

            for ( $i=0, $ien=count($request['order']) ; $i<$ien ; $i++ ) 
                // Convert the column index into the column data property
                $columnIdx = intval($request['order'][$i]['column']);
                $requestColumn = $request['columns'][$columnIdx];

                $columnIdx = array_search( $requestColumn['data'], $dtColumns );
                $column = $columns[ $columnIdx ];

                if ( $requestColumn['orderable'] == 'true' ) 
                    $dir = $request['order'][$i]['dir'] === 'asc' ?
                        'ASC' :
                        'DESC';

                    $orderBy[] = '`'.$column['db'].'` '.$dir;
                
            

            $order = 'ORDER BY '.implode(', ', $orderBy);
        

        return $order;
    


    /**
     * Searching / Filtering
     *
     * Construct the WHERE clause for server-side processing SQL query.
     *
     * NOTE this does not match the built-in DataTables filtering which does it
     * word by word on any field. It's possible to do here performance on large
     * databases would be very poor
     *
     *  @param  array $request Data sent to server by DataTables
     *  @param  array $columns Column information array
     *  @param  array $bindings Array of values for PDO bindings, used in the
     *    sql_exec() function
     *  @return string SQL where clause
     */
    static function filter ( $request, $columns, &$bindings )
    
        $globalSearch = array();
        $columnSearch = array();
        $dtColumns = SSP::pluck( $columns, 'dt' );

        if ( isset($request['search']) && $request['search']['value'] != '' ) 
            $str = $request['search']['value'];

            for ( $i=0, $ien=count($request['columns']) ; $i<$ien ; $i++ ) 
                $requestColumn = $request['columns'][$i];
                $columnIdx = array_search( $requestColumn['data'], $dtColumns );
                $column = $columns[ $columnIdx ];

                if ( $requestColumn['searchable'] == 'true' ) 
                    $binding = SSP::bind( $bindings, '%'.$str.'%', PDO::PARAM_STR );
                    $globalSearch[] = "`".$column['db']."` LIKE ".$binding;
                
            
        

        // Individual column filtering
        for ( $i=0, $ien=count($request['columns']) ; $i<$ien ; $i++ ) 
            $requestColumn = $request['columns'][$i];
            $columnIdx = array_search( $requestColumn['data'], $dtColumns );
            $column = $columns[ $columnIdx ];

            $str = $requestColumn['search']['value'];

            if ( $requestColumn['searchable'] == 'true' &&
             $str != '' ) 
                $binding = SSP::bind( $bindings, '%'.$str.'%', PDO::PARAM_STR );
                $columnSearch[] = "`".$column['db']."` LIKE ".$binding;
            
        

        // Combine the filters into a single string
        $where = '';

        if ( count( $globalSearch ) ) 
            $where = '('.implode(' OR ', $globalSearch).')';
        

        if ( count( $columnSearch ) ) 
            $where = $where === '' ?
                implode(' AND ', $columnSearch) :
                $where .' AND '. implode(' AND ', $columnSearch);
        

        if ( $where !== '' ) 
            $where = 'WHERE '.$where;
        

        return $where;
    


    /**
     * Perform the SQL queries needed for an server-side processing requested,
     * utilising the helper functions of this class, limit(), order() and
     * filter() among others. The returned array is ready to be encoded as JSON
     * in response to an SSP request, or can be modified if needed before
     * sending back to the client.
     *
     *  @param  array $request Data sent to server by DataTables
     *  @param  array $sql_details SQL connection details - see sql_connect()
     *  @param  string $table SQL table to query
     *  @param  string $primaryKey Primary key of the table
     *  @param  array $columns Column information array
     *  @return array          Server-side processing response array
     */
    static function simple ( $request, $sql_details, $table, $primaryKey, $columns )
    
        $bindings = array();
        $db = SSP::sql_connect( $sql_details );

        // Build the SQL query string from the request
        $limit = SSP::limit( $request, $columns );
        $order = SSP::order( $request, $columns );
        $where = SSP::filter( $request, $columns, $bindings );

        // Main query to actually get the data
        $data = SSP::sql_exec( $db, $bindings,
            "SELECT SQL_CALC_FOUND_ROWS `".implode("`, `", SSP::pluck($columns, 'db'))."`
             FROM `$table`
             $where
             $order
             $limit"
        );

        // Data set length after filtering
        $resFilterLength = SSP::sql_exec( $db,
            "SELECT FOUND_ROWS()"
        );
        $recordsFiltered = $resFilterLength[0][0];

        // Total data set length
        $resTotalLength = SSP::sql_exec( $db,
            "SELECT COUNT(`$primaryKey`)
             FROM   `$table`"
        );
        $recordsTotal = $resTotalLength[0][0];


        /*
         * Output
         */
        return array(
            "draw"            => intval( $request['draw'] ),
            "recordsTotal"    => intval( $recordsTotal ),
            "recordsFiltered" => intval( $recordsFiltered ),
            "data"            => SSP::data_output( $columns, $data )
        );
    


    /**
     * Connect to the database
     *
     * @param  array $sql_details SQL server connection details array, with the
     *   properties:
     *     * host - host name
     *     * db   - database name
     *     * user - user name
     *     * pass - user password
     * @return resource Database connection handle
     */
    static function sql_connect ( $sql_details )
    
        try 
            $db = @new PDO(
                "mysql:host=$sql_details['host'];dbname=$sql_details['db']",
                $sql_details['user'],
                $sql_details['pass'],
                array( PDO::ATTR_ERRMODE => PDO::ERRMODE_EXCEPTION )
            );
        
        catch (PDOException $e) 
            SSP::fatal(
                "An error occurred while connecting to the database. ".
                "The error reported by the server was: ".$e->getMessage()
            );
        

        return $db;
    


    /**
     * Execute an SQL query on the database
     *
     * @param  resource $db  Database handler
     * @param  array    $bindings Array of PDO binding values from bind() to be
     *   used for safely escaping strings. Note that this can be given as the
     *   SQL query string if no bindings are required.
     * @param  string   $sql SQL query to execute.
     * @return array         Result from the query (all rows)
     */
    static function sql_exec ( $db, $bindings, $sql=null )
    
        // Argument shifting
        if ( $sql === null ) 
            $sql = $bindings;
        

        $stmt = $db->prepare( $sql );
        //echo $sql;

        // Bind parameters
        if ( is_array( $bindings ) ) 
            for ( $i=0, $ien=count($bindings) ; $i<$ien ; $i++ ) 
                $binding = $bindings[$i];
                $stmt->bindValue( $binding['key'], $binding['val'], $binding['type'] );
            
        

        // Execute
        try 
            $stmt->execute();
        
        catch (PDOException $e) 
            SSP::fatal( "An SQL error occurred: ".$e->getMessage() );
        

        // Return all
        return $stmt->fetchAll();
    


    /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
     * Internal methods
     */

    /**
     * Throw a fatal error.
     *
     * This writes out an error message in a JSON string which DataTables will
     * see and show to the user in the browser.
     *
     * @param  string $msg Message to send to the client
     */
    static function fatal ( $msg )
    
        echo json_encode( array( 
            "error" => $msg
        ) );

        exit(0);
    

    /**
     * Create a PDO binding key which can be used for escaping variables safely
     * when executing a query with sql_exec()
     *
     * @param  array &$a    Array of bindings
     * @param  *      $val  Value to bind
     * @param  int    $type PDO field type
     * @return string       Bound key to be used in the SQL where this parameter
     *   would be used.
     */
    static function bind ( &$a, $val, $type )
    
        $key = ':binding_'.count( $a );

        $a[] = array(
            'key' => $key,
            'val' => $val,
            'type' => $type
        );

        return $key;
    


    /**
     * Pull a particular property from each assoc. array in a numeric array, 
     * returning and array of the property values from each item.
     *
     *  @param  array  $a    Array to get data from
     *  @param  string $prop Property to read
     *  @return array        Array of property values
     */
    static function pluck ( $a, $prop )
    
        $out = array();

        for ( $i=0, $len=count($a) ; $i<$len ; $i++ ) 
            $out[] = $a[$i][$prop];
        

        return $out;
    

我一直收到错误消息,说代码找不到驱动程序,尽管我已经在我的 php 环境中安装了 sqlserv 和 pdo_sqlsrv 驱动程序。导致此错误的代码是否有问题?我的驱动程序不正确吗?对此的任何帮助将不胜感激。我有超过 65,000 行数据要处理,并且一次性将所有数据发送给客户端是不可能的。

【问题讨论】:

【参考方案1】:

我花了一段时间,但我发现了哪里出了问题,现在我的 DataTables 通过服务器端脚本与 SQL Server 一起工作。我发布此解决方案是希望它能帮助像我这样遇到问题的其他人。我已将我的答案分成几部分。

PHP 环境

可以在here 找到用于 php 的 SQLSRV 驱动程序。下载 SQLSRV30.EXE 安装程序包。您可能会发现,当您尝试运行此可执行文件时,您会收到错误消息“这不是有效的 win32 应用程序”如果是这种情况,请使用 7-zip 之类的文件解压缩可执行文件。生成的文件将包含您需要的文件。

解压缩包后,您需要选择正确的驱动程序。大多数 Windows 安装使用非线程安全驱动程序,它们是:

php 5.3 版:

php_sqlsrv_53_nts.dll

php_pdo_sqlsrv_53_nts.dll

php 5.4 版:

php_sqlsrv_54_nts.dll

php_pdo_sqlsrv_54_nts.dll

将相应的文件复制到 php 目录中的 ext 文件夹中。现在修改您的 php.ini 文件以引用这些文件。通过在动态扩展部分下添加一个条目来执行此操作。结果会是这样的:

extension=php_sqlsrv_54_nts.dll

然后在模块部分设置下为驱动程序添加一个部分,如下所示:

[sqlsrv]
sqlsrv.LogSubSystems=-1
sqlsrv.LogSeverity=-1
sqlsrv.WarningsReturnAsErrors=0

这些设置的文档可以在here找到。

一旦您添加了这些驱动程序并在 php.ini 文件中添加了对它们的引用,您还必须确保还安装了 Microsoft SQL Server Client Profile 2012。

These Links have been taken from the PHP.net website:

Microsoft SQL Server Client Profile 2012 x86 Microsoft SQL Server Client profile 2012 x64

执行这些步骤后,重新启动您的网络服务器。驱动程序现在应该已安装并可以使用了。您可以使用您的 info.php 页面进行检查。

服务器端脚本:

现在 Web 服务器已配置为使用 SQL SRV 驱动程序,我们现在可以使用它来查询 SQL Server 数据库。我使用了可用的服务器端脚本here。以下是我发现的一些问题:

<?php
    /* Indexed column (used for fast and accurate table cardinality) */
    $sIndexColumn = "";

    /* DB table to use */
    $sTable = "";

    /* Database connection information */
    $gaSql['user']       = "";
    $gaSql['password']   = "";
    $gaSql['db']         = "";
    $gaSql['server']     = "";

    /*
    * Columns
    * If you don't want all of the columns displayed you need to hardcode $aColumns array with your elements.
    * If not this will grab all the columns associated with $sTable
    */
    $aColumns = array();


    /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
     * If you just want to use the basic configuration for DataTables with PHP server-side, there is
     * no need to edit below this line
     */

    /*
     * ODBC connection
     */
    $connectionInfo = array("UID" => $gaSql['user'], "PWD" => $gaSql['password'], "Database"=>$gaSql['db'],"ReturnDatesAsStrings"=>true);
    $gaSql['link'] = sqlsrv_connect( $gaSql['server'], $connectionInfo);
    $params = array();
    $options =  array( "Scrollable" => SQLSRV_CURSOR_KEYSET );


    /* Ordering */
    $sOrder = "";
    if ( isset( $_GET['iSortCol_0'] ) ) 
        $sOrder = "ORDER BY  ";
        for ( $i=0 ; $i<intval( $_GET['iSortingCols'] ) ; $i++ ) 
            if ( $_GET[ 'bSortable_'.intval($_GET['iSortCol_'.$i]) ] == "true" ) 
                $sOrder .= $aColumns[ intval( $_GET['iSortCol_'.$i] ) ]."
                    ".addslashes( $_GET['sSortDir_'.$i] ) .", ";
            
        
        $sOrder = substr_replace( $sOrder, "", -2 );
        if ( $sOrder == "ORDER BY" ) 
            $sOrder = "";
        
    

    /* Filtering */
    $sWhere = "";
    if ( isset($_GET['sSearch']) && $_GET['sSearch'] != "" ) 
        $sWhere = "WHERE (";
        for ( $i=0 ; $i<count($aColumns) ; $i++ ) 
            $sWhere .= $aColumns[$i]." LIKE '%".addslashes( $_GET['sSearch'] )."%' OR ";
        
        $sWhere = substr_replace( $sWhere, "", -3 );
        $sWhere .= ')';
    
    /* Individual column filtering */
    for ( $i=0 ; $i<count($aColumns) ; $i++ ) 
        if ( isset($_GET['bSearchable_'.$i]) && $_GET['bSearchable_'.$i] == "true" && $_GET['sSearch_'.$i] != '' )  
            if ( $sWhere == "" ) 
                $sWhere = "WHERE ";
             else 
                $sWhere .= " AND ";
            
            $sWhere .= $aColumns[$i]." LIKE '%".addslashes($_GET['sSearch_'.$i])."%' ";
        
    

    /* Paging */
    $top = (isset($_GET['iDisplayStart']))?((int)$_GET['iDisplayStart']):0 ;
    $limit = (isset($_GET['iDisplayLength']))?((int)$_GET['iDisplayLength'] ):10;
    $sQuery = "SELECT TOP $limit ".implode(",",$aColumns)."
        FROM $sTable
        $sWhere ".(($sWhere=="")?" WHERE ":" AND ")." $sIndexColumn NOT IN
        (
            SELECT $sIndexColumn FROM
            (
                SELECT TOP $top ".implode(",",$aColumns)."
                FROM $sTable
                $sWhere
                $sOrder
            )
            as [virtTable]
        )
        $sOrder";

    $rResult = sqlsrv_query($gaSql['link'],$sQuery) or die("$sQuery: " . sqlsrv_errors());

    $sQueryCnt = "SELECT * FROM $sTable $sWhere";
    $rResultCnt = sqlsrv_query( $gaSql['link'], $sQueryCnt ,$params, $options) or die (" $sQueryCnt: " . sqlsrv_errors());
    $iFilteredTotal = sqlsrv_num_rows( $rResultCnt );

    $sQuery = " SELECT * FROM $sTable ";
    $rResultTotal = sqlsrv_query( $gaSql['link'], $sQuery ,$params, $options) or die(sqlsrv_errors());
    $iTotal = sqlsrv_num_rows( $rResultTotal );

    $output = array(
        "sEcho" => intval($_GET['sEcho']),
        "iTotalRecords" => $iTotal,
        "iTotalDisplayRecords" => $iFilteredTotal,
        "aaData" => array()
    );

    while ( $aRow = sqlsrv_fetch_array( $rResult ) ) 
        $row = array();
        for ( $i=0 ; $i<count($aColumns) ; $i++ ) 
            if ( $aColumns[$i] != ' ' ) 
                $v = $aRow[ $aColumns[$i] ];
                $v = mb_check_encoding($v, 'UTF-8') ? $v : utf8_encode($v);
                $row[]=$v;
            
        
        If (!empty($row))  $output['aaData'][] = $row; 
       
    echo json_encode( $output );
?>

索引列

当您指定用于搜索的索引列时,请确保它包含在列数组中!如果在指定要使用分页的列时将其省略,则该分页将不起作用。当不在另一个查询的 TOP X 结果中时,使用此代码对数据表进行分页可以对所有主键执行选择查询。

连接参数

确保连接参数完整且正确。这些是允许脚本连接到数据库所必需的。如果没有任何参数或参数对 SQL 服务器登录不正确,则脚本将永远无法连接到数据库。

列数组

我发现在没有指定列的情况下使用此代码会返回不正确或 NULL 数据。阻止这种情况的最好方法是用我想选择的列名填充数组,每个列名用引号括起来并用逗号分隔。这也是有道理的,为什么要向客户端发送除所需数据之外的任何内容?

客户端

HTML

DataTables 需要一个格式良好的 html 表格才能运行。这意味着有一个带有完整标签的表格。如果返回数据的所有标签都不存在,则 DataTables 将返回错误。如果您有想要返回但不显示的列,那么您可以使用ColVis exntension 并在 java 脚本中设置默认的列视图设置。

Datatable 使用自己的 CCS 文件,因此请确保包含它!

java 脚本

DataTables 使用 Jquery 和它自己的 Javascrpt 文件,因此请确保在脚本标签中包含对它们的引用!

//Datatables Basic server side initilization
$(document).ready(function () 

    //Datatable
    var table = $('#tableID').DataTable(
        "bProcessing": true,
        "bServerSide": true,
        "sAjaxSource": "serverSideScript.php"
    );    
);            

这些是此服务器端脚本工作所需的基本功能。它将使用您在 php 页面中指定的数据库参数在初始绘制时获得前 10 行。您可以在此处添加所需的扩展名,例如 ColVis 和 TableTools。可以在here 找到这些扩展和其他数据表初始化选项的完整文档。

我希望这个答案可以帮助遇到与我相同问题的其他人。

【讨论】:

谢谢伙计!试图找到类似的东西。 它现在有点过时了,不是我会这样做的方式,但它在某种程度上确实有效。我仍然没有找到完美的解决方案,但当我最终找到时,我会更新另一个答案。

以上是关于使用 Datatables v1.10.0 进行服务器端处理的主要内容,如果未能解决你的问题,请参考以下文章

新增MaxCompute数据源支持,平台支持ARM64架构部署,DataEase开源数据可视化分析平台v1.10.0发布

centos7 用kubeadm安装 k8s v1.10.0

kubectl 安装

使用 jQuery DataTables 进行数据可视化

DataTables - 仅对选定的列进行排序?

在 DataTables 中使用 HTML 对数值数据进行排序