0

ファイル全体をロードせずに、潜在的に巨大なテキスト ファイルの最後の行を読み取る必要があります。かなり良い解決策が得られたと思いますが、紛らわしいバグがあります。ほとんどのエントリでは、関数は適切に戻りますが、ある特定の状況では、ファイルの最後の文字が 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" であることを確認しています。

私は何を間違っていますか?

4

1 に答える 1