$file->eof() always returning false when using PHP's SplFileObject in 'r' mode

混江龙づ霸主 提交于 2019-12-23 05:12:20

问题


Why is my PHP script hanging?

$path = tempnam(sys_get_temp_dir(), '').'.txt';
$fileInfo = new \SplFileInfo($path);
$fileObject = $fileInfo->openFile('a');
$fileObject->fwrite("test line\n");
$fileObject2 = $fileInfo->openFile('r');
var_dump(file_exists($path));          // bool(true)
var_dump(file_get_contents($path));    // string(10) "test line
                                       // "
var_dump(iterator_count($fileObject2)); // Hangs on this

If I delete the last line (iterator_count(...) and replace it with this:

$i = 0;
$fileObject2->rewind();
while (!$fileObject2->eof()) {
    var_dump($fileObject2->eof());
    var_dump($i++);
    $fileObject2->next();
}

// Output:
// bool(false)
// int(0)
// bool(false)
// int(1)
// bool(false)
// int(2)
// bool(false)
// int(3)
// bool(false)
// int(4)
// ...

The $fileObject->eof() always returns false so I get an infinite loop.

Why are these things happening? I need to get a line count.


回答1:


Why are these things happening?

You are experiencing a peculiarity in the way that the SplFileObject class is written. Without calling next() and current() methods—using the default (0) flags—the iterator never moves forward.

The iterator_count() function never calls current(); it checks valid() and calls next() only. Your bespoke loops only call one or other of current() and next().

This should be considered a bug (whether in PHP itself, or a failure in the documentation) and the following code should (and does not) work as expected. I invite you to report this misbehaviour.

// NOTE: This currently runs in an infinite loop!
$file = new SplFileObject(__FILE__);
var_dump(iterator_count($file));

Workarounds

One quick sidestep to get things moving is to set the READ_AHEAD flag on the object. This will cause the next() method to read the next available line.

$file->setFlags(SplFileObject::READ_AHEAD);

If, for any reason, you do not want the read ahead behaviour then you must call both next() and current() yourself.

Back to the original problem of two SplFileObjects

The following should now work as you expected, allowing appending to a file and reading its line count.

<?php
$info = new SplFileInfo(__FILE__);
$write = $info->openFile('a');
$write->fwrite("// new line\n");
$read = $info->openFile('r');
$read->setFlags(SplFileObject::READ_AHEAD);
var_dump(iterator_count($read));



回答2:


EDITED 01

If what you need is the number of lines inside the file:

<?php

$path = tempnam(sys_get_temp_dir(), '').'.txt';

$fileInfo = new SplFileInfo($path);
$fileObject = $fileInfo->openFile('a+');

$fileObject->fwrite("Foo".PHP_EOL);
$fileObject->fwrite("Bar".PHP_EOL);

echo count(file($path));  // outputs 2

?>


EDITED 02

This is your code above, but without entering into infinite loops due to the file pointer:

<?php

$path = tempnam(sys_get_temp_dir(), '').'.txt';

$fileInfo = new SplFileInfo($path);
$fileObject = $fileInfo->openFile('a+');

$fileObject->fwrite("Foo".PHP_EOL);
$fileObject->fwrite("Bar");

foreach($fileObject as $line_num => $line) {
    echo 'Line: '.$line_num.' "'.$line.'"'."<br/>";
}
echo 'Total Lines:' . $fileObject->key();

?>

Outputs

Line: 0 "Foo "

Line: 1 "Bar"

Total Lines:2



ORIGINAL ANSWER

The logic applied was a bit off. I simplified the code:

<?php

// set path to tmp with random file name
echo $path = tempnam(sys_get_temp_dir(), '').'.txt';
echo "<br/>";

// new object
$fileInfo = new \SplFileInfo($path);

// open to write
$fileObject = $fileInfo->openFile('a');

// write two lines
$fileObject->fwrite("Foo".PHP_EOL);
$fileObject->fwrite("Bar".PHP_EOL);

// open to read
$fileObject2 = $fileInfo->openFile('r');

// output contents
echo "File Exists: " .file_exists($path);
echo "<br/>";
echo "File Contents: " . file_get_contents($path);
echo "<br/>";

// foreach line get line number and line contents 
foreach($fileObject2 as $line_num => $line) {
  echo 'Line: '.$line_num;
  echo ' With: "'.$line.'" is the end? '.($fileObject2->eof()?'yes':'no')."<br>";
}

?>

Outputs:

/tmp/EAdklY.txt

File Exists: 1

File Contents: Foo Bar

Line: 0 With: "Foo " is the end? no

Line: 1 With: "Bar " is the end? no

Line: 2 With: "" is the end? yes




回答3:


While it may seem counter intuitive, with PHP 5.3.9, this:

<?php
$f = new SplFileObject('test.txt', 'r');
while (!$f->eof()) {
    $f->next();
}

while be an infinite loop, and never exit.

The following will exit, when the end of the file is reached:

<?php
$f = new SplFileObject('test.txt', 'r');
while (!$f->eof()) {
    $f->current();
}

So:

$i = 0;
$fileObject2->rewind();
while (!$fileObject2->eof()) {
    var_dump($fileObject2->eof());
    var_dump($i++);
    $fileObject2->next();
}

should be rewritten as:

$fileObject2->rewind();
while (!$fileObject2->eof()) {
    $fileObject2->current();
}

$i = $fileObject2->key();


来源:https://stackoverflow.com/questions/11706174/file-eof-always-returning-false-when-using-phps-splfileobject-in-r-mode

易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!