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 |
感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!