fgets

(PHP 3, PHP 4, PHP 5)

fgets -- 从文件指针中读取一行

说明

string fgets ( int handle [, int length] )

handle 指向的文件中读取一行并返回长度最多为 length - 1 字节的字符串。碰到换行符(包括在返回值中)、EOF 或者已经读取了 length - 1 字节后停止(看先碰到那一种情况)。如果没有指定 length,则默认为 1K,或者说 1024 字节。

出错时返回 FALSE

通常的缺陷:

习惯了 C 语言中 fgets() 语法的人应该注意到 EOF 是怎样被返回的。

文件指针必须是有效的,并且必须指向一个由 fopen()fsockopen() 成功打开的文件。

以下是一个简单例子:

例子 1. 逐行读取文件

<?php
$handle
= fopen("/tmp/inputfile.txt", "r");
while (!
feof($handle)) {
    
$buffer = fgets($fd, 4096);
    echo
$buffer;
}
fclose($handle);
?>

注: length 参数从 PHP 4.2.0 起成为可选项,如果忽略,则行的长度被假定为 1024。从 PHP 4.3 开始,忽略掉 length 将继续从流中读取数据直到行结束。如果文件中的大多数行都大于 8KB,则在脚本中指定最大行的长度在利用资源上更为有效。

注: 从 PHP 4.3 开始本函数可以安全用于二进制文件。早期的版本则不行。

注: 如果碰到 PHP 在读取文件时不能识别 Macintosh 文件的行结束符,可以激活 auto_detect_line_endings 运行时配置选项。

参见 fread()fgetc()stream_get_line()fopen()popen()fsockopen()stream_set_timeout()


add a note add a note User Contributed Notes
d at foo.com
13-Aug-2006 04:03
For sockets, If you dont want fgets, fgetc etc... to block if theres no data there. set socket_set_blocking(handle,false); and socket_set_blocking(handle,true); to set it back again.
svayn at yahoo dot com
15-Jul-2006 05:21
fgets is SLOW for scanning through large files. If you don't have PHP 5, use fscanf($file, "%s\n") instead.
sam dot bryan at montal dot com
23-May-2006 05:09
An easy way to authenticate Windows Domain users from scripts running on a non-Windows or non-Domain box - pass the submitted username and password to an IMAP service on a Windows machine.

<?php
$server
= 'imapserver';
$user = 'user';
$pass = 'pass';

if (
authIMAP($user, $pass, $server)) {
   echo
"yay";
} else {
   echo
"nay";
}

function
authIMAP($user, $pass, $server) {
  
$connection = fsockopen($server, 143, $errno, $errstr, 30);

   if(!
$connection) return false;

  
$output = fgets($connection, 128); // banner
  
fputs($connection, "1 login $user $pass\r\n");
  
$output = fgets($connection, 128);
  
fputs($connection, "2 logout\r\n");
  
fclose($connection);

   if (
substr($output, 0, 4) == '1 OK') return true;

   return
false;
}
?>
25-Mar-2006 01:36
Macintosh line endings mentioned in docs refer to Mac OS Classic. You don't need this setting for interoperability with unixish OS X.
tavernadelleidee[italy]
09-Mar-2006 07:44
I think that the quickest way of read a (long) file with the rows in  reverse order is

<?php
$myfile
= 'myfile.txt';
$command = "tac $myfile > /tmp/myfilereversed.txt";
passthru($command);
$ic = 0;
$ic_max = 100// stops after this number of rows
$handle = fopen("/tmp/myfilereversed.txt", "r");
while (!
feof($handle) && ++$ic<=$ic_max) {
  
$buffer = fgets($handle, 4096);
   echo
$buffer."<br>";
}
fclose($handle);
?>

It echos the rows while it is reading the file so it is good for long files like logs.

Borgonovo
ecvej
05-Jan-2006 05:20
I would have expected the same behaviour from these bits of code:-

<?php

/*This times out correctly*/
while (!feof($fp)) {
   echo
fgets($fp);
}

/*This times out before eof*/
while ($line=fgets($fp)) {
   echo
$line;
}

/*A reasonable fix is to set a long timeout*/
stream_set_timeout($fp, 180);
while (
$line=fgets($fp)) {
   echo
$line;
}
?>
hackajar <matt> yahoo <trot> com
06-Dec-2005 04:17
When working with VERY large files, php tends to fall over sideways and die. 

Here is a neat way to pull chunks out of a file very fast and won't stop in mid line, but rater at end of last known line.  It pulled a 30+ million line 900meg file through in ~ 24 seconds.

NOTE:
$buf just hold current chunk of data to work with.  If you try "$buf .=" (note 'dot' in from of '=') to append $buff, script will come to grinding crawl around 100megs of data, so work with current data then move on!

//File to be opened
$file = "huge.file";
//Open file (DON'T USE a+ pointer will be wrong!)
$fp = fopen($file, 'r');
//Read 16meg chunks
$read = 16777216;
//\n Marker
$part = 0;

