質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
85.50%
sh

shは、UNIX系OSのシェル操作の1つであり、最も基本的なシェルのことです。

Q&A

解決済

8回答

22685閲覧

shellでファイルの先頭に文字列(改行入り)を挿入したい

jollyjoester

総合スコア1585

sh

shは、UNIX系OSのシェル操作の1つであり、最も基本的なシェルのことです。

2グッド

1クリップ

投稿2015/07/17 11:24

編集2015/07/19 11:19

shellでファイルの先頭に例えば下記のような文字列(改行入り)を挿入したいです。
環境はMac & zshです。

Title message : hogehoge

sed等で簡単にできるかなと思ったのですが意外と手こずり、結局以下のような感じになってます(´・ω・`)
もっとスマートな方法があると思うのですが、シェル力がなさ過ぎてこれが限界でした。

# sample.txtというファイルの先頭に文字列を追加したい場合 # 元のファイルを複製 cp -r sample.txt sample.txt.old # 追加したい文字列をファイルに出力 echo -e "Title\\n\\nmessage : hogehoge\\n" > add.txt # ファイルをくっつけて元のファイルを更新 cat add.txt sample.txt.old > sample.txt # いらんファイル消す rm -rf add.txt sample.txt.old

もっといい方法をご存知の方、どうぞご教授お願いしますm(_ _)m

[追記]
最初sedでやろうとしたときはこんな感じでやりました。
(MacのsedはBSD系ということで-iの後ろの引数にバックアップファイルの拡張子を入れないといけないのですが、バックアップファイルはいらないので""にしてます。)

$ sed -i "" "1iTitle\n\nmessage : hogehoge\n" sample.txt sed: 1: "1iTitle\n\nmessage : ho ...": command i expects \ followed by text

そしたらこんなエラーになってしまい、よくわからなかったので上記の回避策になった次第です。

[追記2]
Mac(BSD)のsedのiコマンドで改行をうまく扱えないのはバグのようでした。2005年のバグが2009年時点でも未解決とコメントされていますが、現在でも未解決のようですね。
http://www.linuxquestions.org/questions/programming-9/sed-error-command-c-expects-%5C-followed-by-text-under-os-x-but-works-in-linux-730997/

[追記3]
多くの皆様に回答していただき、大変感謝です!
どの回答も素晴らしく大変勉強になりました。
ベストアンサーを選ぶのが苦しかったのですが、shの範囲内でシンプルな書き方のできるedコマンドとsedのsコマンドの回答をベストアンサーにさせていただきました。
個人的にはngyukiさんのsedを使わない別解が好きです^^
実際の現場ではperlの回答を採用することになりそうですw

afroscript, ikuwow👍を押しています

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答8

0

ベストアンサー

ファイルを直接書き換えるなら、いまではあまり一般的では無いですが ed コマンドがあります(本来は普通のエディタ)。
裏で新規ファイルを作らずに元のファイルに上書きします。

zsh

1cat <<EOS | ed sample.txt 20a 3Title 4 5message : hogehoge 6 7. 8wq 9EOS

補足:
もし、ed が無ければ ex でもいいです。

追記:
いろいろ試しましたが、BSDのsedだとコマンドラインでのiコマンドは無理そうです。
iコマンドを使うには、別途sedスクリプトを書いたファイルを作って、sed -f するしかなさそうです。

iコマンドでなくsコマンドを使ったpi-chanさんの回答をBSD用に書き直すと、

zsh

1sed -i "" -e $'1s/^/Title\\\n\\\nmessage : hogehoge\\\n/' sample.txt

投稿2015/07/17 14:30

編集2015/07/19 01:35
otn

総合スコア84423

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

jollyjoester

2015/07/18 13:50

ありがとうございます!これもいいですね。edコマンドもうちょい調べてみます!
jollyjoester

2015/07/19 07:32

回答追記ありがとうございます! sコマンドでのsed、期待通りの動作です!
guest

0

こういうことでしょうか?

sh

1sed -i "1iTitle\n\nmessage : hogehoge\n" sample.txt

1i で、1行目に挿入、です。

(CentOS + bash で試しているので Mac + zsh で動かなかったらごめんなさい)


sed on OSX insert at a certain line

Mac の sed はかなり異なっているようです。。。

  • i の後ろにバックスペースと改行が必要
  • i で改行を挿入できない(ので x と p で代替)
sed -i '' -e $'1{\ni\\\nTitle\nx\np\np\nx\ni\\\nmessage : hogehoge\n}' sample.txt

下記と同じです。

sed -i '' '1{ i\ Title x p p x i\ message : hogehoge } ' sample.txt

(Mac + bash で試しているので Mac + zsh で動かなかったらごめんなさい)

投稿2015/07/17 11:52

編集2015/07/18 20:32
ngyuki

総合スコア4514

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

otn

2015/07/17 12:00

-i オプションについての説明もあった方が良いでしょうね。出力ファイルの作成とリネーム(-cも併用するとリネームで無くコピー)を裏で行って、あたかもファイルを上書きしたかのごとくの結果になります。
jollyjoester

2015/07/18 13:47

そうなんです! 最初この方法でいけるかなと思ったのですがMacのsedがBSD系のsedらしくダメだったんです(´;ω;`) (質問に書いといた方がよかったですね。追記します。)
jollyjoester

2015/07/18 14:51

ありがとうございます! それを試した先にさらなるエラーが(泣) (状態を質問に追記しました。)
jollyjoester

2015/07/19 07:41

回答追記ありがとうございます! なるほど、command i expects \ followed by textエラーはそういうことだったのですね。。。 あとすみませんxとpのくだりが理解できてないのですがなぜここはxとpなのでしょうか?m(_ _)m
ngyuki

