while IFS=, read -r Q1 Q2 Q3 Q4; do #Reading from a CSV file
Replace Q2 with the string "Nan" #Replace the second comma separated value with a string
私はファイルから読み取られた2番目の値が文字列に変わり、変数を置き換えるのではなく、実際のファイル自体を変更する2番目の行を実行する方法を探しています。私はsed -iを使用すると思いますが、特にファイルに書き直さずにこれを行うより簡単な方法が何であるかわかりません。
答え1
区切り文字がある単純な形式のファイルには awk が便利な場合が多いです。
たとえば、入力ファイルが与えられると、各行foo.txt
でコンマで区切られた2番目のフィールドが変更され、残りは変更されずに残り、変更されたファイルが印刷されます。
$ awk -F, -v OFS=, '{ $2 = "Nan"; print }' < foo.txt
a,Nan,c,d
e,Nan,g,h
-v FS=,
次に、-v OFS=,
入力フィールドと出力フィールドの区切り文字をコンマに設定し、行の$2 = ...
2番目のフィールドを変更します。すべての行を変更するには、NR == 2
テストを削除するだけです。
1行(123行など)でのみこれを行う必要がある場合は、次のことができます。
$ awk -F, -v OFS=, -v line=123 'NR == line { $2 = "Nan" } { print }' < foo.txt
私が「単純形式」と言ったことに注意してください。それがデータに引用符で囲まれたコンマが表示される可能性がある「カンマ区切り」形式の1つである場合、awkは適切なツールではありません。
答え2
通常のbashではこれを行うことができます(遅く、CSVの詳細を知らない)。
while IFS=, read -r q1 q2 q3 q4; do
echo "$q1,NaN,$q3,$q4"
done < inputFile > outputFile
ln inputFile inputFile.bak # backup input file
mv outputFile inputFile # overwrite input file
追加のツールを使用すると、パフォーマンスとセキュリティを実現できます。
ruby -rcsv -e '
CSV.foreach(ARGV.shift) {|row| row[1] = "NaN"; puts CSV.generate_line row}
' infile > outfile