php读取大文件,使用fseek函数是最为普遍的方式,它不需要将文件的内容全部读入内存,而是直接通过指针来操作,所以效率是相当高效的.在使用fseek来对文件进行操作时,也有多种不同的方法,效率可能也是略有差别的,下面是常用的两种方法.
方法一:
首先通过fseek找到文件的最后一位EOF,然后找最后一行的起始位置,取这一行的数据,再找次一行的起始位置,再取这一行的位置,依次类推,直到找到了$num行。实现代码如下:
整个代码执行完成耗时 0.0095 (s)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
|
function tail( $fp , $n , $base =5)
{
assert( $n >0);
$pos = $n +1;
$lines = array ();
while ( count ( $lines )< = $n ){
try {
fseek ( $fp ,- $pos ,SEEK_END);
} catch (Exception $e ){
fseek (0);
break ;
}
$pos *= $base ;
while (! feof ( $fp )){
array_unshift ( $lines , fgets ( $fp ));
}
}
return array_slice ( $lines ,0, $n );
}
var_dump(tail( fopen ( "access.log" , "r+" ),10));
|
方法二 :
还是采用fseek的方式从文件最后开始读,但这时不是一位一位的读,而是一块一块的读,每读一块数据时,就将读取后的数据放在一个buf里,然后通过换行符(n)的个数来判断是否已经读完最后$num行数据.实现代码如下
整个代码执行完成耗时 0.0009(s).
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
|
$fp = fopen ( $file , "r" );
$line = 10;
$pos = -2;
$t = " " ;
$data = "" ;
while ( $line > 0) {
while ( $t != "n" ) {
fseek ( $fp , $pos , SEEK_END);
$t = fgetc ( $fp );
$pos --;
}
$t = " " ;
$data .= fgets ( $fp );
$line --;
}
fclose ( $fp );
echo $data
|
感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!