我有一个超过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'><<</a> ";
// obtine nr. pagina din urma
$prevpage = $currentpage - 1;
// arata < pt. link la o pagina in urma
echo " <a href='{$_SERVER['PHP_SELF']}?currentpage=$prevpage'><</a> ";
}
// 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 " <a href='{$_SERVER['PHP_SELF']}?currentpage=$nextpage'>></a> ";
// arata >> pt. ultima pagina
echo " <a href='{$_SERVER['PHP_SELF']}?currentpage=$totalpages'>>></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示例代码,请参阅我对这个问题的回答:
如何为很多很多的页面做导航?对数页面导航