Php&mysql大型网站的分页脚本



我有一个超过300万行的大数据库,我需要一个php;mysql分页脚本。我已经有一个,但我通过1000页的水平后,它的工作非常慢,就像几分钟的加载。欢迎任何可以帮助我的脚本/建议。

编辑,我正在使用这个脚本

<?php
// Script de paginare, de la http://www.marplo.net
// Datele pt. conectare la baza de date
// MODIFICATI
$host = "localhost";    // server MySQL
$utilizator = "root"; 
$parola = "parola";
$numebd = "nume_bd";    // nume baza de date
// Conectarea la baza de date
$conn = mysql_connect($host, $utilizator, $parola);
if (!$conn) {
  echo 'Conectare nereusita la MySQL'; 
  exit;
}
// Selectarea bazei de date
if (!mysql_select_db($numebd, $conn)) {
  echo 'Baza de date nu a putut fi selectata deoarece : '. mysql_error();
  exit;
}
// Setarea pentru format UTF-8
$sql = "SET NAMES 'utf8'";
mysql_query($sql, $conn);
// Afla cate linii sunt in tabel (MODIFICATI 'nume_tb') din baza de date  
$sql = "SELECT COUNT(*) FROM `nume_tb`";  
$result = mysql_query($sql, $conn) or trigger_error(E_USER_ERROR);  
$r = mysql_fetch_row($result);  
$numrows = $r[0];  
// Stabileste numarul de linii din tabel afisate in pagina 
$rowsperpage = 10;  
// afla numarul total necesar de pagini 
$totalpages = ceil($numrows / $rowsperpage);        // ceil face rotunjire la int. maxim
// Obtine pagina curenta sau seteaza default 
if (isset($_GET['currentpage']) && is_numeric($_GET['currentpage'])) {  
  // seteaza variabila ca int 
  $currentpage = (int) $_GET['currentpage'];  
} else {  
  // pagina care este initial afisata (pagina default) 
  $currentpage = 1;  
}
// daca pagina curenta e mai mare decat total pagini...
if ($currentpage > $totalpages) {  
  // seteaza pagina curenta la ultima pagina  
  $currentpage = $totalpages;  
} 
// daca pagina curenta e mai mica decat prima pagina...  
if ($currentpage < 1) {  
  // seteaza pagina curenta la prima pagina   
  $currentpage = 1;  
} 
// lista cu pagini, in functie de pagina curenta   
$offset = ($currentpage - 1) * $rowsperpage;  
// obtine datele din tabel (MODIFICATI 'nume_tb') din baza de date  
$sql = "SELECT * FROM `nume_tb` LIMIT $offset, $rowsperpage";  
$result = mysql_query($sql, $conn) or trigger_error(E_USER_ERROR);  
// parcurgerea matricei cu datele obtinute 
while ($list = mysql_fetch_assoc($result)) {  
  // - MODIFICATI numele coloanelor tabelului ('id' si 'texte')
  // Stocheaza datele returnate de MySQL in variabile array pt. fiecare coloana
  $id[] = $list['id'];
  $text[] = $list['texte'];
}
mysql_close();  // Incheie conexiunea cu mysql
/*** Afisarea datelor obtinute ***/
// Parcurge variabilele array setate in bucla WHILE
for($i=0; $i<count($id); $i++) {
  // Aici puteti adauga cod HTML pentru aspectul grafic al afisarii
  echo $id[$i]. " - ". $text[$i]. "<br />";
}
/*** Construirea link-urilor pt. paginare ***/ 
// raza nr. link-uri din jurul celui curent 
$range = 3;
// Link-uri inapoi, daca pagina curenta nu e prima
if ($currentpage > 1) {  
  // arata << pt. link la prima pagina  
  echo " <a href='{$_SERVER['PHP_SELF']}?currentpage=1'>&lt;&lt;</a> &nbsp; ";  
  // obtine nr. pagina din urma 
  $prevpage = $currentpage - 1;  
  // arata < pt. link la o pagina in urma 
  echo " <a href='{$_SERVER['PHP_SELF']}?currentpage=$prevpage'>&lt;</a> &nbsp;";  
} 
// definirea link-urilor din raza paginii curente
for ($x = ($currentpage - $range); $x < (($currentpage + $range) + 1); $x++) {  
  // daca e un nr. de pagina valid ... 
  if (($x > 0) && ($x <= $totalpages)) {  
     // daca nr. e pagina curenta ...  
     if ($x == $currentpage) {  
        // afiseaza nr. pagina fara a fi link  
        echo " [<b>$x</b>] ";  
     // daca nr. nu e pagina curenta ...  
     } else {  
        // il face link  
    echo " <a href='{$_SERVER['PHP_SELF']}?currentpage=$x'>$x</a> ";  
     }  
  }
}
// Daca pagina curenta nu e ultima, afiseaza link inainte si spre ultima pagina
if ($currentpage != $totalpages) {  
  // obtine pagina urmatoare 
  $nextpage = $currentpage + 1;  
   // arata > pt. urmatoarea pagina   
  echo "&nbsp; <a href='{$_SERVER['PHP_SELF']}?currentpage=$nextpage'>&gt;</a> ";  
  //  arata >> pt. ultima pagina
  echo " &nbsp; <a href='{$_SERVER['PHP_SELF']}?currentpage=$totalpages'>&gt;&gt;</a> ";  
}
?>

