亚洲精品中文字幕无乱码_久久亚洲精品无码AV大片_最新国产免费Av网址_国产精品3级片

PHP

PHP中讀取大文件實(shí)現(xiàn)方法

時(shí)間:2024-07-22 19:15:44 PHP 我要投稿
  • 相關(guān)推薦

PHP中讀取大文件實(shí)現(xiàn)方法

  php中讀取大文件怎么實(shí)現(xiàn)?為方便同學(xué)們學(xué)習(xí)PHP語(yǔ)言,yjbys小編為大家分享PHP大文件讀取方法詳解吧,有需要了解的同學(xué)可進(jìn)入?yún)⒖紖⒖肌?/p>

  需求如下: 現(xiàn)有一個(gè)1G左右的日志文件,大約有500多萬(wàn)行, 用php返回最后幾行的內(nèi)容。

  實(shí)現(xiàn)方法:

  1.直接采用file函數(shù)來(lái)操作

  注: 由于 file函數(shù)是一次性將所有內(nèi)容讀入內(nèi)存,而php為了防止一些寫(xiě)的比較糟糕的程序占用太多的內(nèi)存而導(dǎo)致系統(tǒng)內(nèi)存不足,使服務(wù)器出現(xiàn)宕機(jī),所以默認(rèn)情況下 限制只能最大使用內(nèi)存16M,這是通過(guò)php.ini里的memory_limit = 16M來(lái)進(jìn)行設(shè)置,這個(gè)值如果設(shè)置-1,則內(nèi)存使用量不受限制.

  下面是一段用file來(lái)取出這具文件最后一行的代碼.

  整個(gè)代碼執(zhí)行完成耗時(shí) 116.9613 (s).

  代碼如下

  ini_set('memory_limit','-1');

  $file = 'access.log';

  $data = file($file);

  $line = $data[count($data)-1];

  echo $line;

  我機(jī)器是2個(gè)G的內(nèi)存,當(dāng)按下F5運(yùn)行時(shí),系統(tǒng)直接變灰,差不多20分鐘后才恢復(fù)過(guò)來(lái),可見(jiàn)將這么大的文件全部直接讀入內(nèi)存,后果是多少嚴(yán)重,所以不在萬(wàn)不得以,memory_limit這東西不能調(diào)得太高,否則只有打電話給機(jī)房,讓reset機(jī)器了.

  2.直接調(diào)用linux的tail命令來(lái)顯示最后幾行

  在linux命令行下,可以直接使用tail -n 10 access.log很輕易的顯示日志文件最后幾行,可以直接用php來(lái)調(diào)用tail命令,執(zhí)行php代碼如下.

  整個(gè)代碼執(zhí)行完成耗時(shí) 0.0034 (s)

  代碼如下

  file = 'access.log';

  $file = escapeshellarg($file); // 對(duì)命令行參數(shù)進(jìn)行安全轉(zhuǎn)義

  $line = `tail -n 1 $file`;

  echo $line;

  3. 直接使用php的fseek來(lái)進(jìn)行文件操作

  這種方式是最為普遍的方式,它不需要將文件的內(nèi)容全部讀入內(nèi)存,而是直接通過(guò)指針來(lái)操作,所以效率是相當(dāng)高效的.在使用fseek來(lái)對(duì)文件進(jìn)行操作時(shí),也有多種不同的方法,效率可能也是略有差別的,下面是常用的兩種方法.

  方法一:

  首先通過(guò)fseek找到文件的最后一位EOF,然后找最后一行的起始位置,取這一行的數(shù)據(jù),再找次一行的起始位置,再取這一行的位置,依次類推,直到找到了$num行。

  整個(gè)代碼執(zhí)行完成耗時(shí) 0.0095 (s)

  代碼如下

  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的方式從文件最后開(kāi)始讀,但這時(shí)不是一位一位的讀,而是一塊一塊的讀,每讀一塊數(shù)據(jù)時(shí),就將讀取后的數(shù)據(jù)放在一個(gè)buf里,然后通過(guò)換行符(n)的個(gè)數(shù)來(lái)判斷是否已經(jīng)讀完最后$num行數(shù)據(jù).

  整個(gè)代碼執(zhí)行完成耗時(shí) 0.0009(s).

  代碼如下

  $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

  方法三:

  整個(gè)代碼執(zhí)行完成耗時(shí) 0.0003(s)

  代碼如下

  $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 > $chunk) ? $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;

【PHP中讀取大文件實(shí)現(xiàn)方法】相關(guān)文章:

php頁(yè)面緩存實(shí)現(xiàn)方法08-29

PHP中多態(tài)如何實(shí)現(xiàn)呢10-29

PHP中命名空間的方法09-26

Javascript到PHP加密通訊的簡(jiǎn)單實(shí)現(xiàn)方法09-04

php中的構(gòu)造方法與析構(gòu)方法08-29

關(guān)于php堆排序?qū)崿F(xiàn)原理與應(yīng)用方法06-26

Java中g(shù)oto實(shí)現(xiàn)方法06-22

在VBScript中實(shí)現(xiàn)函數(shù)的方法09-23

php中Get方法跟Post方法的區(qū)別09-12

在Word中實(shí)現(xiàn)單鍵粘貼的方法08-12