ファイル全体をロードせずに、潜在的に巨大なテキスト ファイルの最後の行を読み取る必要があります。かなり良い解決策が得られたと思いますが、紛らわしいバグがあります。ほとんどのエントリでは、関数は適切に戻りますが、ある特定の状況では、ファイルの最後の文字が 2 倍になるようです。これが私が作成したテスト スクリプトです (news.txt の内容は "NEWS" という 1 つの単語です)。
<?php
echo "file contains: ".file_get_contents('news.txt')."<br/>";
$lastLine = getLastLine("news.txt");
echo "function returns: ".$lastLine;
function getLastLine($filename){
$f = fopen($filename, 'r');
$cursor = -1;
$line = '';
fseek($f, $cursor, SEEK_END);
$char = fgetc($f);
echo "initial read: <br/>"; // DEBUG
echo $cursor." | ".$char." | ".$line."<br/>"; // DEBUG
// test for empty file
if($char === false)
return false;
// Trim trailing newline chars of the file
while ($char === "\n" || $char === "\r" || $char === " ") {
fseek($f, $cursor--, SEEK_END);
$char = fgetc($f);
}
echo "kill new lines: <br/>"; // DEBUG
echo $cursor." | ".$char." | ".$line."<br/>"; // DEBUG
// Read until the start of file or first newline char
echo "running loop: <br/>"; // DEBUG
while ($char !== false && $char !== "\n" && $char !== "\r" ) {
// Prepend the new char
$line = $char . $line;
echo $cursor." | ".$char." | ".$line."<br/>"; // DEBUG
fseek($f, $cursor--, SEEK_END);
$char = fgetc($f);
}
return trim($line);
}
?>
出力は次のようになります。
file contains: NEWS
initial read:
-1 | S |
kill new lines:
-1 | S |
running loop:
-1 | S | S
-2 | S | SS
-3 | W | WSS
-4 | E | EWSS
-5 | N | NEWSS
function returns: NEWSS
インデックス -1 と -2 は同一のように見えますが、file_get_contents は、ファイルの内容が単なる "NEWS" であることを確認しています。
私は何を間違っていますか?