クラスタから値を収集するスクリプトがあります。場合によっては、これらの値に複数の行があります。データの配置方法を指定する printf 形式がありますが、複数行を考慮しないため、ギャップが曲がります。
The data should look like this:
Service Group AutoStart List System List
foo sys1 sys1
sys2 sys2
代わりにこんな感じです。
Service Group AutoStart List System List
foo sys1
sys2 sys1
sys2
自動起動リストとシステムリストは同じでなければなりませんが、どちらも正しい列に値を強制する方法を理解できませんでした。
sgheader="\n\033[4m\033[1m%-30s %-30s %-15s\033[0m\033[0m"
sgformat="\n%-30s %-30s %-15s"
printf "${sgheader}" "Service Group" "Autostart List" "System List"
printf "${sgformat}" "${svcgroup}" "${autostrtlist}" "${hosts}"
答え1
たぶん、次のようなものがあります。
svcgroup='foo' autostrtlist=$'sys1\nsys2' hosts=$'sys1\nsys2'
paste <(printf '%s\n' "$svcgroup") \
<(printf '%s\n' "$autostrtlist") \
<(printf '%s\n' "$hosts") | expand -t30
(ksh93/zsh/bash 構文)。または、次のシステムでPOSIXly /dev/fd/x
:
paste /dev/fd/3 3<<E3 /dev/fd/4 4<<E4 /dev/fd/5 5<<E5 | expand -t 30
$svcgroup
E3
$autostrtlist
E4
$hosts
E5
とは別にdash
、yash
最新バージョンと同様に、サブシェルbash
によって提供されるパイプの代わりに一時ファイルを使用するため、おそらくより効率的です(移植性が高いだけでなく)。
答え2
変数から新しい行を削除できます。
var=$(echo "$var" | tr -d '\n')
答え3
毎回1行に収まる場合は、いくつかの簡単な方法です。要求どおりに実行するには、列を正しくソートするためにもう少し努力する必要があります。基本的なアイデアは次のとおりです。
#!/bin/bash
inputA="foo"
inputB=$'sys1\nsys2\n'
inputC=$'sys1\nsys2\n'
sgheader="\033[4m\033[1m%-30s %-30s %-15s\033[0m\033[0m\n"
sgformat="%-30s %-30s %-15s\n"
printf "${sgheader}" "Service Group" "Autostart List" "System List"
# This shows two simple ways to do this which use concatenation but
# require that the result still fit in the same space as is used for
# a single line
columnA="$inputA"
columnB=$(echo "$inputB"|awk '{printf("%s,",$0)}'|sed 's/,.\s*$//')
columnC=$(echo "$inputC"|tr '\n' ',')
printf "${sgformat}" "${columnA}" "${columnB}" "${columnC}"
# This is a version which outputs like originally asked. It is much harder
# to tweak the formatting of this version though.
pr -tm <(printf '%s\n' "$inputA") <(printf '%s\n' "$inputB") \
<(printf '%s\n' "$inputC") | expand -t10
私が知っている限り、あなたが望む方法でこれを行う最善の方法は厄介です。その後も、出力をさらに具体化して正しくソートできます。