Perl 文件句柄 - 覆盖现有数据而不是追加/删除

rdb*_*ni0 5 perl file-handling

cat -E test1.txt
Run Code Online (Sandbox Code Playgroud)

输出:

car$
$
$
Run Code Online (Sandbox Code Playgroud)

我只想用“自行车”更改“汽车”并删除新/空行。

这是按预期工作的:

#!/usr/bin/perl -w
open(FILE1,"<","./test1.txt"); @araj=<FILE1>; close(FILE1);
open(FILE2,">","./test1.txt");
map {
s@car@bike@; s@^\n@@;
} @araj;
print(FILE2 @araj);
close(FILE2);
Run Code Online (Sandbox Code Playgroud)

cat -E test1.txt
Run Code Online (Sandbox Code Playgroud)

输出对我来说是 100% 正确的:

bike$
Run Code Online (Sandbox Code Playgroud)

但在上述情况下,我使用的是 2x 打开/关闭文件。所以我使用 2x 文件句柄。
我只想使用1x文件句柄
(这是为了学习目的,只是想了解 +> +>> >> 是如何工作的......)。
例如:

#!/usr/bin/perl -w
open(FILE2,"+<","./test1.txt"); #what file handle should be here? +> , +>> >> .... ?
@araj=<FILE2>;
map {
s@car@bike@; s@^\n@@;
} @araj;
print(FILE2 @araj);
close(FILE2);
Run Code Online (Sandbox Code Playgroud)

输出不正确:

car$
$
$
bike$
Run Code Online (Sandbox Code Playgroud)

为什么这是追加,但没有覆盖?当我使用其他文件句柄时,结果也不正确,例如空文件...哪个文件句柄用于读取和覆盖?

Ste*_*ich 5

为什么这是追加,但没有覆盖?

您首先读取了所有数据,直到文件末尾。这意味着下一次读取或写入的文件位置现在在您读取的所有数据之后,即在文件末尾。如果要从文件开头写入数据,则需要使用seek更改文件位置:

 seek($filehandle,0,0); # position at beginning of file
Run Code Online (Sandbox Code Playgroud)

您写入的下一个数据将从这个新文件位置开始写入,即从文件的开头开始。完成后,您可能需要使用truncatetell获得的当前文件位置,从文件中删除当前文件位置之后的任何数据:

 truncate($filehandle, tell($filehandle));
Run Code Online (Sandbox Code Playgroud)

或者,整个程序:

use strict;
use warnings;
open(my $fh, "+<", "./test1.txt");
my @araj = <$fh>;
for(@araj) {
    s{car}{bike};
    s{^\n}{};
}
seek($fh, 0, 0);           # seek at the beginning of the file
print $fh @araj;
truncate($fh, tell($fh));  # remove everything after current file position
close($fh);
Run Code Online (Sandbox Code Playgroud)