シェルスクリプト総合 その7
■ このスレッドは過去ログ倉庫に格納されています
0001ミスターシェル
2006/09/07(木) 13:00:11スクリプトのお勉強・自慢・腕試しなどにどうぞ。
まずは注意点、リンク、地鎮祭など(>>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 でトレースしましょう。
0448名無しさん@お腹いっぱい。
2006/11/23(木) 20:09:25シリアライズして渡すのかな。ネットワーク越しにも渡せるの?
0449名無しさん@お腹いっぱい。
2006/11/23(木) 20:24:33詳しくはしらんけど
http://www.atmarkit.co.jp/fdotnet/special/powershell02/powershell02_01.html
cmd.exeより100倍ほどつかいやすそうだ。
0450名無しさん@お腹いっぱい。
2006/11/23(木) 20:26:480451名無しさん@お腹いっぱい。
2006/11/23(木) 20:37:550452名無しさん@お腹いっぱい。
2006/11/23(木) 21:59:17判定するにはどうすればいいのでしょうか?
0453名無しさん@お腹いっぱい。
2006/11/23(木) 22:14:56標準出力が、端末か、それ以外(ファイル)かの判定なら、
[ -t 1 ]
でできる。
0455名無しさん@お腹いっぱい。
2006/11/24(金) 10:34:20ちょっとわからないことがあるので質問します。
$ ABC=100
$ VAR=`echo "'\$ABC'"`
$ echo $VAR
$ '100'
(\はバックスラッシュです)
このようになるのですが、実行結果が'$ABC'ではなくて'100'になるのが理解できません。
展開するときに\によって$の効果が打ち消されると思うのですが…。
なぜこうなるかどなたかお教え下さい。お願いします。
0456名無しさん@お腹いっぱい。
2006/11/24(金) 10:38:18VAR=`echo "'\\$ABC'"`
↑
にする必要がある。バッククォートの中の \ は \\ にする。
あと、ここでは結果的に関係ないけど、
最後の echo $VAR は echo "$VAR" にした方がいいな。
0457455
2006/11/24(金) 12:20:02なるほど。$を一つ加えると結果はうまいこといきました。
でもなぜこうなるのかが納得できません。
\\$ABCの部分で一つ目の\が二つ目の\の効果を打ち消して、
$ABCは普通に変数として解釈されechoに渡されると思うのですが…。
それともバッククオートの中では何か特別なルールがあるのでしょうか?
0458名無しさん@お腹いっぱい。
2006/11/24(金) 12:26:12($を加えるんじゃなくて \ を加えるんだけどね)
>バッククオートの中では何か特別なルールがある
と言ってるとおり。
\\$ABC は、バッククォートで1回解釈される時に
\$ABC となってから、その中身が解釈される。
`echo "\\$ABC"`
の中身が解釈される時に
echo "\$ABC"
に変わってから実行されるので。
ちなみに、推奨しないが、バッククォートの代わりに $( )を使うと、
VAR=$(echo "\$ABC")
みたいに、\はひとつで良い。
0459名無しさん@お腹いっぱい。
2006/11/24(金) 12:49:19使える環境では使っとけ。
0460455
2006/11/24(金) 13:15:21ありがとうございました。やっと理解できました。
>>459
$()についても少し調べてみます。使える場合はこっちの方がすっきりしますね。
ご回答いただいた皆様本当にありがとうございました。
0461名無しさん@お腹いっぱい。
2006/11/24(金) 13:16:280462名無しさん@お腹いっぱい。
2006/11/24(金) 13:17:14ってどういう風に解釈されるんですか?
0463名無しさん@お腹いっぱい。
2006/11/24(金) 13:28:13ifconfig_args=AAA
ifn=hoge0
だとすると、
1回目の解釈で、
AAA=$ifconfig_hoge0
になるわな。
で、2回目にifconfig_hoge0の中身がAAAに代入される。
0464名無しさん@お腹いっぱい。
2006/11/24(金) 13:42:06> シェルはポータビリティ大切。
場合によるでしょ。
0465名無しさん@お腹いっぱい。
2006/11/24(金) 14:23:220466名無しさん@お腹いっぱい。
2006/11/24(金) 14:40:40という文字列から 192.168.0.1 の部分を抜き出したいのですが
どのようにすればよいのでしょうか?
0467名無しさん@お腹いっぱい。
2006/11/24(金) 14:46:23スクリプトが読みやすくなるから。
0468名無しさん@お腹いっぱい。
2006/11/24(金) 15:02:08↓を実行。標準入力から hoge.192.168.0.1 の文字列を読ませる。
IFS=. read hoge a b c d
echo $a.$b.$c.$d
>>467
そういう理由は軟弱。却下。
0469名無しさん@お腹いっぱい。
2006/11/24(金) 15:16:230470名無しさん@お腹いっぱい。
2006/11/24(金) 15:20:32???? 「読み易い」って、書き直すってこと?
そもそもポータビリティの意味を誤解してます。
0471名無しさん@お腹いっぱい。
2006/11/24(金) 18:28:430472名無しさん@お腹いっぱい。
2006/11/24(金) 18:37:28伝統的なBourne shellが/bin/shであるような環境では使えない。
Solarisとか。
0473名無しさん@お腹いっぱい。
2006/11/24(金) 20:07:45そのうちSolaris側が /bin/sh -> bash にリンクしてくる可能性すら考えられる。
0474名無しさん@お腹いっぱい。
2006/11/24(金) 21:54:310475名無しさん@お腹いっぱい。
2006/11/24(金) 22:29:240476名無しさん@お腹いっぱい。
2006/11/25(土) 02:19:05------------------------
#!/bin/csh
set i=`ps axwww|grep -i aaaa | grep -v grep | wc -l`
echo $j
------------------------
aaaaは存在しないプロセス。
この実行結果、
「0」でなくて「1」になるのはなぜですか?
0477476
2006/11/25(土) 02:23:46------------------------
#!/bin/csh
set i=`ps axwww|grep -i aaaa | grep -v grep | wc -l`
echo $i
------------------------
0478名無しさん@お腹いっぱい。
2006/11/25(土) 02:41:37パイプで一気に処理している中身を細切れにして中身を確認してみろ。
あと、pgrep があるならそっちを使った方がいい。
0480名無しさん@お腹いっぱい。
2006/11/25(土) 03:22:500481名無しさん@お腹いっぱい。
2006/11/25(土) 19:03:08取得するにはどう記述すればよいのでしょうか?
0482名無しさん@お腹いっぱい。
2006/11/25(土) 19:33:35OSによってifconfigの出力書式が違うかも知れんが、
↓みたいな感じで行ける。
ifconfig -a | sed -n 's/^\([^ ][^ ]*\).*/\1/p'
0483名無しさん@お腹いっぱい。
2006/11/25(土) 19:37:28lanscanだな。
0484名無しさん@お腹いっぱい。
2006/11/25(土) 20:30:330485名無しさん@お腹いっぱい。
2006/11/25(土) 21:05:380486名無しさん@お腹いっぱい。
2006/11/26(日) 14:56:47ミスプリ程度の間違いじゃない、ダメ本みたいだ。
こっちの書き込みによると。
↓
http://pc8.2ch.net/test/read.cgi/linux/1065059126/794
0487名無しさん@お腹いっぱい。
2006/11/27(月) 02:12:37クォートされずに*が展開されて位置パラメタに入っちゃうんだけどなんでなんでだろ。
echoとかの組み込みコマンドだと上手くクォートされるんだけどなぁ。
0488名無しさん@お腹いっぱい。
2006/11/27(月) 07:20:57どこかでクォートし忘れているって事なんだろうな。
そのスクリプトを晒してみな
0489名無しさん@お腹いっぱい。
2006/11/29(水) 21:31:140490名無しさん@お腹いっぱい。
2006/11/29(水) 21:52:550491名無しさん@お腹いっぱい。
2006/11/29(水) 22:31:42それは csh。cshは帰れ。
/bin/shでは set -f
ただし、ちゃんとクォートすれば set -f は必要ない。
0492名無しさん@お腹いっぱい。
2006/11/30(木) 21:22:530493名無しさん@お腹いっぱい。
2006/11/30(木) 21:28:40Bourne Shellでは $10以降は使えない。
bashとかkshとかでは使えるが、${10} とする必要あり。
zshでは $10 でも使える。
0494名無しさん@お腹いっぱい。
2006/11/30(木) 21:31:31${1}0と解釈されるから、${10}としないとだめ。
0495名無しさん@お腹いっぱい。
2006/11/30(木) 21:33:030496名無しさん@お腹いっぱい。
2006/11/30(木) 21:33:29Bourne Shell では ${10} でもダメ。逆に zsh では $10 でも桶。
0497492
2006/12/01(金) 00:40:19shで最後の引数を取得したかったのですが、無理なのですね。
0498名無しさん@お腹いっぱい。
2006/12/01(金) 00:48:110499名無しさん@お腹いっぱい。
2006/12/01(金) 02:24:15"Portable Shell:: Shell script portability pitfalls"
が非常に詳しくて面白い。よくこれだけまとめあげたものだ
0500名無しさん@お腹いっぱい。
2006/12/02(土) 01:04:32人間レベルポータビリティが高いということだと思う。
狭い範囲でのポータビリティを満たすためにトリックを駆使するより、
移植性のない明快な方法複数のほうが意図が読めるから未知プラット
フォームに持って行きやすい(ことがある)。
0501名無しさん@お腹いっぱい。
2006/12/02(土) 01:16:48そういう概念を表すためにわざわざ、可読性(readability)とか
保守性(maintainability)とかいう用語が存在してるでしょうに。
0502名無しさん@お腹いっぱい。
2006/12/02(土) 01:23:51移植性は似てる(生体計算機上で移植性がある=可読性)と
元レスはいいたかったんでないの?
0503名無しさん@お腹いっぱい。
2006/12/02(土) 20:53:48DEF
GHI
JKL
MNO
という文字列を
DEF
GHI
JKL
という部分をごっそり抜いて、
ABC
hogehoge1
hogehoge2
hogehoge3
JKL
としたいのですが、
何か良い案はないでしょうか?
0504名無しさん@お腹いっぱい。
2006/12/02(土) 21:06:340505名無しさん@お腹いっぱい。
2006/12/02(土) 21:08:49それくらい察してやれ。
それより、DEFやGHIなどの文字列があるものとして置換していいのか、
それとも2行目から4行目までを置換するという意味なのか、
質問の意味が複数解釈できるね。
0506名無しさん@お腹いっぱい。
2006/12/02(土) 21:11:49全くそのとおり。
0507名無しさん@お腹いっぱい。
2006/12/02(土) 22:17:15なぜ残っているのか?
0508名無しさん@お腹いっぱい。
2006/12/02(土) 22:24:14だからぁ、察してやれよ、MNOの書き間違いだろ。
0509名無しさん@お腹いっぱい。
2006/12/02(土) 22:27:07$ cat minus.awk
#!/usr/bin/gawk -f
ARGIND == 1 {
minus[ $0 ] = sprintf("hogehoge%d",FNR)
next
}
$0 in minus {
print minus[ $0 ]
next
}
{
}
$ gawk -f minus.awk (抜くキーワードを書いたファイル) (抜かれる方のファイル)
0510名無しさん@お腹いっぱい。
2006/12/02(土) 22:37:23シェルスクリプトでやりたいです。
ABC
DEF
GHI
JKL
MNO
という文字列を
DEF
GHI
JKL
という部分をごっそり抜いて、
ABC
と
MNO
の間に特定の文字列を挿入して、
ABC
hogehoge1
hogehogeXX
hogeYY
MNO
としたいのですが、
何か良い案はないでしょうか?
よろしくお願いします。
0511名無しさん@お腹いっぱい。
2006/12/02(土) 22:43:52↓ほれ、これでいいかい? シェルだけでやったよ。外部コマンドは使ってない。
while read line
do
case $line in
ABC|MNO) echo "$line";;
DEF) echo hogehoge1;;
GHI) echo hogehogeXX;;
JKL) echo hogeYY;;
esac
done < input_file
0512名無しさん@お腹いっぱい。
2006/12/02(土) 22:55:31cat aa |sed -e '/DEF/,/JKL/c\
hogehoge1\
hogehogeXX\
hogeYY'
0513名無しさん@お腹いっぱい。
2006/12/02(土) 22:56:21>>505は読んだのかw?
0514名無しさん@お腹いっぱい。
2006/12/02(土) 22:56:33お約束のcatが無駄です。あと、シェルだけでやりたいと申されております。
0515名無しさん@お腹いっぱい。
2006/12/02(土) 23:02:26> DEF
> GHI
> JKL
> という部分をごっそり抜いて、
これで人に通じると思ってるの? ゆとり世代?
1 ABCの次の行からMNOの前の行まで。
2 2行目から4行目まで
3 2行目から最終行-1まで
4 DEFまたはGHIまたはJKL
5 1文字目がAかM以外
6 1文字目がDまたはGまたはJ
...
0516名無しさん@お腹いっぱい。
2006/12/02(土) 23:06:31お客様、仕様書が曖昧で作業に入れませんのでご確認をお願いいたします。
こいういうことでしょうか?
START-STRING
AAAA
BBB
CC
END-STRING
を
START-STRING
REPLACED-1
REPLACED-2
REPLACED-3
END-STRING
とする。
START-STRINGとEND-STRINGに囲まれた複数行を置換するんでしょ?
読み取れないのは、AAAA,BBB,CCの部分は何でも良いのか、
hogehoge1,hogehogeXX,hogeYYの部分は固定なのか、AAAA,BBB,CCそれぞれに
対応させて置換するのか。
0517510
2006/12/02(土) 23:06:52ですので、
512
でほぼ解決です。
ありがとうございました。
説明がわかりにくくてご迷惑をお掛けしました。
また出直します。
0518名無しさん@お腹いっぱい。
2006/12/02(土) 23:09:28>>512は
DEFとJKLではさまれている行だろうが、どこが
> ABCの次の行からMNOの前の行まで。
なんだよ。
0519名無しさん@お腹いっぱい。
2006/12/02(土) 23:23:28「ほぼ」って言ってるから良いんじゃない?
sed -e '/ABC/,/MNO/c¥
ABC¥
hogehoge1¥
hogehogeXX¥
hogeYY¥
MNO'
程度の修正は誤差であるというくらい理解したんじゃないかna
0520名無しさん@お腹いっぱい。
2006/12/02(土) 23:32:03echo "$i"
done
Bourne Shellでも動きますか?
0521名無しさん@お腹いっぱい。
2006/12/02(土) 23:32:350522520
2006/12/02(土) 23:32:52for ((i=0; i<10; i++)) ; do
echo "$i"
done
Bourne Shellでも動きますか?
0523名無しさん@お腹いっぱい。
2006/12/02(土) 23:38:40for i in 1 2 3 4 5 6 7 8 9; do
echo "$i"
done
が最もポータブル。
seqやjotはあえて使わない。
0524名無しさん@お腹いっぱい。
2006/12/02(土) 23:52:400525名無しさん@お腹いっぱい。
2006/12/02(土) 23:57:42exprくらいは使ってもいいじゃね?
0526名無しさん@お腹いっぱい。
2006/12/05(火) 17:00:20for n in $aaa; do
echo "$n"
done
だと
"AAA
AAA"
BBB
となるのですが、これを
"AAA AAA"
BBB
とするにはどうすればいいのでしょうか?
0527名無しさん@お腹いっぱい。
2006/12/05(火) 17:12:42for n in "$aaa"
常に "$hoge" みたいに " " を付ける癖を付けろ。
0528名無しさん@お腹いっぱい。
2006/12/05(火) 17:13:44欲嫁。それだと逆に全部つながっちゃうが、、
0529名無しさん@お腹いっぱい。
2006/12/05(火) 22:58:15ゴリ押し
aaa='"AAA AAA" BBB'
while test "x$aaa" != x
do
case $aaa in
\"*\"\ *)
echo "$aaa" | sed 's,^\("[^"]*"\).*,\1,'
aaa=`echo "$aaa" | sed 's,^"[^"]*" *,,'` ;;
*\ *)
echo "$aaa" | sed 's, .*,,'
aaa=`echo "$aaa" | sed 's,[^ ]* *,,'` ;;
*)
echo "$aaa"; break ;;
esac
done
0530名無しさん@お腹いっぱい。
2006/12/05(火) 23:20:230531526
2006/12/06(水) 13:55:07精進します
0532名無しさん@お腹いっぱい。
2006/12/06(水) 16:54:030533名無しさん@お腹いっぱい。
2006/12/06(水) 17:48:45foo=$(echo $bar | sed -e 's!\(^.\).*!\1!')
0534名無しさん@お腹いっぱい。
2006/12/06(水) 18:48:23${var%%${var#?}}
0535532
2006/12/06(水) 20:52:29ありがとうございます。
自分なりに無い知恵を絞って出した答えは
foo=`perl -e "print substr('$var', 0, 1)"`
だったんですが、>>534には感動しました。
0536名無しさん@お腹いっぱい。
2006/12/06(水) 21:02:54↓がお勧め。
printf '%c\n' "$var"
0537名無しさん@お腹いっぱい。
2006/12/06(水) 21:42:29echo ${var:0:1}
0538名無しさん@お腹いっぱい。
2006/12/06(水) 21:48:39bash/ksh依存乙。
0539名無しさん@お腹いっぱい。
2006/12/06(水) 21:54:07foo=`expr "$bar" : '^\(.\)'`
0540名無しさん@お腹いっぱい。
2006/12/06(水) 22:49:45みなさん、ありがとうございます。
printfが存在を知ったり、exprの使い方がわかったりと、かなり勉強になりました。
ただ、>>539の正規表現は'\(.\)'`でいいみたいです。
^は正規表現の意味を持っていないので無視される。みたいなワーニングがでました。
0541名無しさん@お腹いっぱい。
2006/12/06(水) 23:48:500542名無しさん@お腹いっぱい。
2006/12/07(木) 00:01:29xargs を使った形に書き換えたいんですが、どう書けばいいんでしょうか?
0543名無しさん@お腹いっぱい。
2006/12/07(木) 00:25:47ていうか grep とか grep -r で済むなら xargs の出番なさそう。
0544542
2006/12/07(木) 01:04:05ファイルの文字コードがそれぞれ違う可能性があるから、
for i in *; do grep hoge $i |nkf -e >> out; done
と同じことをしたいのでした。
この場合も for の行で外部コマンドを呼んでいないから
ARG_MAX の制限はないようですね。
xargs で起動するコマンドの中にパイプやリダイレクトを
入れることはできるか、と聞きたかったのでした。
0545名無しさん@お腹いっぱい。
2006/12/07(木) 02:08:55find * ... -print | xargs -n 1 sh -c 'grep hoge $0 | nkf -e' > out
とかでもできるけど、
find * ... -print | xargs nkf -e | grep hoge > out
の方がエレガントなような。でも、 file: line 形式で grep 結果を
出したいなら、ついでに * 展開で溢れるケースも考慮して
find . ... -print | while read i; do grep hoge $i | nkf -e; done > out
しかし ARG_MAX の心配をするほどファイルがあるなら
find . ... -print | xargs perl -MJcode -lne 'print jcode("$ARGV: $_")->euc if /hoge/' > out
がちょっと長いけど速そう。
0546名無しさん@お腹いっぱい。
2006/12/07(木) 09:27:20find -print0
xargs -0
にしる、と突っ込んでおくべきなのだろうか
0547名無しさん@お腹いっぱい。
2006/12/07(木) 13:53:37仕様バグがどうこうとか。
0548名無しさん@お腹いっぱい。
2006/12/12(火) 15:03:35不安定な回線なのにどの程度不安定なのか把握できず困ってます
切断されてもすぐ再接続されるわけですが、これを記録しておきたいと思います
ping 撃ちまくるのはわかりますが、切断された時と再接続された時を記録するには
どんなスクリプト書けばいいですか?
■ このスレッドは過去ログ倉庫に格納されています