如果一个字符串在特定文件的最后一行的开头匹配,则替换同一行中的另一字符串

问题描述

我有一个文件“ test”

下面是内容

235788@@@235788@@@20200724_103122@@@SUCCESS
235791@@@235791@@@20200724_105934@@@SUCCESS
235833@@@235833@@@20200724_130652@@@FAILURE
235842@@@235842@@@20200724_132721@@@FAILURE
235852@@@235852@@@20200724_134607@@@FAILURE
235791@@@235791@@@20200724_105934@@@SUCCESS

如果此文件最后一行 235791 开头,然后在该行上将字符串“ SUCCESS”替换为“ FAILURE”。

预期产量

235788@@@235788@@@20200724_103122@@@SUCCESS
235791@@@235791@@@20200724_105934@@@SUCCESS
235833@@@235833@@@20200724_130652@@@FAILURE
235842@@@235842@@@20200724_132721@@@FAILURE
235852@@@235852@@@20200724_134607@@@FAILURE
235791@@@235791@@@20200724_105934@@@FAILURE

下面是示例代码

id = 235791
last_build_id  = `tail -1 test | awk -F'@@@' '{print \$1}'`
if (id == last_build_id  ){
    sed -i '$s/SUCCESS/FAILURE/' test
}

我想避免这么多行,而使用一个通过正则表达式组或其他任何简单方式使用的shell命令。

解决方法

sed在这里可能会更容易

$ sed -E '$s/(^235791@.*)SUCCESS$/\1FAILURE/' file

您可以添加-i进行就地更新。

将id作为变量传递

$ id=235791; sed -E '$s/(^'$id'@.*)SUCCESS$/\1FAILURE/' file

理想情况下,您应该在双引号"$id"上加引号,但是如果您确定其中的内容,可能会一无所获。

,

您可以尝试使用GNU awk中显示的示例进行以下操作,编写和测试。您无需为此使用很多命令,我们可以在一个awk本身中完成此操作。

一种内联代码形式:

awk -v id="$your_shell_variable" 'BEGIN{ FS=OFS="@@@" } NR>1{print prev} {prev=$0} END{if($1==id && $NF=="SUCCESS"){$NF="FAILURE"}; print}' Input_file > temp && mv temp Input_file

说明: 添加以上详细说明。

awk -v id="$your_shell_variable"'                                     ##Starting awk program from here.
NR>1{                                     ##Checking condition if prev is NOT NULL then do following.
  print prev                              ##Printing prev here.
}
{
  prev=$0                                 ##Assigning current line to prev here.
}
END{                                      ##Starting END block of this program from here.
  if($1==id && $NF=="SUCCESS"){       ##Checking condition if first field is 235791 and last field is SUCCESS then do following.
    $NF="FAILURE"                         ##Setting last field FAILURE here.
  }
  print                                   ##Printing last line here.
}
'  Input_file > temp && mv temp Input_file                           ##Mentioning Input_file name here.


第二个解决方案: 根据Ed先生的评论,awk中的某些$1$NF不支持{ {1}}部分,因此,如果以上内容不适用于某人,请尝试以下更通用的解决方案。

一种内联形式的解决方案(自明确提出要求以来):

END

详细级别(非一个班轮代码):

awk -v id="$your_shell_variable" 'BEGIN{ FS=OFS="@@@" } NR>1{print prev} {prev=$0} END{num=split(prev,array,"@@@");if(array[1]==id && array[num]=="SUCCESS"){array[num]="FAILURE"};for(i=1;i<=num;i++){val=(val?val OFS:"")array[i]};print val}'  Input_file  > temp && mv temp Input_file
,

使用GNU sed

sed -E '${/^235791\>/ s/SUCCESS$/FAILURE/}' file

或者在MacOS上使用BSD

sed -E '${/^235791@/ s/SUCCESS$/FAILURE/;}' file

使用“文件中的最后一个X”时,通常更容易反转文件并使用“第一个X”:

tac file | awk '
    BEGIN {FS = OFS = "@@@"} 
    NR == 1 && $1 == 235791 && $NF == "SUCCESS" {$NF = "FAILURE"}
    1
' | tac
,
$ awk -v val='235791' '
    BEGIN { FS=OFS="@@@" }
    NR>1 { print prev }
    { prev=$0 }
    END {
        $0=prev
        if ($1 == val) {
            $NF="FAILURE"
        }
        print
    }
' file
235788@@@235788@@@20200724_103122@@@SUCCESS
235791@@@235791@@@20200724_105934@@@SUCCESS
235833@@@235833@@@20200724_130652@@@FAILURE
235842@@@235842@@@20200724_132721@@@FAILURE
235852@@@235852@@@20200724_134607@@@FAILURE
235791@@@235791@@@20200724_105934@@@FAILURE