シェルスクリプト総合 その13
■ このスレッドは過去ログ倉庫に格納されています
0001名無しさん@お腹いっぱい。
2008/10/16(木) 00:48:38スクリプトのお勉強・自慢・腕試しなどにどうぞ。
まずは注意点、リンク、地鎮祭など(>>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 でトレースしましょう。
前スレ
シェルスクリプト総合 その12
http://pc11.2ch.net/test/read.cgi/unix/1218277263/
0402名無しさん@お腹いっぱい。
2008/12/03(水) 12:10:38それだと、スペース入りファイル名の問題が再発します。
0403名無しさん@お腹いっぱい。
2008/12/03(水) 12:15:50> =の左辺では、単に「変数名」と書くだけで変数名とみなされる
> のに、右辺では単なる文字列とみなされてしまう、
Cでも、左辺値と右辺値は区別されている。左辺値になれない式が存在する。
Cでも文脈によって意味が変わるということだ。
0404名無しさん@お腹いっぱい。
2008/12/03(水) 12:18:43「文脈で変動している」
と言っても、コンピュータにはとっては筋が通っており、解析自体が
不可能なわけではない。
しかし、問題は、そのような例外が多くあると、シェルスクリプトの
初心者が他人のスクリプトを見た時に混乱しやすいことと、意味を調べ
たくても、検索し辛いことにある。
・「(英)単語」であれば検索しやすいが、記号だとしづらい。
・文脈依存性が強いと、調査に時間がかかる。
0405名無しさん@お腹いっぱい。
2008/12/03(水) 12:20:36変動はしているが、x に x + 1 を代入したい時は素直に、
x = x + 1;
と書けばよい。一方、シェルスクリプトでは、複雑なことになる。
evalやらを使うんだと思うが。
0406名無しさん@お腹いっぱい。
2008/12/03(水) 12:25:050407名無しさん@お腹いっぱい。
2008/12/03(水) 12:28:30もともとシェルはコマンド実行ユーティリティだから
普通のプログラム言語っぽく使えるだけて満足だがなあ
つ−か、書き方によってはサブシェルになる方が問題かと
0408名無しさん@お腹いっぱい。
2008/12/03(水) 12:28:47> evalやらを使うんだと思うが
↑
ハイハイ、あなたの理解度はその程度ですねw
evalは使いません。勉強してから出直してね。
0409名無しさん@お腹いっぱい。
2008/12/03(水) 12:30:28bash限定でいいなら、
((x = x + 1))
って素直に書けるよ。
さらに、
((x++))
という書き方もできる。
0410名無しさん@お腹いっぱい。
2008/12/03(水) 12:33:52別に自慢のために書いているんじゃなく、シェルスクリプトの問題点を
考えたいと思っているだけなのに。
0411名無しさん@お腹いっぱい。
2008/12/03(水) 12:36:490412名無しさん@お腹いっぱい。
2008/12/03(水) 12:37:36> しかし、問題は、そのような例外が多くあると、シェルスクリプトの
> 初心者が他人のスクリプトを見た時に混乱しやすいこと
↑
自分が理解できていないことを「例外」ということにしたいのですね、わかります
0413名無しさん@お腹いっぱい。
2008/12/03(水) 12:37:490414名無しさん@お腹いっぱい。
2008/12/03(水) 12:41:33勉強不足であろうがなかろうが、言語に問題があるかどうかが
議論したかったのに。
0415名無しさん@お腹いっぱい。
2008/12/03(水) 12:50:40そういうことを本気で議論したいなら、
まずは一通りでもその言語(シェルスクリプト)を習得してからじゃないと。
(批判しているつもりが、勉強不足のせいで批判ポイントがずれているしね)
>>386 のような間違いだらけの文章を書いたのでは、
どこに行っても誰にも相手にしてもらえないよ。
0416名無しさん@お腹いっぱい。
2008/12/03(水) 13:04:01$(cat file) と "$(cat file)"
の違いだけじゃん。
0417名無しさん@お腹いっぱい。
2008/12/03(水) 13:20:13面倒だから、間違い箇所を挙げるだけにするよ。
(1)まず、
>"$(cat file)" が `cat file`
(2)
>本やネットでも検索しようがないから。
本の索引で記号は検索できる。
あと、man sh (man bash) でも lessで検索できる。
(3)
>文脈で記号の意味が変動
しない
(4)
>ならば、`cat $variable` ではどちらか
そもそも、' 'と ` ` の意味を理解していない。
「どちらか」という問い自体が間違い
(5)
>推測不可能だ
マニュアルに載ってる。推測なんて不要。
0418名無しさん@お腹いっぱい。
2008/12/03(水) 13:20:46>PATH = xxxx:$PATH
スペースを入れちゃ駄目
(7)
> =があれば代入だとみなされる
というわけでもない
(8)
>シェル変数は小文字、環境変数は大文字
とは限らない
(9)
>シェル変数にpathがあり
そんなものない
ざっと挙げても9箇所間違い。もっと細かく見ればもっと間違いが出てくるだろう。
0419名無しさん@お腹いっぱい。
2008/12/03(水) 13:23:160420名無しさん@お腹いっぱい。
2008/12/03(水) 13:32:21単語である点として間違っているのと、議論の筋である線として間違っているのは別だろうに。
しかし>>386の言うような問題があるとして、今更仕様を変えられるわけでもなし…
他人に理解しにくいように書けというならばどの言語でもできるのだから、
"シェルスクリプトにおいて可読性の高めるためにはどのようにすればよいか?また、スレの住人はどのような工夫をしているか?"
というのが妥当な議題になるのか。
0421名無しさん@お腹いっぱい。
2008/12/03(水) 13:33:30>(2)
>>本やネットでも検索しようがないから。
>本の索引で記号は検索できる。
>あと、man sh (man bash) でも lessで検索できる。
検索できません。試しに、Googleで、$(cat file)や、
"$(cat file)"を検索する方法を明示してください。
>(3)
>>文脈で記号の意味が変動
>しない
しまくってます。C/C++/Java の様な言語と比べれば雲泥の差です。
詳しくは知りませんが、シェルスクリプトは、「文脈依存文法」とでも
呼ぶべき様な言語に属すると思います。
>(4)
>>ならば、`cat $variable` ではどちらか
>そもそも、' 'と ` ` の意味を理解していない。
>「どちらか」という問い自体が間違い
理解してます。その上で、バッククォーテーションの中で、$varが展開さ
れるかどうかは、マニュアルを読まないことには推測できないと言っている
わけです。
0422名無しさん@お腹いっぱい。
2008/12/03(水) 13:35:13>>推測不可能だ
>マニュアルに載ってる。推測なんて不要。
マニュアルに載っていたとしても、推測できる言語と出来ない言語では、
分かりやすさが全く違ってきます。Makefile なども非常に頭の良い人でも
マニュアルを読まずして推測するのが難しい構文が多いのです。
マニュアルを読まずして推定できる、と言う言語仕様を開発するのは、
ある種の力量が必要なのです。
>(7)
>> =があれば代入だとみなされる
>というわけでもない
シンボル名=
で、代入と見なされるはずです。違うというなら、それはそれで
文脈依存性が強い言語と言うことになりますが。
>(8)
>>シェル変数は小文字、環境変数は大文字
>とは限らない
>(9)
>>シェル変数にpathがあり
>そんなものない
これについては、シェルの種類に依存するようです。
0423名無しさん@お腹いっぱい。
2008/12/03(水) 13:40:33文章の間違いって、許容できるのは漢字の間違いとか、そういうレベルのことだろ。
議論の筋の根幹にかかわる部分で理解不足による間違いがあれば、
議論をしてもポイントずれまくりで話にならない。
シェルスクリプトについて教えて欲しいなら、
素直に「教えてください」って言えよ。
議論ふっかけを装って、実は自分の理解不足なところを質問したいだけだろ?
0424名無しさん@お腹いっぱい。
2008/12/03(水) 13:43:38> 理解してます。その上で、バッククォーテーションの中で、$varが展開さ
> れるかどうかは、マニュアルを読まないことには推測できない
↑
それは「バッククォーテーションを理解していない」という。
そもそも ` ` と $var の展開は別物なんだから。
0425386
2008/12/03(水) 13:46:49>素直に「教えてください」って言えよ。
>議論ふっかけを装って、実は自分の理解不足なところを質問したいだけだろ?
違います。シェルスクリプトの問題点を整理し、新しい言語仕様の策定に
役立てたいと思っているだけです。確かに、現状のシェルスクリプトにつ
いての理解も不十分ではありましたが。その点については反省します
「文脈依存性」について、考えてみましょう。
`xxx`は、コマンドxxxを実行し、その標準出力の結果をそこに展開する、
という意味です。しかし、これも、もし、
expand_stdout("command aaa${var}bbb")
expand_stdout('command aaa${var}bbb')
と書くのであったならば、マニュアルを読むことなく、${var}が展開
されるかどうかが一目瞭然であったんです。
これを、バッククォーテーション一つで実現しようとしたこと自体に
問題があるのではないでしょうか?
0426386
2008/12/03(水) 13:49:03仕様を知りませんが、
`"command ${var}"`
で${var}を展開できるんですか?すか。
だとすると、今度は、double quotation が back quatationの中で
「展開される」
という事実がマニュアルなしでは推定できません。
0427386
2008/12/03(水) 13:54:37--->expand_param("文字列")
「文字列をコマンドとして実行し標準出力を展開する」
--->expand_stdout("文字列")
もし、こういう仕様であったなら、
"$(cat file)" は、丁度、
expand_param(expand_stdout("cat file"))
というように書け、なんの推定の余地もないはずだったのです。
もしこうであったなら、マニュアルを読まなくても、色々な展開パターン
を好きなように書けます。展開の順序も推定の余地がありませんし。
0428386
2008/12/03(水) 13:55:32「文字列中のパラメータ・ラインとして展開する」
--->
「文字列をパラメータ・ラインとして展開する」
0429386
2008/12/03(水) 13:58:41を実行したい、という場合ならば、
expand_stdout(expand_param("command $var"));
で書けるわけです。この場合、""や``の中で色々な記号が「展開」される
かどうかを憶えている必要もないし、マニュアルを読み込む必要もありま
せん。
0430名無しさん@お腹いっぱい。
2008/12/03(水) 13:58:46記述量を多くして読みやすくするための言語じゃない。
前提知識なしに読みやすくしたいなら、外部コマンドを呼ぶにも、
execute_external_command("ls", "-l");
と書くような言語を作らないといけない。
他の適当なスクリプト言語使っとけと。
0431名無しさん@お腹いっぱい。
2008/12/03(水) 13:58:57Google で検索可能だなんて >>417 は言ってないよ。
0432386
2008/12/03(水) 14:01:35「本やネット」で検索できるかどうかを論じているわけです。
ネットで検索する場合、Googleで検索可能でないと、いったい何で
検索可能なんでしょうか?
記号単独の検索でも難しいのに、文脈依存性が高い言語仕様の場合、
探し出すのに非常に時間がかかります。
0434名無しさん@お腹いっぱい。
2008/12/03(水) 14:04:480435386
2008/12/03(水) 14:05:180436名無しさん@お腹いっぱい。
2008/12/03(水) 14:05:46100歩ゆずって、plan9 の rc だ。
もっと勉強しろ。
0438名無しさん@お腹いっぱい。
2008/12/03(水) 14:26:00お前の頭の悪さには負けるよ。
仕様も理解せずに批判しようと言うのだから。
レベルが低すぎて話にならない。
0439名無しさん@お腹いっぱい。
2008/12/03(水) 14:28:0330分で読めるsh(1)さえ読まずに、
批判しようと言うのが間違ってる。
0440名無しさん@お腹いっぱい。
2008/12/03(水) 14:34:08一般人は30分では読めん。
0441名無しさん@お腹いっぱい。
2008/12/03(水) 14:36:18閉鎖的であっただからか。
0442名無しさん@お腹いっぱい。
2008/12/03(水) 14:54:18無理に延命させるスレということでいいよ。
まったく別の言語仕様のシェルがほしいって話なら
perl でも emacs でも好きなものを chsh で設定すればいいじゃないか。
このスレでやることじゃない。
0443名無しさん@お腹いっぱい。
2008/12/03(水) 14:55:45素人でも頑張ればなんとかかんとか使える敷居の低さが普及の最大の要因だと思う。
ところでシェルスクリプトとは全く別のスクリプトの策定に協力しろなどというのはこのスレの趣旨とは違うだろ。
シェルスクリプトで可読性が低いな、とか不便だな、とか思ったことはありませんか?と情報を集めるならばまだしも、議論を求めるのはお門違いだ。
0444名無しさん@お腹いっぱい。
2008/12/03(水) 15:34:04シェル作ろうスレ立てて、
絶対投げ出さずみんなをひっぱっていくくらいの気概があるなら、
ついていくやつもいるんじゃなかろうか。
ここはそういう場ではないということだ。
0445名無しさん@お腹いっぱい。
2008/12/03(水) 19:17:17劣悪な状態だなんて思ってる人いる?
がっちり書きたいならperlなりなんなり使えばいいだけだし
0446名無しさん@お腹いっぱい。
2008/12/03(水) 19:54:280447名無しさん@お腹いっぱい。
2008/12/03(水) 22:19:35Windowsとの親和性が高いことの他にだが。
0448名無しさん@お腹いっぱい。
2008/12/03(水) 22:29:27コマンドインタプリタ/スクリプティング用途には起動が致命的に重くて
俺には使い道が無い
.NETやJava系は、そのプロセス起動の重さのお陰で
Unix流のツールボックスアプローチ的使い方には
全く不向きなプラットフォームだと思っている
計算性能においてははるかに劣るLLの方が、マシ
0449名無しさん@お腹いっぱい。
2008/12/03(水) 22:55:000450>>381
2008/12/03(水) 23:37:48ありがとうございます。
ふと思ったのですが、数値比較の場合、どのようになりますか?
----- コーテーションなし------------------------------------
$ cat hoge.sh
#!/bin/sh
hoge=
if [ $hoge -eq 10 ]; then
echo hello
fi
$ ./hoge.sh
[: -eq: unexpected operator
----- コーテーションあり--------------------------------------
$ cat hoge.sh
#!/bin/sh
hoge=
if [ "$hoge" -eq 10 ]; then
echo hello
fi
$ ./hoge.sh
[: : bad number
どっちにしても、エラーがでる、、、、orz
0451名無しさん@お腹いっぱい。
2008/12/04(木) 00:09:08おそるべきはニートだなと改めておもった
0452名無しさん@お腹いっぱい。
2008/12/04(木) 00:45:33一体何をしたいんだ?
純正シェルでも使えるのか知らんが、適当にデフォルトの値(ここでは0)を決めておいて
${hoge:-0}
と書けば、hoge が宣言されていないか、宣言されているが値が無いときに 0 に評価される
0453名無しさん@お腹いっぱい。
2008/12/04(木) 01:03:380454>>450
2008/12/04(木) 07:03:48おぉ、ありがっとん。
0455名無しさん@お腹いっぱい。
2008/12/04(木) 08:35:29sedの置き換え文字列に改行を含む場合、以下のようになります。
hoge() {
echo "start"
sed 's/hoge/foo \
bar/'
echo "end"
}
しかし、これですと、インデントがそろわず、不恰好になります。
インデントをずらす方法はないでしょうか?
0456名無しさん@お腹いっぱい。
2008/12/04(木) 10:33:510457名無しさん@お腹いっぱい。
2008/12/04(木) 10:37:11sedの内容を、シェル関数外でシェル変数(regex)とかに代入しておいて、
シェル関数内では sed "$regex" と書けば良い。
>>456
気にしろ
0458名無しさん@お腹いっぱい。
2008/12/04(木) 13:03:26あと最近は、OSXのやつらの台頭でスペース文字含みのファイルとか
余裕で出てくるようになったからその辺を気にしないといけなくなった
0459名無しさん@お腹いっぱい。
2008/12/04(木) 13:18:18regewxを定義するところが不恰好になるだけじゃ…
>>455
きにすんな
0460名無しさん@お腹いっぱい。
2008/12/04(木) 13:28:40bar/' | sed -e 's/ //g'`
二行目はhtab使ってるから注意しな。
0461名無しさん@お腹いっぱい。
2008/12/04(木) 13:39:41マニュアルを読まないと推定できないような言語じゃなくて。
0462名無しさん@お腹いっぱい。
2008/12/04(木) 13:47:47だから、インデントのいらない階層であるところの「シェル関数外」で
regexに代入するんだろ。
もともとインデントゼロのスコープに書くんだから、
不恰好にならない。
頭使え。
0463名無しさん@お腹いっぱい。
2008/12/04(木) 14:01:24N='
'
hoge() {
sed "s/hoge/foo ${N}bar/"
}
>>456
気にしろw
0464名無しさん@お腹いっぱい。
2008/12/04(木) 14:06:14なぜかこの環境から書き込むとバックスラッシュが化けるので、、
ダブルクォートの中だからバックスラッシュは2つね。
0465名無しさん@お腹いっぱい。
2008/12/04(木) 23:01:38ってほどでもないかw
0466名無しさん@お腹いっぱい。
2008/12/06(土) 10:43:43気にすんな
0467名無しさん@お腹いっぱい。
2008/12/06(土) 11:20:20>マニュアルを読まないと推定できないような言語じゃなくて。
シェルスクリプトすら書けないんなら、この業界から足をあらってカタギになれ!
0468名無しさん@お腹いっぱい。
2008/12/06(土) 11:29:540469名無しさん@お腹いっぱい。
2008/12/06(土) 11:43:31マニュアルを読んでも推定しかできないんなら、カタギになれ!
0470名無しさん@お腹いっぱい。
2008/12/06(土) 12:07:180471名無しさん@お腹いっぱい。
2008/12/06(土) 13:06:54何言ってんだおまえ
0472名無しさん@お腹いっぱい。
2008/12/06(土) 21:49:180473名無しさん@お腹いっぱい。
2008/12/06(土) 22:13:05FOUND=false
netstat -lnx | grep /tmp/ssh | while read i; do
set -- $i
if [ -O "$9" ]; then
SSH_AUTH_SOCK=$9; export SSH_AUTH_SOCK
SSH_AGENT_PID=`echo $9 | cut -d. -f2`; export SSH_AGENT_PID
FOUND=true
fi
done
$FOUND || eval `ssh-agent -s`
シェルスクリプトでwhileはサブシェルなのだと何回やれば判るのかと・・・
0474名無しさん@お腹いっぱい。
2008/12/06(土) 22:27:370475名無しさん@お腹いっぱい。
2008/12/07(日) 00:28:050476名無しさん@お腹いっぱい。
2008/12/07(日) 00:37:510477名無しさん@お腹いっぱい。
2008/12/07(日) 01:58:210478名無しさん@お腹いっぱい。
2008/12/07(日) 03:53:340479名無しさん@お腹いっぱい。
2008/12/07(日) 15:29:090480名無しさん@お腹いっぱい。
2008/12/07(日) 17:40:520481名無しさん@お腹いっぱい。
2008/12/07(日) 19:13:550482名無しさん@お腹いっぱい。
2008/12/07(日) 20:46:490483名無しさん@お腹いっぱい。
2008/12/07(日) 20:50:130484名無しさん@お腹いっぱい。
2008/12/08(月) 03:56:00もう質問者いなそうだけどOSによっては改行として"^J"を使う手も
vi だと挿入モードの時、ctrl + v してから ctrl + j で入力
sed 's/hoge/foo \
bar/'
を
sed 's/hoge/foo ^Jbar/'
OSを選ぶので汎用性もないしオプションなしのcatやmoreだと"^J"の部分が
改行になるかもと色々問題ありまくりだけど場合によってはこんな方法もってことで
あと"^J"使える場合はヒアドキュメントも cat - <<END^JOK^JEND と1行で書ける
意味あるかどうか人それぞれだろうけど
0485名無しさん@お腹いっぱい。
2008/12/08(月) 11:22:53改行扱えないsedは存在しないんじゃないかな。
0486名無しさん@お腹いっぱい。
2008/12/08(月) 11:45:11そういう問題じゃないんだよ。
もとの質問の >>455 よくよめ
UNIX環境では ^J は改行そのものなので、
>>484 の方法では全く解決にならない。
0487名無しさん@お腹いっぱい。
2008/12/08(月) 12:06:41ぶっちゃけPerl使えばいいんじゃね?
0488名無しさん@お腹いっぱい。
2008/12/08(月) 21:51:54だからちゃんと見てやれよw
0489487
2008/12/08(月) 22:49:22あ、ごめん、俺のは横レスだから>>485あたりの人とは別人
>>455見る限りでは、GNUではないsedだから置換文字列に\nが
使えなくてああいう書き方をしてて、それがいやだってんでしょ?
そう読めたけど
0490名無しさん@お腹いっぱい。
2008/12/08(月) 22:53:55俺には^Jをどう表示するかというエディタの問題に思えるが…
>>456>>459>>466>>487
禿同
0491名無しさん@お腹いっぱい。
2008/12/09(火) 04:04:02分かる方が居たらお願いします。
http://pc11.2ch.net/test/read.cgi/linux/1210999497/611
0492名無しさん@お腹いっぱい。
2008/12/09(火) 10:23:370493名無しさん@お腹いっぱい。
2008/12/09(火) 10:29:170494名無しさん@お腹いっぱい。
2008/12/10(水) 22:03:11if [ 変数 != 文字列A ] | [ 変数 != 文字列B ];then
ERROR処理
fi
ネスト無しでやる場合これって有りかな?
数値じゃないから-oは使えないんでこれしかやりようがなかったんすけど
有識者の方頼みます
0495名無しさん@お腹いっぱい。
2008/12/10(水) 22:08:05| じゃなくて || な。
↓
if [ 変数 != 文字列A ] || [ 変数 != 文字列B ]; then
あと、数式じゃなくても -o 使えるよ。
↓
if [ 変数 != 文字列A -o 変数 != 文字列B ]; then
-o が使えないのは [[ ]] の場合。その場合は、
↓
if [[ 変数 != 文字列A || 変数 != 文字列B ]]; then
0496名無しさん@お腹いっぱい。
2008/12/10(水) 22:12:30意味的には、やりたいことは -o (||) じゃなくて、-a (&&) のような気がする。
0497名無しさん@お腹いっぱい。
2008/12/10(水) 22:14:20この場合、if [ ] よりも case を使った方が簡単に書けるよ。
0498名無しさん@お腹いっぱい。
2008/12/10(水) 22:17:11実は|でいけたんっすよ
テストケース作って試したんですけど、一応正常系も異常系も通りました。。。
-oについては明日また試してみます
ありがとうございます
0499名無しさん@お腹いっぱい。
2008/12/10(水) 22:20:22ミドルのステータスチェックを増やしたんすよ
元が共通シェルだから余り大幅な変更は出来ないんです
caseの方が良いのはその通りなんですが・・
0500名無しさん@お腹いっぱい。
2008/12/10(水) 22:21:10行けたように勘違いしただけで正常に変数チェックされていない。
| は パイプ。パイプでは、A | B の場合、Bが真か偽かしかチェックされない。
(Aは無関係になる)
具体的には、
if [ 変数 != 文字列A ] | [ 変数 != 文字列B ];then
だと、
[ 変数 != 文字列B ] だけがチェックされて、
[ 変数 != 文字列A ] の結果は無視される。
それをたまたま「行けた」と勘違いしただけ。
全然行けてないので勘違いしないように。
0501名無しさん@お腹いっぱい。
2008/12/10(水) 22:25:43確かに文字列Bの追加したステータスチェックは通ったけど、文字列Aは怪しいですね。
了解しました
参考にさせていただきます。
>>495
>>496
>>497
>>500
有難うございました。
■ このスレッドは過去ログ倉庫に格納されています