PHP Several quick read large file examples thanks for breaking me down. Delivery time: 2014-10-16 Source: Triple
In PHP, the quickest way to read a file is to use functions such as file, file_get_contents, and just a few lines of code to do what we need. But when the file being manipulated is a large file, these functions may be out of the way, the following will start from a requirement to illustrate the common method of operation when reading large files.
Demand
There is a 800M log file, approximately more than 5 million lines, with PHP returning the last few lines of content.
Implementation method
1. Use the file function directly to manipulate
Because the file function reads everything into memory at once, and PHP prevents some poorly written programs from taking up too much memory and causing the server to go down, the default limit is to use only 16M of memory, which is PHP.ini in memory_limit = 16M to set, this value if set-1, the memory makes the usage unrestricted.
Here is a piece of code that uses file to fetch the last line of the document:
The code is as follows |
|
<?php Ini_set (' Memory_limit ', '-1 '); $file = ' Access.log '; $data = file ($file); $line = $data[count ($data)-1]; Echo $line; ?> |
The entire code execution takes time 116.9613 (s).
My machine is 2 g of memory, when the press F5 run, the system directly gray, almost 20 minutes after the recovery, it can be seen so large files directly into memory, the consequences is how much serious, so not million can not, memory_limit this thing can not be adjusted too high, otherwise only call room, Let the reset machine out.
2. Directly call the Linux Tail command to display the last few lines
Under the Linux command line, you can directly use Tail-n access.log very easy to display the last few lines of the log file, you can directly use PHP to invoke the tail command, execute PHP code as follows:
The code is as follows |
|
<?php $file = ' Access.log '; $file = Escapeshellarg ($file); To safely escape command-line arguments $line = ' tail-n 1 $file '; Echo $line; ?> |
Complete code Execution time 0.0034 (s)
3. Use PHP fseek directly for file operation
This is the most common way, it does not need to read all the contents of the file, but directly through the pointer to operate, so efficiency is quite efficient. There are a number of different ways to use fseek to manipulate files, and the efficiency may also be slight, and here are two common methods:
Method One
First, find the last EOF of the file by fseek, then find the beginning of the last line, take this row of data, then find the starting position of the next line, then take the position of this line, and so on, until the $num line is found.
#实现代码如下
The code is as follows |
|
<? PHP $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 ?> |
Complete code Execution time 0.0095 (s)
Method Two
Or use fseek way from the end of the file to read, but this is not a single read, but a piece of reading, each read a piece of data, the data will be read in a buf, and then by the number of newline characters (\ n) to determine whether the last $num line of data is read.
#实现代码如下
code as follows |
&NBSP; |
<? php $fp = fopen ($file, "R"); $num = 10; $chunk = 4096; $fs = sprintf ("%u", FileSize ($file)); $max = (intval ($fs) = = Php_int_max)? Php_int_max:filesize ($file); for ($len = 0; $len < $max; $len + = $chunk) { $seekSize = ($max -$len > $chun k)? $chunk: $max-$len; fseek ($fp, ($len + $seekSize) *-1, seek_end); $readData = fread ($fp, $seekSize). $readData; if (Substr_count ($readData, "\ n") >= $num + 1) { preg_match ("! *?\n) {". ($num). "}$!", $readData, $match); $data = $match [0]; break; } } Fclose ($fp); echo $data; ? |
The entire code execution takes time 0.0009 (s).
Method Three
code as follows |
&NBSP; |
<? PHP 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+")); ? |
Complete code Execution time 0.0003 (s)
PHP Several quick read large file examples