请在下次发布新问题之前使用搜索功能和相关问题小工具

大极限问题:

如果您需要前几行,使用索引排序是有效的,即使发生了一些额外的过滤,因此您需要按索引扫描比LIMIT所要求的更多的行。但是,如果您处理的是带有较大偏移量的LIMIT查询,则效率将受到影响。LIMIT 1000,10可能比LIMIT 0,10慢得多。的确,大多数用户不会浏览超过10页的搜索结果,但搜索引擎机器人很可能会这样做。我见过机器人在我的项目中查看200多个页面。另外,对于许多网站来说,如果没有注意到这一点,就很容易发起DOS攻击——从很少的连接中请求大量的页面,这就足够了。如果你不做任何其他事情,确保你阻止请求过大的页码。

对于某些情况,例如,如果结果是静态的,那么预先计算结果以便您可以查询它们的位置可能是有意义的。因此,代替查询限制1000,10,你将有1000和1009之间的WHERE位置有相同的效率,任何位置(只要它被索引)

ORDER BY…LIMIT Performance Optimization

相关问题:

由于您正在使用MySQL,您可以利用LIMIT命令:

$start = ($current_page - 1) * $rows_per_page; //Entry 1 on page 1 is index 0
$select = "SELECT * FROM `your_table` LIMIT $start, $rows_per_page"

一些小事:

  • 您似乎只使用'id'和'text '从每一行。在这种情况下,如果只有两列,那么使用SELECT *是可以的,但是如果有其他列,那就浪费了。使用SELECT id,texte代替,如果是这种情况。

  • 您可以通过执行while (list(id[],text[]) = mysql_fetch_row());来摆脱中间变量$list(如果您的行将id和文本作为前两列,或者根据我前面的要点更改SELECT)。类似地,对于检索行计数:list($numrows) = mysql_fetch_row($result);可以工作。

  • 你可以从你所有的href中删除{$_SERVER['PHP_SELF']}。"?Currentpage =6"是一个完全有效的(相对的)href。对于较短的html,考虑使用'p'代替'currentpage'作为查询字符串变量。


现在有几个主要的:


(1)速度问题绝对是MySQL的问题,从我可以看到-没有大的循环在你的代码。

Try (a)让'id'成为主键(如果它还不是);或者如果你已经有另一个主键列,就把它作为索引),(b)在select语句中添加ORDER BY id子句来获取数据。这肯定有帮助。

如果没有,作为最后的手段您可以创建第二个表来索引您的行:

CREATE TABLE nume_tb_idx (
  idx int NOT NULL PRIMARY KEY,
  id int NOT NULL FOREIGN KEY REFERENCES nume_tb (id) ON UPDATE CASCADE ON DELETE CASCADE
);

(假设您的id列是int)。

当向nume_tb添加一行时,必须向该表添加一个条目,并且在从nume_tb删除一行后,重新编号后面的条目(您可以为此使用触发器)。其思想是'idx'中的值总是从1开始的连续数字,直到nume_tb中的总行数。然后,您将使用以下SELECT语句而不是使用LIMIT:

"SELECT id,texte FROM nume_tb_idx INNER JOIN nume_tb USING (id)
WHERE idx BETWEEN $offset AND ".$offset+$rowsperpage-1." ORDER BY idx"

这应该总是很快的。


(2)你做页面导航链接的方式是相当无用的。对于像您这样的大量页面,理想的解决方案是"对数"页面导航。有关描述和PHP示例代码,请参阅我对这个问题的回答:

如何为很多很多的页面做导航?对数页面导航

最新更新