シェルスクリプト総合 その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/
0771名無しさん@お腹いっぱい。
2007/06/28(木) 18:49:360772名無しさん@お腹いっぱい。
2007/06/28(木) 19:09:45/dev/urandom はポータブルではないわけで、、
0773名無しさん@お腹いっぱい。
2007/06/28(木) 19:10:47>>771
catが無駄です。
0774名無しさん@お腹いっぱい。
2007/06/28(木) 19:45:410775名無しさん@お腹いっぱい。
2007/06/28(木) 19:53:310776名無しさん@お腹いっぱい。
2007/06/28(木) 20:07:270777名無しさん@お腹いっぱい。
2007/06/28(木) 20:09:050778名無しさん@お腹いっぱい。
2007/06/28(木) 20:13:150779名無しさん@お腹いっぱい。
2007/06/29(金) 01:27:29ていうか、間違いだらけの知識を持ってるやつほどSolarisをバカにする傾向があるな。
0780名無しさん@お腹いっぱい。
2007/06/29(金) 01:33:43Sol10 からね。Sol9 までは static link。
0781名無しさん@お腹いっぱい。
2007/06/29(金) 03:23:19の間違いでした
0782名無しさん@お腹いっぱい。
2007/06/30(土) 16:06:14747じゃないけど "$@" -> ${1+"$@"} とするのは set -u でもエラーにならない為かと
すくなくともある環境では0を除いた位置変数が未設定なら前者はパラメータ未設定エラーになるから
別に""の互換性の理由だけで${1+"$@"}にする訳ではないでしょ
0783782
2007/06/30(土) 16:14:29正) 為でもあるかと
0784名無しさん@お腹いっぱい。
2007/06/30(土) 17:05:31実際に ${1+"$@"} が使われてるのは set -u のところじゃない。
コマンドのラッパーで引数を渡す時の話。
なので、>>782 の話は当たっていない。
0785名無しさん@お腹いっぱい。
2007/06/30(土) 17:10:16空文字列が残らないようにするため。
他に、${@+"$@"} という書き方もある(あった)。
今議論になってるのはそこじゃなくて、
未だに ${1+"$@"} が必要なシェルが現存しているかどうか、
現存しているなら そのOS名は? という質問。
0786名無しさん@お腹いっぱい。
2007/06/30(土) 17:59:27「でもあるかと」と訂正してるんだが
実際 set -u 設定状態でコマンドラッパーに"$@"で渡そうとしても
$@ を評価(展開)する段階で未設定と判定された訳で
空文字列が残るかどうかなんて今時大抵解消されてるでしょ
幾つかのOS、幾つかのB系シェルを触ったことはあるけど個人的には見たことがない
ただ単純に ${1+"$@"} -> "$@" として問題ないか?と聞かれたら
set -u で問題が起きるからダメだと
0787名無しさん@お腹いっぱい。
2007/06/30(土) 18:24:29で、その set -u で問題が出たOSって何?
0788782
2007/06/30(土) 18:25:32もともと空文字列対応で今時 ${1+"$@"} とする必要はあるか?かもしれんが、
たとえその為の対応が不要になったとしても、${1+"$@"} -> "$@" にしていいことにはならんよと言いたい
理由は以下のスクリプトを実行するとエラーになる環境もあるからと
#!/bin/sh
set -u
echo "$@"
0789名無しさん@お腹いっぱい。
2007/06/30(土) 18:29:44だから、そのエラーになる環境は何なのかと。
俺のところの *BSD/Solarisとあと犬で、エラーになるのはひとつもないのだが。
0790名無しさん@お腹いっぱい。
2007/06/30(土) 18:56:23そのSolarisじゃないの? ほんとに試してみた? バージョンによるのかな?
結局互換性の最大の問題はいつもSolarisだなww
0791名無しさん@お腹いっぱい。
2007/06/30(土) 20:36:50いまのSolarisをしらない奴がいてもしょうがない
0792名無しさん@お腹いっぱい。
2007/07/03(火) 01:38:27おそーーーーーーーーいので、あまり使いたくない。
0793名無しさん@お腹いっぱい。
2007/07/03(火) 12:33:550794名無しさん@お腹いっぱい。
2007/07/04(水) 12:36:32時代はmv /bin /ms.green
0795名無しさん@お腹いっぱい。
2007/07/04(水) 22:29:33test 1000000 -lt 9999999999
の結果が正にならないんだけど。数値型の値に制限とかあるのかな?
0796名無しさん@お腹いっぱい。
2007/07/04(水) 22:31:41bashだと無問題。「正」じゃなくて「真(0)」な。
0797名無しさん@お腹いっぱい。
2007/07/04(水) 22:35:060798名無しさん@お腹いっぱい。
2007/07/04(水) 22:35:10実装依存だな。シェルによっては 31bit整数最大の 2147483647 が扱える最大値。
0799名無しさん@お腹いっぱい。
2007/07/04(水) 22:37:55ありがと。ちなみに対処法って何かあったりする?違う言語使うしかないか。
0800名無しさん@お腹いっぱい。
2007/07/04(水) 22:42:13bcに喰わせるとか。bcだともっと大きい数まで扱える。
echo '1000000 < 9999999999' | bc
bcでは、testとは逆で、真の場合1になって、それが標準出力に出る。
0801名無しさん@お腹いっぱい。
2007/07/04(水) 22:55:380802名無しさん@お腹いっぱい。
2007/07/05(木) 22:47:110803名無しさん@お腹いっぱい。
2007/07/05(木) 23:01:130804名無しさん@お腹いっぱい。
2007/07/06(金) 01:13:45(echo obase=2; echo 1.2.3.4 | tr '.' '\012') | bc | xargs -n 1 printf %.8d
0805名無しさん@お腹いっぱい。
2007/07/06(金) 01:23:00旧:(echo obase=2; echo 1.2.3.4 | tr '.' '\012') | bc | xargs -n 1 printf %.8d
新:printf %d%.8d%.8d%.8d `echo obase=2.1.2.3.4 | tr . \; | bc`
0806名無しさん@お腹いっぱい。
2007/07/06(金) 10:00:27tcsh では $ test.sh で動作するのに、
bash では $ test.sh で動作せず、
$ bash -c test.sh でも動作しませんでした。
/usr/local/bin/以下に置いたところ、
bash上の $ test.shは動作しましたが、
やはり、$ bash -c test.sh は動作しません。
この原因は、どのあたりにあるのでしょうか。
ヒントでよいので教えてください。
0807名無しさん@お腹いっぱい。
2007/07/06(金) 10:26:25(t)cshと(ba)sh では、PATHの変数が違う。本当はPATHを通し忘れているというオチ。
あるいは、.bashrcでPATHが再設定されてしまっているとか。
0808806
2007/07/06(金) 10:42:36bash上で、$ set | grep PATH すると、$HOME/binも入っているのですが、
やっぱり、PATH関係っぽいですよねぇ。
$ bash -c test.shができないのが気持ち悪いです。
ちなみに、$ $HOME/bin/test.sh では実行できました。
0809名無しさん@お腹いっぱい。
2007/07/06(金) 10:46:13setじゃなくて、printenvで確認した方がいい。
シェル変数のPATHのみセットされていて、exportされてない可能性がある。
すると、bash -c とかやった時、新しいbashにはPATHが引き継がれないから、
そういう現象が起きる。
0810806
2007/07/06(金) 10:59:27exportされないというのは盲点でした。
おっしゃるとおり、
$ printenv | grep PATH
をしたところ、$HOME/binも含まれていました。
一応、.bashrcでの設定はきいているんですよね。
ググったところ、Cygwin上では、
bashのバグ?かなにかで、bash -c がきかない現象があるようです。
遅くなりましたが、自分の環境は linux 、bash 3.1 です。
0811名無しさん@お腹いっぱい。
2007/07/06(金) 11:09:470812806
2007/07/06(金) 12:10:14席を外していまして、遅くなってすいません。
スクリーンダンプは以下のようなものでよろしいでしょうか。
実はtest.shは、navi2chインライン画像表示のためのシェルスクリプトで引数もとります。
i) bash上で実行
$ bash --verbose -c navi2ch.makethumb http://www.google.co.jp/intl/ja_jp/images/logo.gif
navi2ch.makethumb
$ sh -x navi2ch.makethumb http://www.google.co.jp/intl/ja_jp/images/logo.gif
+ tmp=/tmp/navi2ch-thumbnails
+ origfile=/tmp/navi2ch-thumbnails/www.google.co.jp/intl/ja_jp/images/logo.gif
+ thumbfile=/tmp/navi2ch-thumbnails/www.google.co.jp/intl/ja_jp/images/logo.gif.jpg
+ thumbsize=300x150
+ '[' '!' -f /tmp/navi2ch-thumbnails/www.google.co.jp/intl/ja_jp/images/logo.gif ']'
+ '[' -z '' ']'
+ /usr/bin/wget http://www.google.co.jp/intl/ja_jp/images/logo.gif -q -N -x -P /tmp/navi2ch-thumbnails
+ '[' '!' -f /tmp/navi2ch-thumbnails/www.google.co.jp/intl/ja_jp/images/logo.gif.jpg
+ thumbsize=300x150
+ '[' '!' -f /tmp/navi2ch-thumbnails/www.google.co.jp/intl/ja_jp/images/logo.gif ']'
+ '[' -z '' ']'
+ /usr/bin/wget http://www.google.co.jp/intl/ja_jp/images/logo.gif -q -N -x -P /tmp/navi2ch-thumbnails
+ '[' '!' -f /tmp/navi2ch-thumbnails/www.google.co.jp/intl/ja_jp/images/logo.gif ']'
++ identify -format %n /tmp/navi2ch-thumbnails/www.google.co.jp/intl/ja_jp/images/logo.gif
+ scene=1
+ '[' '!' -s /tmp/navi2ch-thumbnails/www.google.co.jp/intl/ja_jp/images/logo.gif.jpg -o /tmp/navi2ch-thumbnails/www.google.co.jp/intl/ja_jp/images/logo.gif.jpg -ot /tmp/navi2ch-thumbnails/www.google.co.jp/intl/ja_jp/images/logo.gif ']'
+ '[' 1 -gt 1 ']'
+ convert -sample 300x150 /tmp/navi2ch-thumbnails/www.google.co.jp/intl/ja_jp/images/logo.gif /tmp/navi2ch-thumbnails/www.google.co.jp/intl/ja_jp/images/logo.gif.jpg
+ echo -n /tmp/navi2ch-thumbnails/www.google.co.jp/intl/ja_jp/images/logo.gif.jpg
/tmp/navi2ch-thumbnails/www.google.co.jp/intl/ja_jp/images/logo.gif.jpg
0813806
2007/07/06(金) 12:13:07上の方の
$ bash --verbose -c
では、画像は取得されませんでした。
下の方の
$ sh -x
は、画像が取得されました。
0814名無しさん@お腹いっぱい。
2007/07/06(金) 12:18:15フルパス指定じゃだめ?
0815名無しさん@お腹いっぱい。
2007/07/06(金) 12:32:040816名無しさん@お腹いっぱい。
2007/07/06(金) 12:39:22test.shが問題だというから、
ちゃんと問題を切り分けて小さくしてるのかと思ったら、それかよ
.bashrcの中身も、printenv PATH そのものも、隠さず出したら?
0817806
2007/07/06(金) 12:52:00もう一度試したところ、bash上でも
$ navi2ch.makethumb 引数
が実行できました。
しかし、
$ bash -c navi2ch.makethumb 引数
は、やはり実行できませんでした。
>>814
フルパス指定だとbash上でも実行できます。
一応、回避策として、/usr/local/bin/に置いているので、具体的な支障はないのですが、
"bash -c"だけがうまく動いてくれないのはなぜだろうと不思議に思って、質問しました。
だから、本当はこんなにレスしていただくほどのことでなくて、申し訳ないです。
>>815
nkf でチェックしたところ、EUC-JPでした。(環境はja_JP.UTF-8)
cat -v で見ても、改行コードらしきものはありませんでした。
0818名無しさん@お腹いっぱい。
2007/07/06(金) 13:06:23bash -c 'navi2ch.makethumb 引数'
じゃないと駄目だろ
0819806
2007/07/06(金) 13:07:41#!/bin/sh
echo test
のようなシェルスクリプトですと、~/bin/以下に置いて、
$ bash -c test.sh
test
と実行できます。
$ printenv PATH
/home/mona/bin:/usr/local/bin:/usr/bin:/bin:/usr/bin/X11:/usr/games
ふだんbashを使っていないので、~/.bashrcはありものです。
~/.bashrc
PATH=$HOME/bin:$PATH
↑とりあえず、これだけにしました。
これでもやはり、
$ bash -c navi2ch.makethumb http://www.google.co.jp/intl/ja_jp/images/logo.gif
では画像を取得できませんでした。
"bash -c"はいろいろときびしそうですね。
0820806
2007/07/06(金) 13:12:21$ bash -c 'navi2ch.makethumb 引数'
で画像を取得出来ました。
超基本的なことに気付かず、おさわがせして申し訳ないです。
どうもすみませんでした。
ほんとうにごめんなさい。
0821名無しさん@お腹いっぱい。
2007/07/15(日) 17:20:37(sleep $ALIVE_TIME; ps $$ && kill -INT $$; sleep 1; ps $$ && kill -KILL $$)
1秒待ってもう1度だけkillしてるのはなぜなんですかね?
0822名無しさん@お腹いっぱい。
2007/07/15(日) 17:56:03よく見ろ。-INT と -KILL で違うだろ。
0823名無しさん@お腹いっぱい。
2007/07/15(日) 22:02:120824名無しさん@お腹いっぱい。
2007/07/16(月) 22:04:04sedなどを使ってやればできないことはないですが、
if文一発でできればいいなと。。
0825名無しさん@お腹いっぱい。
2007/07/16(月) 22:07:47できない。
正規表現と言ってるが、実際にはワイルドカードで十分なことが多い。
ワイルドカードなら ifの代わりに caseを使えばできる。
0826名無しさん@お腹いっぱい。
2007/07/16(月) 22:08:010827名無しさん@お腹いっぱい。
2007/07/16(月) 22:25:250828名無しさん@お腹いっぱい。
2007/07/16(月) 23:48:40...
0829名無しさん@お腹いっぱい。
2007/07/17(火) 23:10:26機能・性能に問題なきゃそれで充分だろ。
コンピュータ (て言うか、道具) なんて楽するためにあるんだから。
0831名無しさん@お腹いっぱい。
2007/07/22(日) 20:41:39作りたいのですが、実装イメージがまったくわからず困ってします。
Win相手に対してrshは使えないであろうし。。。何かいい方法をご存知の方
いらっしゃいませんか。よろしくお願いいたします。
0832名無しさん@お腹いっぱい。
2007/07/22(日) 20:44:380833名無しさん@お腹いっぱい。
2007/07/22(日) 21:38:16cygwin + sshdとかな。
0834名無しさん@お腹いっぱい。
2007/07/22(日) 23:07:17Windows が 2k / XP なら、telnet でどうぞ。
0835名無しさん@お腹いっぱい。
2007/07/22(日) 23:07:46>Win相手に対してrshは使えないであろうし
使えるよ
http://sohda.net/cygwin/treebbs/treebbs.cgi?kako=1&all=232&s=232
http://www.uhero.info/techinfo/CygwinSSH_setup/index.html
0836824
2007/07/22(日) 23:08:49結局、caseにしときました。
0837831
2007/07/22(日) 23:48:25アドバイスありがとうございます。cygwinについて
あまり知識がないので調べてみます。
>>834
telnetで可能ということは、Win上でtelnetサービスを
稼動させておけばいいということでしょうか?
明日にでも実施してみようと思います。
ありがとうございました。
0838名無しさん@お腹いっぱい。
2007/07/24(火) 01:24:43> 稼動させておけばいいということでしょうか?
手元の 2k だと特にわざわざ入れた覚えはないから、多分
標準で入ってると思う。
サービスの中に telnet ってあるはずだから、それを起動
するだけ。
常用するなら、スタートアップの種別を「自動」にする。
0839名無しさん@お腹いっぱい。
2007/07/24(火) 03:17:550840名無しさん@お腹いっぱい。
2007/07/24(火) 07:39:360841名無しさん@お腹いっぱい。
2007/07/24(火) 21:43:20#!/bin/csh
echo 検索するファイル名を入力してください:
set filename = $<
(find /* -name $filename -type f -print > /dev/tty) > & /dev/null
「hoge.txt」を検索する場合はうまくいくのですが、
ワイルドカードを用いた「hoge\*」の検索をすると検索を行わずに一瞬で終了してしまいます。
setで変数に"\*"を入力するのは不可能なのでしょうか?
0842名無しさん@お腹いっぱい。
2007/07/24(火) 21:45:140843名無しさん@お腹いっぱい。
2007/07/24(火) 21:50:39>>1
0844名無しさん@お腹いっぱい。
2007/07/24(火) 21:52:21クオートすれば?
0845名無しさん@お腹いっぱい。
2007/07/24(火) 22:31:26csh捨てろ。クズ。
0846名無しさん@お腹いっぱい。
2007/07/24(火) 22:34:460847名無しさん@お腹いっぱい。
2007/07/24(火) 22:44:48すんまそん、会社でシェルシェル言ってるもんで癖がついてました。
クオート試してみることにします。
cshは会社の都合なんすよ
0848名無しさん@お腹いっぱい。
2007/07/25(水) 06:21:28最初の行に
sh
と書きゃいいじゃん。
0849名無しさん@お腹いっぱい。
2007/07/25(水) 12:34:58実社会で勉強してね
cshからshに変えるのは容易ではないのだよ
シェルスクリプトは1人で作るのではないし
1つの会社で作るものでもない
0850名無しさん@お腹いっぱい。
2007/07/25(水) 14:06:37いくつもの会社にまたがって何人もの人がよってたかって
>>841みたいなスクリプトを書いているわけだな。すげぇ。
つーか、csh を使うにしてもせめて #!/bin/csh -f にしろよ。
0851名無しさん@お腹いっぱい。
2007/07/25(水) 14:13:50の例は、わざわざcshの欠点が目立つような典型例だな。
ワイルドカードの展開の問題とか、
findのstderrを捨てるために苦労してるところとか。
そもそも、findをインタラクティブに実行させるようなスクリプトは、
システム管理用スクリプトじゃない。初心者ユーザー向けに作ったものだろ。
こんなの、/bin/shで書き直せば一発。
>>841 が抱えているcshでの問題も一気に解決する。
>>849 へ。
みんな実社会で「cshスクリプトは使わない」が常識だといってるんだよ。
>>849 は自分の会社名を公表しない方がいいよ。会社の不名誉になるからね。
0852名無しさん@お腹いっぱい。
2007/07/25(水) 14:15:450853849
2007/07/25(水) 14:57:09ま、会社名を挙げる気は無いけど、
おれんとこでは全部のスクリプトがcshだし
スクリプトの作成・保守を依頼している会社も
cshで作るようになっている。
OSが古いもの・新しいものが混在しているのも
理由ではあるが、
常識など、会社によって違うものだろ。
シェルがダメなのは分かるが、
シェルを変えろとか無理な話だ
0854名無しさん@お腹いっぱい。
2007/07/25(水) 14:59:19実行しても何も起きません。
どうすればよいのでしょうか?
また、linuxのbashみたいに↑↓キーで履歴表示をしたいのですが、
kshで実現可能でしょうか?
0855名無しさん@お腹いっぱい。
2007/07/25(水) 15:07:05> OSが古いもの・新しいものが混在しているのも理由ではあるが、
/bin/shが存在しないUnixなんて無いんじゃないのか
> シェルがダメなのは分かるが、シェルを変えろとか無理な話だ
へえ。cshがダメなのははるか前からの常識なのに、誰もその
やりかたを変えようとせず、変えるのは「無理」だと思ってるんだ。
しかも古いスクリプトだけでなく、新しいスクリプトもわざわざ
cshで書き、ゴミを製造し続けていると。
勿論、C++やJavaやPythonなんて誰も使わないんだろうな。
ソース管理は未だにCVSでさえない、いやソース管理なんて概念もないね?
お前さんがアホなのか、会社がアホなのかは知らないが、実に見事に
アホだな。
0856名無しさん@お腹いっぱい。
2007/07/25(水) 17:35:21そんなに必死に弁解しなくていいよ
0857名無しさん@お腹いっぱい。
2007/07/25(水) 20:55:10だってさ、起動スクリプトとかOSに元々内蔵されてるスクリプトはshだよね?
そういうスクリプトを修正する時も、cshで書き直してるの?
0858名無しさん@お腹いっぱい。
2007/07/25(水) 23:13:350859名無しさん@お腹いっぱい。
2007/07/25(水) 23:54:41おまけに最近スクリプトでsql実行してるのを、「COBOLで実装しろ」って指摘された…
SQLの使い方調査があって、スクリプトが調査範囲から漏れたからだと。
0860名無しさん@お腹いっぱい。
2007/07/26(木) 00:27:56つまりOSインストール時からあるスクリプトも全部cshで書き直してるって事か?
0861名無しさん@お腹いっぱい。
2007/07/26(木) 00:40:110862名無しさん@お腹いっぱい。
2007/07/26(木) 01:32:53なんでcsh使うことになったの?
0863名無しさん@お腹いっぱい。
2007/07/26(木) 04:05:17csh好きな勉強不足の若者の脳内会社なんだから許してやれよw
0864名無しさん@お腹いっぱい。
2007/07/26(木) 10:27:50皆さんがんばってね
0865名無しさん@お腹いっぱい。
2007/07/26(木) 14:51:51cshで書かれたシェルスクリプトがいっぱい付いてくるから
仕方が無い
0866名無しさん@お腹いっぱい。
2007/07/26(木) 14:57:19戻り値チェックするくらいのもんだからな。
0867名無しさん@お腹いっぱい。
2007/07/26(木) 17:05:580868名無しさん@お腹いっぱい。
2007/07/26(木) 18:40:090869名無しさん@お腹いっぱい。
2007/07/27(金) 00:19:380870名無しさん@お腹いっぱい。
2007/07/27(金) 01:30:36俺はシェルスクリプトがないと会社で生きて行けんが。
■ このスレッドは過去ログ倉庫に格納されています