PHP分页显示制作详细讲解


Posted in PHP onNovember 19, 2008

每页多少条记录($PageSize)?

当前是第几页($CurrentPageID)?

现在只要再给我一个结果集,我就可以显示某段特定的结果出来。

至于其他的参数,比如:上一页($PreviousPageID)、下一页($NextPageID)、总页数($numPages)等等,都可以根据前边这几个东西得到。

以mysql数据库为例,如果要从表内截取某段内容,sql语句可以用:select * from table limit offset, rows。看看下面一组sql语句,尝试一下发现其中的规率。

前10条记录:select * from table limit 0,10

第11至20条记录:select * from table limit 10,10

第21至30条记录:select * from table limit 20,10

……

这一组sql语句其实就是当$PageSize=10的时候取表内每一页数据的sql语句,我们可以总结出这样一个模板:
select * from table limit ($CurrentPageID - 1) * $PageSize, $PageSize

拿这个模板代入对应的值和上边那一组sql语句对照一下看看是不是那么回事。搞定了最重要的如何获取数据的问题以后,剩下的就仅仅是传递参数,构造合适的sql语句然后使用php从数据库内获取数据并显示了。以下我将用具体代码加以说明。

3、简单代码

请详细阅读以下代码,自己调试运行一次,最好把它修改一次,加上自己的功能,比如搜索等等。
<?php
// 建立数据库连接
$link = mysql_connect("localhost", "mysql_user", "mysql_password")

 or die("Could not connect: " . mysql_error());
// 获取当前页数
if( isset($_GET['page']) ){
  $page = intval( $_GET['page'] );
}
else{
  $page = 1;
}
// 每页数量
$PageSize = 10;
// 获取总数据量
$sql = "select count(*) as amount from table";
$result = mysql_query($sql);
$row = mysql_fetch_row($result);
$amount = $row['amount'];
// 记算总共有多少页
if( $amount ){
  if( $amount < $page_size ){ $page_count = 1; }

  //如果总数据量小于$PageSize,那么只有一页
  if( $amount % $page_size ){

 //取总数据量除以每页数的余数

  $page_count = (int)($amount / $page_size) + 1;

  //如果有余数,则页数等于总数据量除以每页数的结果取整再加一
  }else{

  $page_count = $amount / $page_size;

 //如果没有余数,则页数等于总数据量除以每页数的结果
  }
}
else{
  $page_count = 0;
}
// 翻页链接
$page_string = '';
if( $page == 1 ){
  $page_string .= '第一页|上一页|';
}
else{
  $page_string .= '<a href=?page=1>第一页</a>|<a href=?page='.($page-1).'>上一页</a>|';
}
if( ($page == $page_count) || ($page_count == 0) ){
  $page_string .= '下一页|尾页';
}
else{
  $page_string .= '<a href=?page='.($page+1).'>下一页</a>|<a href=?page='.$page_count.'>尾页</a>';
}
// 获取数据,以二维数组格式返回结果
if( $amount ){
  $sql = "select * from table order by id desc limit ". ($page-1)*$page_size .", $page_size";
  $result = mysql_query($sql);
 
  while ( $row = mysql_fetch_row($result) ){

  $rowset[] = $row;
  }
}else{
  $rowset = array();
}
// 没有包含显示结果的代码,那不在讨论范围,只要用foreach就可以很简单的用得到的二维数组来显示结果
?>

4、OO风格代码

