シェルスクリプト総合 その8
■ このスレッドは過去ログ倉庫に格納されています
0001名無しさん@お腹いっぱい。
2007/02/15(木) 14:28:44スクリプトのお勉強・自慢・腕試しなどにどうぞ。
まずは注意点、リンク、地鎮祭など(>>1-6くらい)をご覧ください。
□お約束
・特記なき場合はBourne Shell(/bin/sh)がデフォルトです。
bash/zsh/ksh/ashなどに依存する場合は明示しましょう。
Linuxユーザは/bin/shの正体がbashなので特に注意。
・csh/tcshのシェルスクリプトは推奨されません。
(理由は「csh-whynot」でググれ)
・UNIXにはシェルスクリプトに便利な小さなコマンドがいろいろあります。
manや参考リンクを見ましょう。
aproposないしはman -kでそれらしい単語による簡単な検索もできます。
・シェルスクリプトのことをシェルってゆーな
・シェルで使えるワイルドカード等は正規表現ではありません。
正規表現の話題はスレ違い(正規表現スレへ)
□初心者へのアドバイス:
・適した道具を判断するのも頭の重要な使い方。シェルスクリプトよりも
RubyやPerlの方が適した仕事には素直にそちらを使いましょう。
・知らないコマンドが出てきたらmanを引きましょう。
・思い通りに動かないときは、まずは sh -x でトレースしましょう。
前スレ
シェルスクリプト総合 その7
http://pc10.2ch.net/test/read.cgi/unix/1157601611/
0320名無しさん@お腹いっぱい。
2007/03/28(水) 01:47:010321名無しさん@お腹いっぱい。
2007/03/28(水) 02:04:060322名無しさん@お腹いっぱい。
2007/03/28(水) 02:30:58awkでやれ
awk '{line[NR]=$0}END{for(i=NR;i>0;i--){if(line[i]=="検索パターン") print line[i];}}' input.txt
0323名無しさん@お腹いっぱい。
2007/03/28(水) 02:50:520324名無しさん@お腹いっぱい。
2007/03/29(木) 16:52:36行数は同じでそれぞれの行が関係のある物になっています。
現在、その二つの行をそれぞれスペースで区切って接続を行いたく考えています。
A.txt
山田
山下
B.txt
太郎
二郎
AB.txt
山田 太郎
山下 二郎
A.txtとB.txtからAB.txtを作成する妙案などございましたらご教示いただけませんでしょうか?
よろしくお願い致します。
0325名無しさん@お腹いっぱい。
2007/03/29(木) 16:56:53paste A.txt B.txt > AB.txt
0326名無しさん@お腹いっぱい。
2007/03/29(木) 17:09:44PASTE(1)
名前
paste - ファイルを行単位でマージする
大変失礼いたしました…orz
シェルスクリプトとかそう言うレベルで無かったことをお許し下さい…。
0327名無しさん@お腹いっぱい。
2007/03/30(金) 09:37:33幸子の場合、a.txtが山田となっていたら
里中に変更する。という条件付きくらいにしとけば
スクリプトでしなきゃいかんけどな。
0328名無しさん@お腹いっぱい。
2007/03/30(金) 15:20:07そのenvelope Fromを、実データのFromに置換して/usr/bin/vacation
に渡すスクリプトを書きたいのですが・・・
sed -n 's/^From: [^<]*<\([^>]*\).*/\1/p'の内容を覚えさせておいて、
もう一度先頭から読み込んで置換すればよいと思いますが、
その方法がわからないのです。
微妙な内容なのでググるためのキーワードもいまいちで、ヒットしません。
別にsedでなくてpealやawkでもよいのですが、何かいい方法はありますでしょうか?
0329名無しさん@お腹いっぱい。
2007/03/30(金) 15:26:46そのまま
・覚えさせる->ファイルに書く
・もう一度先頭から読み込んで置換する->そうする
でいいんじゃないの?
0330名無しさん@お腹いっぱい。
2007/03/30(金) 16:01:32頭とケツにそれぞれ「”」を付けたいです。
何か良い方法は有りませんでしょうか?
0331名無しさん@お腹いっぱい。
2007/03/30(金) 16:08:360332名無しさん@お腹いっぱい。
2007/03/30(金) 16:17:30パイプが無駄。一発でできる。
sed 's/\(.*\)/"\1"/'
0333名無しさん@お腹いっぱい。
2007/03/30(金) 16:24:130334328
2007/03/30(金) 16:31:53難しく考えずに、そうすることにします。
0335名無しさん@お腹いっぱい。
2007/03/30(金) 16:44:44ありがとうございます。
頑張ってsed覚えます。
0336328
2007/03/30(金) 17:10:27と思ったら、メール出力から起動されるプログラムは
きわめて弱い実行権限(nobody)しかないので、/tmpにさえ
ファイルを作れません。(人間がテストしたときは巧くいきましたが・・)
権限を強めればセキュリティーリスクが増大するし、
やはり、スクリプトの中で記録と標準入力の再読み込みが必要な模様です。
まあ記録は置換文字列に変数を使えることが判ったので何とかなりそうですが、
「初めからもう一度実行」が途方に暮れてしまいますね。
何かよい方法ありますでしょうか?
0337名無しさん@お腹いっぱい。
2007/03/30(金) 17:14:32・そもそも sendmail でも postfix でも .forward のファイル所有者と
同じ権限でプログラムが起動する。nobody にはならない。
0338328
2007/03/30(金) 18:36:24なるほど
では、ほかの原因を考察して見ます。
ありがとうございます。
0339名無しさん@お腹いっぱい。
2007/03/30(金) 22:30:53sed 's/.*/"&"/'
でもいいな。
0340名無しさん@お腹いっぱい。
2007/03/30(金) 23:05:57正規表現は使わない方が速いかな。
awkで、
awk '{print "\""$0"\""}'
0341名無しさん@お腹いっぱい。
2007/04/03(火) 01:43:34「\」を前に置いても駄目でした…
0342名無しさん@お腹いっぱい。
2007/04/03(火) 02:26:06system が起動しているシェルに食われたんじゃね?
0343名無しさん@お腹いっぱい。
2007/04/04(水) 00:17:45例えばこんな感じ。実際はこんなシンプルじゃないけど。
awk '{system("grep \"^$\" $1")}' input.file
systemが食われたのか、systemの引数の中では引用符が使えないのか、切り分けができません。
0344名無しさん@お腹いっぱい。
2007/04/04(水) 00:26:480345名無しさん@お腹いっぱい。
2007/04/04(水) 01:00:44あそっか、$1は""の外でした。。
awk '{system("grep \"^$\" "$1)}' input.file
本物は業務で使う予定だからコピペできないんだよね。
0346名無しさん@お腹いっぱい。
2007/04/07(土) 02:11:040347名無しさん@お腹いっぱい。
2007/04/08(日) 22:55:59ちゃんと伝わっていなかったようで。
awkの中にシェルの変数の$1入れてもawkがもっていっちゃうからダメなんだって。
0348名無しさん@お腹いっぱい。
2007/04/09(月) 22:22:34そうか?普通に↓とか使えてるぞ?
awk '{sum[substr($0,a,b)]+=substr($0,c,d);}END{for(n in sum)print n,sum[n];}' input.txt
0349名無しさん@お腹いっぱい。
2007/04/10(火) 20:40:57>>343や>>345の例だと、$1をシェルがもっていっちゃって動かないだろうから、
本来やりたいことである
>awkのsystem関数に与えるコマンド文字列の中で「"」や「'」って使える?
の問題とは違ったところでハマっているでしょ、ってことを書いただけなんだけど。
($1にはinput.txtを入れたいのだろうと推測)
それに対して、$数字は awkで使用できるよ、と書かれても困っちゃうんだが。
0350名無しさん@お腹いっぱい。
2007/04/11(水) 06:22:32言いたいだろうことは理解するが、347がそれが伝わる書き方とは思えなかったぞ。
例えば、何を誰がもっていっちゃうと書いたか、347と349を比較してみ。
0351名無しさん@お腹いっぱい。
2007/04/13(金) 00:00:330352名無しさん@お腹いっぱい。
2007/04/13(金) 01:37:02フィールドセパレータ(デフォルト空白文字)で区切られたn番目のフィールドを
$nで参照できるんだよな。
awk '{system("grep \"^$\" "$1)}' input.file
で$1はinput.fileから読み込んだカレントレコードの1番目のフィールドを
参照すると。
$数字がawkで使えるのは良いとして結局"や'は
system関数の引数の中で使えるのかどうか?
0353名無しさん@お腹いっぱい。
2007/04/13(金) 16:02:04Debianを使っているときにはUTF-8を、
FreeBSDを使っているときにはEUC-JPを
環境変数LANGに設定するようにしたいのですが
どのようにコードをかけばよいのでしょうか?
0354名無しさん@お腹いっぱい。
2007/04/13(金) 16:07:45uname とか hostname あたり見るとか。
0355353
2007/04/13(金) 16:42:04unameとif文を使えばできると思うんですけど
それでsourceコマンドを実行したときに
ちゃんとわりあたるかどうか・・・
0356名無しさん@お腹いっぱい。
2007/04/13(金) 16:54:07おいおい、sourceなんて使わないよ。
あと、ifじゃなくてcaseを使うのが定石。
case `uname -s` in
Linux) export LANG=ja_JP.UTF-8;;
FreeBSD) export LANG=ja_JP.eucJP;;
esac
0357名無しさん@お腹いっぱい。
2007/04/13(金) 16:55:19なんで source が出てくるのかよくわからん。
0358名無しさん@お腹いっぱい。
2007/04/15(日) 15:34:49Cとかだとswtichは敬遠される傾向にあるけどな。
0359名無しさん@お腹いっぱい。
2007/04/15(日) 17:19:240360名無しさん@お腹いっぱい。
2007/04/15(日) 17:33:16if [ `uname -s` = Linux ]; then
export LANG=ja_JP.UTF-8
elif [ `uname -s` = FreeBSD ]; then
export LANG=ja_JP.eucJP
fi
となって、unameの実行が2回になって無駄になる。
かと言ってunameを1回にしようとすると、
tmp=`uname -s`
if [ $tmp = Linux ]; then
export LANG=ja_JP.UTF-8
elif [ $tmp = FreeBSD ]; then
export LANG=ja_JP.eucJP
fi
となって、シェル変数1つが余分に要るので美しくない。
よって、caseを使うのがモストエレガント。
0361名無しさん@お腹いっぱい。
2007/04/15(日) 17:37:03お前の美学はよくわかった。
0362名無しさん@お腹いっぱい。
2007/04/15(日) 17:41:220363名無しさん@お腹いっぱい。
2007/04/16(月) 19:45:580364名無しさん@お腹いっぱい。
2007/04/16(月) 19:58:240365名無しさん@お腹いっぱい。
2007/04/16(月) 20:37:020366名無しさん@お腹いっぱい。
2007/04/16(月) 22:11:510367名無しさん@お腹いっぱい。
2007/04/16(月) 22:42:570368名無しさん@お腹いっぱい。
2007/04/16(月) 23:46:400369名無しさん@お腹いっぱい。
2007/04/16(月) 23:52:070370名無しさん@お腹いっぱい。
2007/04/17(火) 00:16:38sh
0371名無しさん@お腹いっぱい。
2007/04/17(火) 00:41:10どの OS の?
0372名無しさん@お腹いっぱい。
2007/04/17(火) 00:47:25普通のbshなら.しか使えない。
HP-UXやAIXのshは中身がkshなんで。
0373名無しさん@お腹いっぱい。
2007/04/21(土) 02:11:17http://journal.mycom.co.jp/news/2007/04/20/023/index.html
0374名無しさん@お腹いっぱい。
2007/04/24(火) 14:20:370375名無しさん@お腹いっぱい。
2007/04/25(水) 21:35:530376名無しさん@お腹いっぱい。
2007/04/25(水) 21:45:150377名無しさん@お腹いっぱい。
2007/04/25(水) 21:50:110378名無しさん@お腹いっぱい。
2007/04/25(水) 21:52:270379名無しさん@お腹いっぱい。
2007/04/25(水) 21:55:090380名無しさん@お腹いっぱい。
2007/04/25(水) 22:01:270381名無しさん@お腹いっぱい。
2007/04/25(水) 22:06:430382名無しさん@お腹いっぱい。
2007/04/26(木) 00:34:12その筋質問箱は付いてますか?
0383名無しさん@お腹いっぱい。
2007/04/26(木) 04:36:490384名無しさん@お腹いっぱい。
2007/04/26(木) 10:22:570385名無しさん@お腹いっぱい。
2007/05/02(水) 17:38:53どうすればよいのでしょうか?
sedだと只それだけで一文字目の後ろに改行を挿入してホールドスペースに入れて、
y/abcdefghijklmnopqrstuvwxyz/ABCDEFGHIJKLMNOPQRSTUVWXYZ/ てなことして
また連結して改行を削除みたいな、バカ手間が掛かるので投げ出してしまった。
もっと簡潔に出来たら良いなと思うのですが・・・
0386名無しさん@お腹いっぱい。
2007/05/02(水) 17:42:25echo aaa | sed 's/\(.\)/\U\1/'
0388385
2007/05/02(水) 18:42:47しかし、なんでもコマンドを高機能なやつに入れ替えるのもどうかと思うし、
使っているOSがGNUツール群を標準装備していないのもそれなりに理由があるだろうし
高々文字列変換で新たなツールを使うのには、消極的になりますね。
0389名無しさん@お腹いっぱい。
2007/05/02(水) 19:11:07echo "abc" | perl -pe 's/(.)/\U\1/'
0390名無しさん@お腹いっぱい。
2007/05/02(水) 19:18:210391385
2007/05/02(水) 20:06:42有り難うございます。
awkのsubstrで文字位置指定して、toupperで大文字に変換ということですね。
perlの方はgsedと互換な感じの構文ですね。(てかgsedが模倣した?)
>>390
済みません。
awk: illegal statement
input record number 1, file
source line number 1
と成ります。
いずれにしてもいろいろと勉強になります。
>>386,>>389,>>390の皆さん、ありがとうございました!
0392名無しさん@お腹いっぱい。
2007/05/03(木) 02:41:58互換性維持のためなんだろうけど。
0393名無しさん@お腹いっぱい。
2007/05/03(木) 09:18:45emacsのCTRLキーはめんどい。
エロい人、設定のしかた教えてチョ。
0394名無しさん@お腹いっぱい。
2007/05/03(木) 10:08:20くだらない質問はここに書き込め!Part 52
http://pc11.2ch.net/test/read.cgi/unix/1176049968/
0395名無しさん@お腹いっぱい。
2007/05/03(木) 18:24:22bashつかえw
0396名無しさん@お腹いっぱい。
2007/05/04(金) 16:26:20例:「aaa bbb ccc ...」
各ワードに接頭辞と接尾辞を付加したいが、sedでできるかな?
例:「prefix.aaa.postfix prefix.bbb.postfix prefix.ccc.postfix ...」
ちなみにワード数は不定。
置換後の文字列長はLINE_MAX以下でないと駄目?
0397名無しさん@お腹いっぱい。
2007/05/04(金) 16:44:540398名無しさん@お腹いっぱい。
2007/05/04(金) 16:56:59$ s="aaa bbb ccc"
$ echo $s | sed -e 's/\([^ ]*\)/prefix.\1.postfix/g'
prefix.aaa.postfix prefix.bbb.postfix prefix.ccc.postfix
0399名無しさん@お腹いっぱい。
2007/05/04(金) 23:17:48tex処理を簡単にしたいと考えたのですが、これが分からずにつまっています。
0400名無しさん@お腹いっぱい。
2007/05/05(土) 00:39:090401名無しさん@お腹いっぱい。
2007/05/05(土) 06:17:46$1,$2...って9個までしか使えないんじゃ?
396は「ワード数不定」って言ってるから、10個以上もありうるだろ。
0402名無しさん@お腹いっぱい。
2007/05/05(土) 06:21:090403名無しさん@お腹いっぱい。
2007/05/05(土) 14:01:50アホー
>>400は>>399へのレスだ。
0404名無しさん@お腹いっぱい。
2007/05/05(土) 23:04:16多分、コマンドでこう起動したのと同じ扱いになっている思う。
$ xxx.sh < "abc.tex"
スクリプト側でリダイレクトされる前のファイル名を
取得する方法が分かれば、どうにかなるんじゃないかな。
0405名無しさん@お腹いっぱい。
2007/05/06(日) 01:12:45スクリプ中で使用するコマンド(今回はmogrify)がインストールされているかどうか
調べるのに手っ取り早い方法ありますか?
今のところ思いつくのは、PATHから「:」で区切って1つづつ抜き出し、その直下に
mogrifyがあるか「test -x」する方法です。
0406405
2007/05/06(日) 01:26:07whereis使えばいいか。
whereisって見つからなくても0返すんですね・・・
whereis -b mogrifya | grep -v ':$' > /dev/null
if [ $? ...
って感じかな
もっとスマートな方法ありましたらご教授ください。
0407名無しさん@お腹いっぱい。
2007/05/06(日) 01:47:140408名無しさん@お腹いっぱい。
2007/05/06(日) 02:09:28...
は?
0409名無しさん@お腹いっぱい。
2007/05/06(日) 02:18:36whichか、なるほど
解決しましたありがとうございました。
0410405
2007/05/06(日) 02:45:19&&や||の後に複数のコマンドを書くにはどうすればいいのでしょうか?
見つからなかった場合、NGと表示しexitしたいのですが、
which mogrify > /dev/null 2>&1 && echo 'bc ok' || (echo 'bc NG'; exit 1)
としてもミニシェルから抜けるだけなのです
if文で書けばいいだけなのですが、勉強の意味でも知っておきたいのです。
0411名無しさん@お腹いっぱい。
2007/05/06(日) 03:18:41echo $*
exit
}
とか定義しちゃえば?
0412名無しさん@お腹いっぱい。
2007/05/06(日) 03:28:54... || { echo NG; exit 1; }
は?
0413410
2007/05/06(日) 11:45:38いいですね。たくさんコマンド実行したいときに使わせていただきます。
>412
おー、意図したとおりになりました。今回はこちらを使わせていただきます。
ありがとうございました。
ちなみにcommand1が成功したときはcommand2、失敗したときはcommand3という時に、
command1 && command2 || command3
という書き方は問題ないですか?
動作は意図したとおりなんだけど、「|| command3」はcommand2の結果に
かかってくるように見えるのでちょっと気持ち悪い・・・
0414名無しさん@お腹いっぱい。
2007/05/06(日) 12:35:35>command1 && command2 || command3
>という書き方は問題ないですか?
command2が falseを返した時、command3まで実行されてしまうのが問題。
if - else の代わりにするなら、
command1 && { command2; true; } || { command3; }
と書けば完全。command2や command3は、; で区切って複数コマンド可能。
0415413
2007/05/06(日) 14:40:39なるほど。
command2にはechoとかしか入れてなかったので気づかなかった・・・
勉強になりました。ありがとうございます。
0416名無しさん@お腹いっぱい。
2007/05/11(金) 23:38:51上記コマンドを指定の条件をつけてwhileによる繰り返しでログを整形する
シェルをつくりたいのですが、末尾に">>ファイル名"としても指定のファイル
にリダイレクトできません。
while〜doneの後、シェルスクリプト実行時の末尾に>>を指定してもだめです。
出力は画面のままです。(通常のtail -fのイメージ)
&でバックグラウンドで実行しても同様です。
tail -f で読み取っている内容を別のログファイルに出力するにはどうすれば
よいですか?
0417名無しさん@お腹いっぱい。
2007/05/12(土) 09:13:180418名無しさん@お腹いっぱい。
2007/05/12(土) 10:06:110419名無しさん@お腹いっぱい。
2007/05/14(月) 21:13:31■ このスレッドは過去ログ倉庫に格納されています