从所有文件中提取2个值的Shell脚本?

4

我有一个目录,里面装满了像这样的文件:

[Location]
state=California
city=Palo Alto

[Outlet]
id=23
manager=John Doe

我想写一个小脚本,输出每个文件的一行内容,例如:
John Doe,Palo Alto

我该怎么做呢?我猜需要使用一些grep和循环操作。目前我已经有了以下代码:

#!/bin/bash
echo Manager,City > result.txt
for f in *.config
do
  cat "$f" | grep manager= >> result.txt
  cat "$f" | grep city= >> result.txt
done

但当然这是不完整的,因为grep会将整行返回到单独的一行,而我只想要等号后面的部分。

3个回答

2
echo Manager,City > result.txt

for f in *.config; do
    manager=$(awk -F= '$1=="manager" {print $2}' "$f")
    city=$(   awk -F= '$1=="city"    {print $2}' "$f")

    echo "$manager,$city"
done >> result.txt

awk -F= 使用等号作为字段分隔符,然后检查所需变量($1)并打印它们的值($2)。$(cmd) 捕获命令的输出,并产生可以分配给两个变量 $manager$city 的字符串。


2

类似于John Kugelman的回答,但使用grep

echo Manager,City > result.txt
for file in *.config; do 
    name=$(grep -oP '(?<=manager\=).*' "$file")
    location=$(grep -oP '(?<=city\=).*' "$file")
    echo "$name,$location"
done >> result.txt

@stwissel 你选择了这个答案,这个回答也是受它启发的!所以一切都好!;) - jaypal singh

1
你可以使用单个awk命令来完成此操作,如下所示的剪辑版:
pax> cat 1.config
[Location]
state=California
city=Palo Alto

[Outlet]
id=23
manager=John Doe

pax> cat 2.config
[Location]
state=Western Australia
city=Perth

[Outlet]
id=24
manager=Pax Diablo

pax> awk '
     /^city=/   {gsub (/^city=/, "", $0); city=$0}
     /^manager=/{gsub(/^manager=/, "", $0); print $0 "," city}
' *.config
John Doe,Palo Alto
Pax Diablo,Perth

请注意,这假定城市在经理之前,并且所有文件都有城市和经理。如果这些假设是不正确的, awk 脚本会变得更加复杂,但仍然可行。
在这种情况下,它会变成这样:
awk '
    FNR==1       {city = ""; mgr = ""}
    /^city=/     {gsub (/^city=/, "", $0); city = $0}
    /^manager=/  {gsub (/^manager=/, "", $0); mgr = $0}
                 {if (city!="" && mgr!=""){
                     print mgr "," city; city = ""; mgr = "";
                 }}
' *.config

这样做的作用是使顺序无关紧要。它在每个文件的开头将城市和经理变量重置为空字符串,并仅在找到相关行时将它们存储起来。每一行之后,如果两者都设置了,它就会打印并清除它们。

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接