以下代码中的数据库连接是使用的pear db类进行处理
<?php
// FileName: Pager.class.php
// 分页类,这个类仅仅用于处理数据结构,不负责处理显示的工作
Class Pager
{
  var $PageSize;

//每页的数量
  var $CurrentPageID;

//当前的页数
  var $NextPageID;

  //下一页
  var $PreviousPageID;

  //上一页
  var $numPages;

//总页数
  var $numItems;

//总记录数
  var $isFirstPage;

 //是否第一页
  var $isLastPage;

  //是否最后一页
  var $sql;

 //sql查询语句
 
 function Pager($option)
  {

  global $db;

  $this->_setOptions($option);

  // 总条数

  if ( !isset($this->numItems) )

  {

  $res = $db->query($this->sql);

  $this->numItems = $res->numRows();

  }

  // 总页数

  if ( $this->numItems > 0 )

  {

  if ( $this->numItems < $this->PageSize ){ $this->numPages = 1; }

  if ( $this->numItems % $this->PageSize )

  {

  $this->numPages= (int)($this->numItems / $this->PageSize) + 1;

  }

  else

  {

  $this->numPages = $this->numItems / $this->PageSize;

  }

  }

  else

  {

  $this->numPages = 0;

  }

 

  switch ( $this->CurrentPageID )

  {

  case $this->numPages == 1:

  $this->isFirstPage = true;

  $this->isLastPage = true;

  break;

  case 1:

  $this->isFirstPage = true;

  $this->isLastPage = false;

  break;

  case $this->numPages:

  $this->isFirstPage = false;

  $this->isLastPage = true;

  break;

  default:

  $this->isFirstPage = false;

  $this->isLastPage = false;

  }

 

  if ( $this->numPages > 1 )

  {

  if ( !$this->isLastPage ) { $this->NextPageID = $this->CurrentPageID + 1; }

  if ( !$this->isFirstPage ) { $this->PreviousPageID = $this->CurrentPageID - 1; }

  }

 

  return true;
  }
 
  /***

*

* 返回结果集的数据库连接

* 在结果集比较大的时候可以直接使用这个方法获得数据库连接,然后在类之外遍历,这样开销较小

* 如果结果集不是很大,可以直接使用getPageData的方式获取二维数组格式的结果

* getPageData方法也是调用本方法来获取结果的

*

***/
 
  function getDataLink()
  {

  if ( $this->numItems )

  {

  global $db;

 

  $PageID = $this->CurrentPageID;

 

  $from = ($PageID - 1)*$this->PageSize;

  $count = $this->PageSize;

  $link = $db->limitQuery($this->sql, $from, $count);  //使用Pear DB::limitQuery方法保证数据库兼容性

 

  return $link;

  }

  else

  {

  return false;

  }
  }
 
  /***

*

* 以二维数组的格式返回结果集

*

***/
 
  function getPageData()
  {

  if ( $this->numItems )

  {

  if ( $res = $this->getDataLink() )

  {

 

  if ( $res->numRows() )

  {

  while ( $row = $res->fetchRow() )

  {

  $result[] = $row;

  }

  }

  else

  {

  $result = array();

  }

 

  return $result;

  }

  else

  {

  return false;

  }

  }

  else

  {

  return false;

  }
  }
 
  function _setOptions($option)
  {

  $allow_options = array(

  'PageSize',

  'CurrentPageID',

  'sql',

  'numItems'

  );

 

  foreach ( $option as $key => $value )

  {

  if ( in_array($key, $allow_options) && ($value != null) )

  {

  $this->$key = $value;

  }

  }

 

  return true;
  }
}
?>
<?php
// FileName: test_pager.php
// 这是一段简单的示例代码,前边省略了使用pear db类建立数据库连接的代码
require "Pager.class.php";
if ( isset($_GET['page']) )
{
  $page = (int)$_GET['page'];
}
else
{
  $page = 1;
}
$sql = "select * from table order by id";
$pager_option = array(

  "sql" => $sql,

  "PageSize" => 10,

  "CurrentPageID" => $page
);
if ( isset($_GET['numItems']) )
{
  $pager_option['numItems'] = (int)$_GET['numItems'];
}
$pager = @new Pager($pager_option);
$data = $pager->getPageData();
if ( $pager->isFirstPage )
{
  $turnover = "首页|上一页|";
}
else
{
  $turnover = "<a href='?page=1&numItems=".$pager->numItems."'>首页</a>|<a href='?page=".$pager->PreviousPageID."&numItems=".$pager->numItems."'>上一页</a>|";
}
if ( $pager->isLastPage )
{
  $turnover .= "下一页|尾页";
}
else
{
  $turnover .= "<a href='?page=".$pager->NextPageID."&numItems=".$pager->numItems."'>下一页</a>|<a href='?page=".$pager->numPages."&numItems=".$pager->numItems."'>尾页</a>";
}
?>

需要说明的地方有两个:

这个类仅仅处理数据,并不负责处理显示,因为我觉得将数据的处理和结果的显示都放到一个类里边实在是有些勉强。显示的时候情况和要求多变,不如自己根据类给出的结果处理,更好的方法是根据这个Pager类继承一个自己的子类来显示不同的分页,比如显示用户分页列表可以:
<?php
Class MemberPager extends Pager
{
  function showMemberList()
  {

  global $db;

 

  $data = $this->getPageData();
// 显示结果的代码

  // ......
  }
}
/// 调用
if ( isset($_GET['page']) )
{
  $page = (int)$_GET['page'];
}
else
{
  $page = 1;
}
$sql = "select * from members order by id";
$pager_option = array(

  "sql" => $sql,

  "PageSize" => 10,

  "CurrentPageID" => $page
);
if ( isset($_GET['numItems']) )
{
  $pager_option['numItems'] = (int)$_GET['numItems'];
}
$pager = @new MemberPager($pager_option);
$pager->showMemberList();
?>

