場当たり的に覚えた sed の使い方
テキストファイルの置換に使う sed
(セド) コマンド。独特の構文で体系的に覚えられていないのだが、やりたいことから逆引きして場当たり的に覚えてきたことをまとめてみようと思う。
sed
コマンドは OS によって若干の差異があるが、基本的には同様に使える。今回の記事は、Windows の GitBash、MacOS 標準のターミナル、Linux 環境のどこかで試したメモをベースに書いているので、環境差異で動かなかった場合はご容赦を…。
ちなみに、sed の名称は「stream editor」の略らしい。
目次
基本的な使い方
sed
コマンドの基本的な構文は以下のとおり。
$ sed 【オプション】 【スクリプト】 【操作対象のファイル名】
【スクリプト】
は文字列操作のための式のこと。本来 -e
オプションで渡すモノだが、-e
オプションは省略できるので上のように書いた。
テキストを置換する
任意のテキストを置換するには、以下のように書く。コレが最も基本的なパターン。
$ sed 's/【置換したい文字列】/【置換後の文字列】/g' 【ファイル名】
置換したい文字列 (検索する文字列) はケースセンシティブ、大文字小文字を区別するので、hoge
と HOGE
は別物になることに注意。
g
オプション
スクリプト内の g
は Global の意味。sed
は入力を行単位に処理するのだが、g
オプションを付けない場合はその1行で最初に出てきた文字列しか置換しない。
つまり、例えば
hoge foo hoge who
hoge bar hoge baz
というテキストファイルがあったとして、g
オプションなしで hoge
を置換すると
$ sed 's/hoge/fuga/' example.txt
fuga foo hoge who
fuga bar hoge baz
このように、その行の2つ目以降の hoge
は置換されない。
g
オプションを付ければ、以下のように全て置換される。
$ sed 's/hoge/fuga/g' example.txt
fuga foo fuga who
fuga bar fuga baz
正規表現との組み合わせ
この式の中では正規表現も使えるので、^
で行頭、$
で行末を示したり、.*
で任意の文字列を指定したりできる。
# 行頭が「example」で始まるものを「replaced」にする。行頭を指定しているので g オプションの有無は関係なし
$ sed 's/^example/replaced/' example.txt
# 行末が「example」で終わるものを「replaced」にする。これも g オプションの有無は関係なし
$ sed 's/example$/replaced/' example.txt
# 「ex【任意の文字列】le」に該当する箇所を「replaced」に置換する
$ sed 's/ex.*le/replaced/g' example.txt
デリミタ
式の中では、基本的には /
がデリミタとして扱われるので、文字列中に /
を含む場合はエスケープが必要になる。
# 「http://」部分を「ftp://」に置換する
$ sed 's/http:\/\//ftp:\/\//g' example.txt
なお、デリミタは s
の次に書いた文字がデリミタとして使われるので、@
などに変更しても良い。
# デリミタを「@」に変更。こうすれば「/」をエスケープしてもしなくても書ける
$ sed 's@http://@ftp://@g' TEST.txt
行を削除する
これまでの 's/検索/置換/'
とは別に、/検索/d
という式を使うと、条件に合う行をまるごと削除できる。
# 「hoge」を含む行を削除する
$ sed '/hoge/d' example.txt
ファイルに上書き保存する
これまでのコマンドだと、置換結果は標準出力に表示されるだけで、ファイルに置換結果が反映されない。操作対象のファイルに置換結果を反映して上書き保存するには、-i
オプションを使う。
# 「example.txt」中の「hoge」を「fuga」に置換して上書き保存する
$ sed -i 's/hoge/fuga/g' example.txt
より実践的なサンプル
何らかの理由で、設定ファイルの特定行をコメントアウトしたり、外したりしたい時。
# 何かの設定ファイル
my_setting: HOGEFUGA
my_setting
で始まる行の先頭に #
を付与してコメントアウトし、上書き保存する。
$ sed -i 's/^my_setting/# my_setting/g' example.conf
こうしてコメントアウトした行をまた戻す時。
# 何かの設定ファイル
# my_setting: HOGEFUGA
今度は # my_setting
で始まる行を検索してやれば良い。
$ sed -i 's/^# my_setting/my_setting/g' example.conf
以上
今回紹介したのは sed
の機能のごく一部だが、これらの知識だけでも一旦はそれらしく置換処理ができるようになったと思う。
sed
や正規表現をきちんと読める人もなかなか多くはないと思うので、あまり極端に多用しすぎず、人間が読んで分かりやすい範囲の利用に留めておいた方が良いかな、と個人的には思う。