How to replace a string in an existing file in Perl
Asked Answered
C

4

91

I want to replace the word "blue" with "red" in all text files named as 1_classification.dat, 2_classification.dat and so on. I want to edit the same file so I tried the following code, but it does not work. Where am I going wrong?

@files = glob("*_classification.dat");
foreach my $file (@files)
{
    open(IN,$file) or die $!;
    <IN>;
    while(<IN>)
    {
        $_ = '~s/blue/red/g';
        print IN $file;
    }
    close(IN)
}
Contracted answered 9/8, 2011 at 10:45 Comment(0)
D
187

Use a one-liner:

$ perl -pi.bak -e 's/blue/red/g' *_classification.dat

Explanation

  • -p processes, then prints <> line by line
  • -i activates in-place editing. Files are backed up using the .bak extension
  • The regex substitution acts on the implicit variable, which are the contents of the file, line-by-line
Delhi answered 9/8, 2011 at 10:49 Comment(8)
Yeah, or no quotes at all, if the code doesn't contain spaces.Rondel
Using the * globbing in arguments does not seem to work in windows.Selfaddressed
I notied that on windows I had to used double quotes around the regexLothair
glob hack to support wildcard command line arguments (*.dat) for Windows users: BEGIN { @ARGV = map +glob, @ARGV }Delhi
What if search and replacement strings are in variables? perl -pi -e 's/$oldKey/$trimmedNewKey/g' AppConstants.txt, did not work for meEndorsed
@nr5, Look at my answer below. I had a case it didn't work for me either and the issue was the Regular Expression processing,Runoff
How do I do substitution atomically (per-file, not per-line) without any backup files? Should i use -i or not? I don't understand what "in-place editing" means. I want to do substitution in files that I'm watching for changes, so it needs to be atomic.Sortilege
@nr5 The problem is that bash variables don't expand within single quoted strings. Use double quotes.Crinkly
A
22

None of the existing answers here have provided a complete example of how to do this from within a script (not a one-liner). Here is what I did:

rename($file, $file . '.bak');
open(IN, '<' . $file . '.bak') or die $!;
open(OUT, '>' . $file) or die $!;
while(<IN>)
{
    $_ =~ s/blue/red/g;
    print OUT $_;
}
close(IN);
close(OUT);
Assimilable answered 5/12, 2017 at 23:11 Comment(1)
ugly misuse of $_Triumph
R
14

$_='~s/blue/red/g';

Uh, what??

Just

s/blue/red/g;

or, if you insist on using a variable (which is not necessary when using $_, but I just want to show the right syntax):

$_ =~ s/blue/red/g;
Rondel answered 9/8, 2011 at 10:55 Comment(0)
R
4

It can be done using a single line:

perl -pi.back -e 's/oldString/newString/g;' inputFileName

Pay attention that oldString is processed as a Regular Expression.
In case the string contains any of {}[]()^$.|*+? (The special characters for Regular Expression syntax) make sure to escape them unless you want it to be processed as a regular expression.
Escaping it is done by \, so \[.

Runoff answered 2/5, 2020 at 9:4 Comment(0)

© 2022 - 2024 — McMap. All rights reserved.