1. 소개
페이징 표시는 대량의 데이터를 탐색하고 표시하는 매우 일반적인 방법이며 웹 프로그래밍에서 가장 일반적으로 처리되는 이벤트 중 하나입니다. 웹 프로그래밍 베테랑들에게는 이런 코드를 작성하는 것이 숨 쉬는 것처럼 자연스러운 일이지만, 초보자들에게는 이 문제에 대해 혼동하는 경우가 많기 때문에 이 문제를 자세히 설명하기 위해 특별히 이 글을 썼습니다. 이 기사를 마치면 페이징 표시의 원리와 구현 방법을 어느 정도 이해하게 될 것입니다. 이 글은 초보자가 읽기에 적합하며, 모든 샘플 코드는 PHP로 작성되었습니다.
2. 원리
데이터베이스의 결과 집합이 표시를 위해 인위적으로 세그먼트로 나누어지는 것을 의미합니다. 여기에는 두 가지 초기 매개변수가 필요합니다.
페이지당 레코드 수($PageSize)는 얼마입니까?
현재 페이지($CurrentPageID)는 어떤 페이지인가요?
이제 다른 결과 세트를 제공하기만 하면 특정 결과를 표시할 수 있습니다.
이전 페이지($PreviousPageID), 다음 페이지($NextPageID), 총 페이지 수($numPages) 등과 같은 기타 매개변수는 이전 항목을 기반으로 얻을 수 있습니다.
mysql 데이터베이스를 예로 들어 보겠습니다. 테이블의 특정 콘텐츠를 가로채려면 sql 문을 사용할 수 있습니다. 다음 SQL 문 세트를 살펴보고 규칙을 찾아보세요.
처음 10개 레코드: 테이블 제한 0,10에서 * 선택
11~20번째 레코드: 테이블 제한 10,10에서 * 선택
21~30번째 레코드 : select * from tablelimit 20,10
...
이 SQL 문 세트는 실제로 $PageSize=10일 때 테이블의 각 데이터 페이지를 가져오는 SQL 문입니다. 이러한 템플릿을 요약할 수 있습니다.
select * from tablelimit($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: 테이블 제한 오프셋, 행에서 * 선택
pgsql: 테이블 제한 m 오프셋 n에서 * 선택
......
따라서 클래스에서 결과를 얻으려면 pear db 클래스의limitQuery 메소드를 사용해야 합니다.
좋아요, 이 글을 읽으시면 시간 낭비라고 생각하지 않으셨으면 좋겠습니다.
위에서는 PHP 페이징 내용을 포함하여 PHP 페이징 디스플레이 제작에 대한 자세한 설명과 함께 PHP 튜토리얼에 관심이 있는 친구들에게 도움이 되기를 바랍니다.
관련 기사: