如何匹配模式然后复制多行?
How do I match a pattern and then copy multiple lines?
我正在处理两个文件。第一个文件是我必须搜索的主数据库文件。第二个文件是我可以制作的文件,它允许我命名我想从 master 数据库中提取的项目。我设法制作了一个 AWK 解决方案,它将搜索主数据库并提取与第二个文件匹配的确切行。但是,我不知道如何将匹配后的行复制到我的新文件中。
主数据库看起来像这样:
40005X/50005/60005/3/10/9/
10038A/20038/30038/0/5/23./XXXX/
10039A/20039/30039/0/5/23./XXXX/
10040A/20040/30040/0/5/23./XXXX/
10041A/20041/30041/0/5/23./XXXX/
10042A/20042/30042/0/5/23./XXXX/
10043A/20043/30043/0/5/23./XXXX/
10044A/20044/30044/0/5/23./XXXX/
10045A/20045/30045/0/5/23./XXXX/
10046A/20046/30046/0/5/23./XXXX/
40006X/50006/60006/3/10/3/
10047A/20047/30047/0/5/23./XXXX/
10048A/20048/30048/0/5/23./XXXX/
10049A/20049/30049/0/5/23./XXXX/
40007X/50007/60007/3/10/3/
10050A/20050/30050/0/5/23./XXXX/
10051A/20051/30051/0/5/23./XXXX/
10052A/20052/30052/0/5/23./XXXX/
40008X/50008/60008/3/10/1/
10053A/20053/30053/0/5/23./XXXX/
40009X/50009/60009/3/10/3/
10054A/20054/30054/0/5/23./XXXX/
10055A/20055/30055/0/5/23./XXXX/
10056A/20056/30056/0/5/23./XXXX/
40010X/50010/60010/3/10/3/
10057A/20057/30057/0/5/23./XXXX/
10058A/20058/30058/0/5/23./XXXX/
10059A/20059/30059/0/5/23./XXXX/
在我的示例中,以 4000 开头的行是我要匹配的第一行。该行的最后一个数字告诉我要复制多少行。所以在第一行中,40005X/50005/60005/3/10/9/
,我将匹配 40005X,该行中的 9 告诉我下面有 9 行我需要用它复制。
第二个文件非常简单,看起来像这样:
40005X
40007X
40008X
随着脚本找到每个匹配项,我想将信息从第一个文件移动到一个新文件以供分析。最终结果如下所示:
40005X/50005/60005/3/10/9/
10038A/20038/30038/0/5/23./XXXX/
10039A/20039/30039/0/5/23./XXXX/
10040A/20040/30040/0/5/23./XXXX/
10041A/20041/30041/0/5/23./XXXX/
10042A/20042/30042/0/5/23./XXXX/
10043A/20043/30043/0/5/23./XXXX/
10044A/20044/30044/0/5/23./XXXX/
10045A/20045/30045/0/5/23./XXXX/
10046A/20046/30046/0/5/23./XXXX/
40007X/50007/60007/3/10/3/
10050A/20050/30050/0/5/23./XXXX/
10051A/20051/30051/0/5/23./XXXX/
10052A/20052/30052/0/5/23./XXXX/
40008X/50008/60008/3/10/1/
10053A/20053/30053/0/5/23./XXXX/
我目前拥有的与第一行匹配的代码是这样的:
#! /bin/ksh
file1=input_file
file2=input_masterdb
file3=output_test
awk -F'/' 'NR==FNR {id[]; next} in id' $file1 $file2 > $file3
我在 AWK 方面取得了最大的成功,但是我愿意接受任何建议。但是,我正在 UNIX 系统上进行此工作。我想将它保留为 KSH 脚本,因为我使用的大多数其他脚本都是用这种格式编写的,而且我最熟悉它。
感谢您的帮助!!
您能否尝试使用 GNU awk
中显示的示例进行跟踪、编写和测试。考虑到你想从这里的数字 X 开始打印行。根据 OP 的问题,Input_file2 是只有 ID 的文件,Input_file1 是主文件。
awk '
{
sub(/ +$/,"")
}
FNR==NR{
a[[=10=]]
next
}
/^[0-9]+X/{
match([=10=],/[0-9]+\/$/)
no_of_lines_to_print=substr([=10=],RSTART,RLENGTH-1)
found=count=""
}
{
if(count==no_of_lines_to_print){ count=found="" }
for(i in a){
if(match([=10=],i)){
found=1
print
next
}
}
}
found{
++count
}
count<=no_of_lines_to_print && count!=""
' Input_file2 Input_file1
您现有的 awk
正确匹配 ids 文件中的行,您现在需要添加一个条件以在读取匹配行的最后一个字段后提前打印 N 行。所以我们将设置一个变量p
为要打印的行数加一(当前行),并减少每行打印。
awk -F'/' 'NR==FNR{id[[=10=]]; next} in id{p=+1} p-->0{print}' file1 file2
或与最后一个条件相同,更“笨拙”(作者 Ed Morton)并涵盖大文件的任何可能的极端情况
awk -F'/' 'NR==FNR{id[[=11=]]; next} in id{p=+1} p&&p--' file1 file2
此处省略了 print
条件,因为它是默认操作,只要减少 p
为正,条件再次为真。
另一个
$ awk -F/ 'NR==FNR {a[]; next}
!n && in a {n=$(NF-1)+1}
n&&n--' file2 file1
40005X/50005/60005/3/10/9/
10038A/20038/30038/0/5/23./XXXX/
10039A/20039/30039/0/5/23./XXXX/
10040A/20040/30040/0/5/23./XXXX/
10041A/20041/30041/0/5/23./XXXX/
10042A/20042/30042/0/5/23./XXXX/
10043A/20043/30043/0/5/23./XXXX/
10044A/20044/30044/0/5/23./XXXX/
10045A/20045/30045/0/5/23./XXXX/
10046A/20046/30046/0/5/23./XXXX/
40007X/50007/60007/3/10/3/
10050A/20050/30050/0/5/23./XXXX/
10051A/20051/30051/0/5/23./XXXX/
10052A/20052/30052/0/5/23./XXXX/
40008X/50008/60008/3/10/1/
10053A/20053/30053/0/5/23./XXXX/
这会注意是否有任何内容行与给定的 ID 匹配。这只会在指定的打印行数之后寻找另一个 id。
我正在处理两个文件。第一个文件是我必须搜索的主数据库文件。第二个文件是我可以制作的文件,它允许我命名我想从 master 数据库中提取的项目。我设法制作了一个 AWK 解决方案,它将搜索主数据库并提取与第二个文件匹配的确切行。但是,我不知道如何将匹配后的行复制到我的新文件中。
主数据库看起来像这样:
40005X/50005/60005/3/10/9/
10038A/20038/30038/0/5/23./XXXX/
10039A/20039/30039/0/5/23./XXXX/
10040A/20040/30040/0/5/23./XXXX/
10041A/20041/30041/0/5/23./XXXX/
10042A/20042/30042/0/5/23./XXXX/
10043A/20043/30043/0/5/23./XXXX/
10044A/20044/30044/0/5/23./XXXX/
10045A/20045/30045/0/5/23./XXXX/
10046A/20046/30046/0/5/23./XXXX/
40006X/50006/60006/3/10/3/
10047A/20047/30047/0/5/23./XXXX/
10048A/20048/30048/0/5/23./XXXX/
10049A/20049/30049/0/5/23./XXXX/
40007X/50007/60007/3/10/3/
10050A/20050/30050/0/5/23./XXXX/
10051A/20051/30051/0/5/23./XXXX/
10052A/20052/30052/0/5/23./XXXX/
40008X/50008/60008/3/10/1/
10053A/20053/30053/0/5/23./XXXX/
40009X/50009/60009/3/10/3/
10054A/20054/30054/0/5/23./XXXX/
10055A/20055/30055/0/5/23./XXXX/
10056A/20056/30056/0/5/23./XXXX/
40010X/50010/60010/3/10/3/
10057A/20057/30057/0/5/23./XXXX/
10058A/20058/30058/0/5/23./XXXX/
10059A/20059/30059/0/5/23./XXXX/
在我的示例中,以 4000 开头的行是我要匹配的第一行。该行的最后一个数字告诉我要复制多少行。所以在第一行中,40005X/50005/60005/3/10/9/
,我将匹配 40005X,该行中的 9 告诉我下面有 9 行我需要用它复制。
第二个文件非常简单,看起来像这样:
40005X
40007X
40008X
随着脚本找到每个匹配项,我想将信息从第一个文件移动到一个新文件以供分析。最终结果如下所示:
40005X/50005/60005/3/10/9/
10038A/20038/30038/0/5/23./XXXX/
10039A/20039/30039/0/5/23./XXXX/
10040A/20040/30040/0/5/23./XXXX/
10041A/20041/30041/0/5/23./XXXX/
10042A/20042/30042/0/5/23./XXXX/
10043A/20043/30043/0/5/23./XXXX/
10044A/20044/30044/0/5/23./XXXX/
10045A/20045/30045/0/5/23./XXXX/
10046A/20046/30046/0/5/23./XXXX/
40007X/50007/60007/3/10/3/
10050A/20050/30050/0/5/23./XXXX/
10051A/20051/30051/0/5/23./XXXX/
10052A/20052/30052/0/5/23./XXXX/
40008X/50008/60008/3/10/1/
10053A/20053/30053/0/5/23./XXXX/
我目前拥有的与第一行匹配的代码是这样的:
#! /bin/ksh
file1=input_file
file2=input_masterdb
file3=output_test
awk -F'/' 'NR==FNR {id[]; next} in id' $file1 $file2 > $file3
我在 AWK 方面取得了最大的成功,但是我愿意接受任何建议。但是,我正在 UNIX 系统上进行此工作。我想将它保留为 KSH 脚本,因为我使用的大多数其他脚本都是用这种格式编写的,而且我最熟悉它。
感谢您的帮助!!
您能否尝试使用 GNU awk
中显示的示例进行跟踪、编写和测试。考虑到你想从这里的数字 X 开始打印行。根据 OP 的问题,Input_file2 是只有 ID 的文件,Input_file1 是主文件。
awk '
{
sub(/ +$/,"")
}
FNR==NR{
a[[=10=]]
next
}
/^[0-9]+X/{
match([=10=],/[0-9]+\/$/)
no_of_lines_to_print=substr([=10=],RSTART,RLENGTH-1)
found=count=""
}
{
if(count==no_of_lines_to_print){ count=found="" }
for(i in a){
if(match([=10=],i)){
found=1
print
next
}
}
}
found{
++count
}
count<=no_of_lines_to_print && count!=""
' Input_file2 Input_file1
您现有的 awk
正确匹配 ids 文件中的行,您现在需要添加一个条件以在读取匹配行的最后一个字段后提前打印 N 行。所以我们将设置一个变量p
为要打印的行数加一(当前行),并减少每行打印。
awk -F'/' 'NR==FNR{id[[=10=]]; next} in id{p=+1} p-->0{print}' file1 file2
或与最后一个条件相同,更“笨拙”(作者 Ed Morton)并涵盖大文件的任何可能的极端情况
awk -F'/' 'NR==FNR{id[[=11=]]; next} in id{p=+1} p&&p--' file1 file2
此处省略了 print
条件,因为它是默认操作,只要减少 p
为正,条件再次为真。
另一个
$ awk -F/ 'NR==FNR {a[]; next}
!n && in a {n=$(NF-1)+1}
n&&n--' file2 file1
40005X/50005/60005/3/10/9/
10038A/20038/30038/0/5/23./XXXX/
10039A/20039/30039/0/5/23./XXXX/
10040A/20040/30040/0/5/23./XXXX/
10041A/20041/30041/0/5/23./XXXX/
10042A/20042/30042/0/5/23./XXXX/
10043A/20043/30043/0/5/23./XXXX/
10044A/20044/30044/0/5/23./XXXX/
10045A/20045/30045/0/5/23./XXXX/
10046A/20046/30046/0/5/23./XXXX/
40007X/50007/60007/3/10/3/
10050A/20050/30050/0/5/23./XXXX/
10051A/20051/30051/0/5/23./XXXX/
10052A/20052/30052/0/5/23./XXXX/
40008X/50008/60008/3/10/1/
10053A/20053/30053/0/5/23./XXXX/
这会注意是否有任何内容行与给定的 ID 匹配。这只会在指定的打印行数之后寻找另一个 id。