实现datatable服务端分页
在上一篇文章中介绍了客户端分页的使用(https://blog.csdn.net/qq_38403662/article/details/85066142),这里介绍一下服务端分页的使用。
什么是服务器模式?
是不是发现在处理太多 dom 数据或者 ajax 一次性把数据获得后,datatables 表现的不是很满意?这是肯定的, 因为 dt 需要渲染,创建 tr/td ,所以数据越多,速度就越慢。 为了解决这个问题 datatables 提供了 服务器模式,把本来客户端所做的事情交给服务器去处理, 比如排序(order)、分页(paging)、过滤(filter)。对于客户端来说这些操作都是比较消耗资源的, 所以打开服务器模式后不用担心这些操作会影响到用户体验。
当你打开服务器模式的时候,每次绘制表格的时候,datatables 会给服务器发送一个请求(包括当前分页,排序,搜索参数等等)。datatables 会向 服务器发送 一些参数 去执行所需要的处理,然后在服务器组装好 相应的数据 返回给 datatables。
页面代码
省略插件导入代码~
首先页面
页面部分需要写的只是一个table标签,可根据自己的需求加上相应的样式和布局
<div class="content">
<div class="content-panel">
<div class="row">
<form id="form_condition">
<div class="row">
<div class="col-lg-6 form-group">
<label>姓名label>
<input type="text" name="name" class="form-control"/>
div>
<div class="col-lg-6 form-group">
<label>驾驶证编号label>
<input type="text" name="drivercard" class="form-control"/>
div>
div>
<div class="row">
<div class="col-lg-6 form-group">
<label>性别label>
<select name="sex" class="form-control">
<option value="">全部option>
<option value="男">男option>
<option value="女">女option>
select>
div>
<div class="col-lg-6 form-group">
<label>手机号label>
<input type="text" name="mobile" class="form-control"/>
div>
div>
<div class="row">
<div class="col-lg-6 form-group">
<label>状态label>
<select name="state" class="form-control">
<option value="">全部option>
<option value="0">就职状态option>
<option value="-1">离职状态option>
select>
div>
<div class="col-lg-6 form-group text-right">
<label style="display: block;"> label>
<a class="btn btn-success" th:href="${#httpservletrequest.getcontextpath() '/admin/driver/edit'}" >添加a>
<button class="btn btn-success" type="button" onclick="datatableinitial()">查询button>
<button class="btn btn-success" type="reset" >重置button>
div>
div>
form>
div>
div>
div>
<div class="content">
<div class="content-panel">
<div class="row">
<table class="table table-bordered table-hover" id="driverlisttable">table>
div>
div>
div>
js代码
当开启了 服务器模式时,datatables 会发送如下参数到服务器
ps:需要说明的是
如果你是 java 开发者,那么使用struts2的需要注意,会有错误抛出,因为处理不了类似 columns[i][search][regex]的变量
如果是你 .net 开发者,那么可能会遇到 maxquerystringlength 的错误
如果是你 php 开发者,那么恭喜你,php天生支持以上参数的解析,自动转为数组,好不公平啊
名称 | 类型 | 描述 |
---|---|---|
draw(secho) | integer | 绘制计数器。这个是用来确保ajax从服务器返回的是对应的(ajax是异步的,因此返回的顺序是不确定的)。 要求在服务器接收到此参数后再返回 |
start(idisplaystart) | integer | 第一条数据的起始位置,比如0代表第一条数据 |
length(idisplaylength) | integer | 告诉服务器每页显示的条数,这个数字会等于返回的 data集合的记录数,可能会大于因为服务器可能没有那么多数据。这个也可能是-1,代表需要返回全部数据(尽管这个和服务器处理的理念有点违背) |
这里就只介绍几个必须的,更多全局搜索、排序等参数传递请参阅凯发网娱乐官网下载官网datatables中文网
举个栗子:
js代码:
var otable;
$(function () {
datatableinitial();
});
/**
* 构建datatable
*/
function datatableinitial(){
if (typeof otable == "undefined"){
otable = $("#driverlisttable").datatable({
"bpaginate":true,//翻页功能
"sservermethod":"post",//请求方式
"bserverside":true,//是否开启服务端分页
"bprocessing":true,//是否显示加载ing
"bfilter":false,//是否开启过滤
"bsort":false,//是否开启排序
"sajaxsource":"admin/driver",//请求地址
"fnserverparams":function (aodata) {//查询条件,额外参数,
var $form = $("#form_condition");//如果没有额外参数请省略
aodata.push({
"name":"name",
"value":$form.find("[name=name]").val()
});
aodata.push({
"name":"drivercard",
"value":$form.find("[name=drivercard]").val()
});
aodata.push({
"name":"sex",
"value":$form.find("[name=sex]").val()
});
aodata.push({
"name":"mobile",
"value":$form.find("[name=mobile]").val()
});
aodata.push({
"name":"state",
"value":$form.find("[name=state]").val()
});
},
"aocolumns":[{//每一列对应的数据
"mdata":"id",
"stitle":"序号",
"mrender":function (data,type,full,meta) {//不清楚参数内的数据的可使用console.log();打印出来看看
return meta.row1meta.settings._idisplaystart;
}
},{
"mdata":"name",
"stitle":"姓名"
},{
"mdata":"drivercard",
"stitle":"驾驶证编号"
},{
"mdata":"sex",
"stitle":"性别"
},{
"mdata":"age",
"stitle":"年龄"
},{
"mdata":"mobile",
"stitle":"手机"
},{
"mdata":"state",
"stitle":"状态",
"mrender":function (data,type,full,meta) {
return data == 0?"就职状态":"离职状态";
}
},{
"mdata":"id",
"stitle":"操作",
"width":"15%",
"mrender":function (data,type,full,meta) {
var str = "";
str = "";
str = "";
return str;
}
}]
});
} else{
// otable.fncleartable(0);
var osettings = otable.fnsettings();
//注意下面的driverlisttable是你需挂载的表格的id
osettings._idisplaylength = parseint($(
'[name=driverlisttable_length] option:selected').val());
$('.datatables_length select').val(
$('[name=driverlisttable_length] option:selected').val());
//osettings._idisplaystart = 0;
otable.fndraw();//重绘表格
}
}
后台处理
一旦 datatables 发送了请求,上面的参数就会传送给服务器,那么你需要接受到这些参数并做相应的逻辑处理然后按照下面的格式讲组装好的json数据返回 (不是每个参数都需要接受处理,根据自己的业务需要)
名称 | 类型 | 描述 |
---|---|---|
draw(secho) | integer | 必要。上面提到了,datatables发送的draw是多少那么服务器就返回多少。 这里注意,作者出于安全的考虑,强烈要求把这个转换为整形,即数字后再返回,而不是纯粹的接受然后返回,这是 为了防止跨站脚本(xss)攻击。 |
recordstotal(itotalrecords) | integer | 必要。即没有过滤的记录数(数据库里总共记录数) |
recordsfiltered(itotaldisplayrecords) | integer | 必要。过滤后的记录数(如果有接收到前台的过滤条件,则返回的是过滤后的记录数) |
data(aadata) | array | 必要。表中中需要显示的数据。这是一个对象数组,也可以只是数组,区别在于 纯数组前台就不需要用 columns绑定数据,会自动按照顺序去显示 ,而对象数组则需要使用 columns绑定数据才能正常显示。 注意这个 data的名称可以由 ajaxoption ajax不定时一讲 的 ajax.datasrcoption ajax.datasrc 1不定时一讲 ajax.datasrc 2不定时一讲 控制 |
error | integer | 可选。你可以定义一个错误来描述服务器出了问题后的友好提示 |
(ps:括号里面是我使用的,前面是凯发网娱乐官网下载官网说明的,对于datatables的参数,有很多种写法,讲道理应该都可以,试试就知道了?)
响应报文例子:
凯发网娱乐官网下载官网响应报文例子:
{
"draw": 1,
"recordstotal": 57,
"recordsfiltered": 57,
"data": [
{
"dt_rowid": "row_3",
"dt_rowdata": {
"pkey": 3
},
"first_name": "angelica",
"last_name": "ramos",
"position": "system architect",
"office": "london",
"start_date": "9th oct 09",
"salary": "$2,875"
},
{
"dt_rowid": "row_17",
"dt_rowdata": {
"pkey": 17
},
"first_name": "ashton",
"last_name": "cox",
"position": "technical author",
"office": "san francisco",
"start_date": "12th jan 09",
"salary": "$4,800"
},
...
]
}
后台处理例子:
/**
* 通过datatable获取司机列表
* @param pageinfomation datatable传递参数(这个自己根据传递的参数去封装一个对象即可)
* @param driver 自定义搜索条件参数
* @return
*/
@requestmapping(value="/driver",method = requestmethod.post)
@responsebody
public map<string ,object> driverdatatablelist(pageinfomation pageinfomation, driver driver){
map<string,object> map = new hashmap<>();
try {
map = this.driverservice.driverdatatableinfo(pageinfomation,driver);
}catch (exception e){
e.printstacktrace();
map.put("secho",pageinfomation.getsecho());
map.put("itotalrecords",0);
map.put("itotaldisplayrecords",0);
map.put("aadata",null);
}
return map;
}
运行效果图
资源地址
接下来附上博客使用的datatable插件资源的地址,需要可自取
百度网盘地址:链接: http://www.51sjk.com/upload/articles/1/0/251/251056_20210626000347319.jpg 提取码: 2eiy 复制这段内容后打开百度网盘手机app,操作更方便哦
项目的github地址:https://github.com/jamesluozhiwei/bus.git
csdn下载地址:https://download.csdn.net/download/qq_38403662/10862576
有问题和不足欢迎留言讨论