2015/07/19 11:26

i で改行を挿入することができなかったので、その代わりに x と p を使っています。 最初の x でパターンスペースを空にして、続きの2個の p でパターンスペースを出力、次の x でパターンスペースを元に戻しています。
jollyjoester

2015/07/19 11:36

なるほど!理解しました。(理解しましたが使いこなせる気がしない^^;;;) もっと精進しますm(_ _)m 貴重な回答ありがとうございます!
guest

0

perl なら下記のようにすると良いかと思います。

sh

1perl -i -pe 's/^/Title\n\nmessage : hogehoge\n/ if $. == 1' sample.txt

-i は sed の -i と同じ。
-p は perl を sed と同じように動かすためのオプションです。
-e はその後に perl スクリプトが続きます。

スクリプトの $. は読んだファイルの行番号です。後置if で1行目だけ置換しています。

投稿2015/07/18 22:32

ngyuki

総合スコア4514

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

jollyjoester

2015/07/19 07:28

こちらもperlという選択肢を気づかせてくれてありがとうございます! 動作も期待通りです^^/
guest

0

こんな感じで如何でしょうか?

perl -i -0777pe 's/^/Title\n\nmessage : hogehoge\n/' sample.txt

行の区切り文字を指定するオプション「0」のパラメータに、文字の割り当てがないコード「777」を指定すると、ファイルを行単位ではなく一括で読み込み処理をします。

「i」オプションの挙動はMacでも変わりません。

巨大ファイルに対しては使用できませんが、ファイルの先頭に文字列を挿入する簡便法として有用かと思います。

投稿2015/07/18 22:11

pi-chan

総合スコア5936

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

jollyjoester

2015/07/19 07:26

perl!そうだったPerl使ったことないんでMacで普通にperl使えるの忘れてました。。。 期待通りの動作です^^/
guest

0

sed を使わない別解です。

rm -f sample.txt を挟むことで cat - sample.txtcat > sample.txt を別のファイル(別の i-node)にしています。

sh

1echo -e "Title\n\nmessage : hogehoge\n" | cat - sample.txt | (rm -f sample.txt ; cat > sample.txt )

これ・・cat が sample.txt を開くより先に rm が sample.txt を削除するとダメそうなので、次のように sample.txt はシェルに開かせたほうがいいかもしれません。

sh

1cat <(echo -e "Title\n\nmessage : hogehoge\n") - < sample.txt | (rm -f sample.txt ; cat > sample.txt)

これなら sample.txt は cat が開始するより先にシェルが開くので確実だと思います。

(CentOS + bash で試しているので Mac + zsh で動かなかったらごめんなさい)

投稿2015/07/17 12:01

編集2015/07/18 23:03
ngyuki

総合スコア4514

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

jollyjoester

2015/07/18 13:49

Macでもいけました! これはいい感じですね(`・ω・´)
guest

0

# 追加したい文字列をファイルに出力 # ファイルをくっつけて元のファイルを更新

の 2つのプロセスは

echo -e "Title\\n\\nmessage : hogehoge\\n"|cat - sample.txt.old > sample.txt

と一発でできます。

上書き更新自体は、お気づきの通り、難しい処理なので 一時ファイル無しで、ファイルを上書き更新するシェルスクリプト のような方法しかないのではないでしょうか。

&& 等を用いれば、行数を削ることはできますが、実質、複製・上書き・削除の3つは必要であると思います。

投稿2015/07/17 11:50

takotakot

総合スコア1111

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

jollyjoester

2015/07/18 13:43

回答ありがとうございます! ハイフンで標準入出力になるってこと知らなかったので勉強になりました。
guest

0

1行目だけ置換するというアイデアを拝借すると、sed ならこのように書けますね。

sed -i"" -e '1s/^/Title\n\nmessage : hogehoge\n/' sample.txt

投稿2015/07/18 22:57

編集2015/07/18 23:04
pi-chan

総合スコア5936

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

jollyjoester

2015/07/19 07:24

おお!-iと""をくっつけるんですか!! いけたと思ったんですが改行がnになっちゃいました。。。\\nにしてみてもダメっすね。。。
pi-chan

2015/07/19 11:19

勘違いだったら申し訳ありませんが、ひょっとして、スクリプト部分 ``` 1s/^/Title\n\nmessage : hogehoge\n/ ``` をダブルクォート `"` で括っていませんか? そこはシングルクォート `'` で括らないとダメですよ。 自分のMacで確認すると想定通りの結果になっていますが… ``` $ cat sample.txt test1 test2 test3 $ $ sed -i"" -e '1s/^/Title\n\nmessage : hogehoge\n/' sample.txt $ $ cat sample.txt Title message : hogehoge test1 test2 test3 $ ```
jollyjoester

2015/07/19 11:26

回答いただいたスクリプトをコピペで試したのですが自分の環境だと改行がnになってしまうんです。。。 Titlennmessage : hogehogenfirst line MacはOS X10.10.4で $ echo $SHELL /bin/zsh なんですが(´・ω・`)
jollyjoester

2015/07/19 11:29

``` $ cat sample.txt test1 test2 test3 $ sed -i"" -e '1s/^/Title\n\nmessage : hogehoge\n/' sample.txt $ cat sample.txt Titlennmessage : hogehogentest1 test2 test3 ```
guest

0

参考情報

google で "sh file insert header" を検索した結果からの抜粋です。

投稿2015/07/17 14:45

katoy

総合スコア22324

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

jollyjoester

2015/07/18 14:03

情報ありがとうございます! いろいろ試してみたいと思います(^^/ が、Mac上でshを使った場合だとあんまりなさそうですね(´・ω・`)
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.50%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問