第二个需要说明的地方就是不同数据库的兼容性,在不同的数据库里截获一段结果的写法是不一样的。
mysql: select * from table limit offset, rows
pgsql: select * from table limit m offset n
......

所以要在类里边获取结果的时候需要使用pear db类的limitQuery方法。

ok,写完收功,希望花时间看完这些文字的你不觉得是浪费了时间。

PHP 相关文章推荐
在WIN98下以apache模块方式安装php
Oct 09 PHP
PHP 变量类型的强制转换
Oct 23 PHP
php in_array 函数使用说明与in_array需要注意的地方说明
Apr 13 PHP
CI框架中zip类应用示例
Jun 17 PHP
搭建基于Docker的PHP开发环境的详细教程
Jul 01 PHP
非常全面的php日期时间运算汇总
Nov 04 PHP
如何使用微信公众平台开发模式实现多客服
Jan 06 PHP
Symfony2安装的方法(2种方法)
Feb 04 PHP
php实现和c#一致的DES加密解密实例
Jul 24 PHP
ThinkPHP框架使用redirect实现页面重定向的方法实例分析
Apr 12 PHP
PHP使用标准库spl实现的观察者模式示例
Aug 04 PHP
PHP设计模式之观察者模式定义与用法分析
Apr 04 PHP
php在字符串中查找另一个字符串
Nov 19 #PHP
IIS php环境配置PHP5 MySQL5 ZendOptimizer phpmyadmin安装与配置
Nov 18 #PHP
Linux下 php5 MySQL5 Apache2 phpMyAdmin ZendOptimizer安装与配置[图文]
Nov 18 #PHP
php 遍历显示文件夹下所有目录、所有文件的函数,没有分页的代码
Nov 14 #PHP
利用discuz实现PHP大文件上传应用实例代码
Nov 14 #PHP
php下载远程文件类(支持断点续传)
Nov 14 #PHP
PHP ajax 分页类代码
Nov 13 #PHP
You might like
PHP date函数参数详解
2006/11/27 PHP
生成ubuntu自动切换壁纸xml文件的php代码
2010/07/17 PHP
php下拉选项的批量操作的实现代码
2013/10/14 PHP
PHP中SQL查询语句的id=%d解释(推荐)
2016/12/10 PHP
TP框架实现上传一张图片和批量上传图片的方法分析
2020/04/23 PHP
javascript获取当前日期时间及其它操作函数
2011/01/11 Javascript
JS对select控件option选项的增删改查示例代码
2013/10/21 Javascript
js简单的表格添加行和删除行操作示例
2014/03/31 Javascript
Javascript实现简单二级下拉菜单实例
2014/06/15 Javascript
js实现select跳转功能代码
2014/10/22 Javascript
jquery实现标签支持图文排列带上下箭头按钮的选项卡
2015/03/14 Javascript
js实现星星打分效果的方法
2020/07/05 Javascript
JavaScript中清空数组的三种方式
2017/03/22 Javascript
jQuery中.attr()和.data()的区别分析
2017/09/03 jQuery
Vue.js 使用v-cloak后仍显示变量的解决方法
2018/11/19 Javascript
Numpy中的mask的使用
2018/07/21 Python
Python中实例化class的执行顺序示例详解
2018/10/14 Python
利用Python将文本中的中英文分离方法
2018/10/31 Python
pytorch中如何使用DataLoader对数据集进行批处理的方法
2019/08/06 Python
From CSV to SQLite3 by python 导入csv到sqlite实例
2020/02/14 Python
Pandas时间序列:时期(period)及其算术运算详解
2020/02/25 Python
windows下的pycharm安装及其设置中文菜单
2020/04/23 Python
Python爬虫实现自动登录、签到功能的代码
2020/08/20 Python
Python 整行读取文本方法并去掉readlines换行\n操作
2020/09/03 Python
python里glob模块知识点总结
2021/01/05 Python
jupyter 添加不同内核的操作
2021/02/06 Python
Farfetch台湾官网:奢侈品牌时尚购物平台
2019/06/17 全球购物
什么是lambda函数
2013/09/17 面试题
毕业生应聘幼儿园的自荐信
2013/11/20 职场文书
优秀驾驶员先进事迹材料
2014/05/04 职场文书
优秀党务工作者事迹材料
2014/05/07 职场文书
事业单位考核材料
2014/05/21 职场文书
人力资源管理专业自荐信
2014/06/24 职场文书
汽车服务工程专业自荐信
2014/09/02 职场文书
Python实现排序方法常见的四种
2021/07/15 Python
Django对接elasticsearch实现全文检索的示例代码
2021/08/02 Python