while(!feof($fp)) {
   $rbuf = fread($fp, $read);
   for($i=$read;$i > 0 || $n == chr(10);$i--) {
       $n=substr($rbuf, $i, 1);
       if($n == chr(10))break;
       //If we are at the end of the file, just grab the rest and stop loop
       elseif(feof($fp)) {
           $i = $read;
           $buf = substr($rbuf, 0, $i+1);
           break;
       }
   }
   //This is the buffer we want to do stuff with, maybe thow to a function?
   $buf = substr($rbuf, 0, $i+1);
   //Point marker back to last \n point
   $part = ftell($fp)-($read-($i+1));
   fseek($fp, $part);
}
fclose($fp);
kpeters AT-AT monolithss DEE OH TEE com
01-Dec-2005 09:51
It appears that fgets() will return FALSE on EOF (before feof has a chance to read it), so this code will throw an exception:

while (!feof($fh)) {
  $line = fgets($fh);
  if ($line === false) {
   throw new Exception("File read error");
  }
}
dandrews OVER AT 3dohio DOT com
08-Jan-2005 03:11
Saku's example may also be used like this:

<?php
 
@ $pointer = fopen("$DOCUMENT_ROOT/foo.txt", "r"); // the @ suppresses errors so you have to test the pointer for existence
  
if ($pointer) {
     while (!
feof($pointer)) {
        
$preTEXT = fgets($pointer, 999);
        
// $TEXT .= $preTEXT;  this is better for a string
      
$ATEXT[$I] = $preTEXT// maybe better as an array
      
$I++;
     }
    
fclose($pointer);
   }
?>
angelo [at] mandato <dot> com
19-Nov-2004 10:43
Sometimes the strings you want to read from a file are not separated by an end of line character.  the C style getline() function solves this.  Here is my version:
<?php
function getline( $fp, $delim )
{
  
$result = "";
   while( !
feof( $fp ) )
   {
      
$tmp = fgetc( $fp );
       if(
$tmp == $delim )
           return
$result;
      
$result .= $tmp;
   }
   return
$result;
}

// Example:
$fp = fopen("/path/to/file.ext", 'r');
while( !
feof($fp) )
{
  
$str = getline($fp, '|');
  
// Do something with $str
}
fclose($fp);
?>
lelkesa
04-Nov-2004 06:54
Note that - afaik - fgets reads a line until it reaches a line feed (\\n). Carriage returns (\\r) aren't processed as line endings.
However, nl2br insterts a <br /> tag before carriage returns as well.
This is useful (but not nice - I must admit) when you want to store a more lines in one.
<?php
function write_lines($text) {
 
$file = fopen('data.txt', 'a');
 
fwrite($file, str_replace("\n", ' ', $text)."\n");
 
fclose($file);
}

function
read_all() {
 
$file = fopen('data.txt', 'r');
  while (!
feof($file)) {
  
$line = fgets($file);
   echo
'<u>Section</u><p>nl2br'.($line).'</p>';
  }
 
fclose($file);
}
?>

Try it.
06-Sep-2004 06:05
If you need to simulate an un-buffered fgets so that stdin doesnt hang there waiting for some input (i.e. it reads only if there is data available) use this :
<?php

  
function fgets_u($pStdn) {

          
$pArr = array($pStdn);

       if (
false === ($num_changed_streams = stream_select($pArr, $write = NULL, $except = NULL, 0))) {
           print(
"\$ 001 Socket Error : UNABLE TO WATCH STDIN.\n");
           return
FALSE;
       } elseif (
$num_changed_streams > 0) {
               return
trim(fgets($pStdn, 1024));
       }
          
   }

?>
rstefanowski at wi dot ps dot pl
13-Aug-2004 12:03
Take note that fgets() reads 'whole lines'. This means that if a file pointer is in the middle of the line (eg. after fscanf()), fgets() will read the following line, not the remaining part of the currnet line. You could expect it would read until the end of the current line, but it doesn't. It skips to the next full line.
timr
17-Jun-2004 10:13
If you need to read an entire file into a string, use file_get_contents().  fgets() is most useful when you need to process the lines of a file separately.
Saku
05-Jun-2004 08:47
As a beginner I would have liked to see "how to read a file into a string for use later and not only how to directly echo the fgets() result. This is what I derived:
<?php
 
@ $pointer = fopen("$DOCUMENT_ROOT/foo.txt", "r"); // the @ suppresses errors so you have to test the pointer for existence
  
if ($pointer) {
     while (!
feof($pointer)) {
        
$preTEXT = fgets($pointer, 999);
        
$TEXT = $TEXT . $preTEXT;
     }
    
fclose($pointer);
   }
?>
flame
22-May-2004 11:44
fread is binary safe, if you are stuck with a pre 4.3 version of PHP.
Pete
23-Feb-2004 08:35
If you have troubles reading binary data with versions <= 4.3.2 then upgrade to 4.3.3
The binary safe implementation seems to have had bugs which were fixed in 4.3.3