シェルスクリプト総合 その19
■ このスレッドは過去ログ倉庫に格納されています
0001シェルスクリプトライター
2011/12/10(土) 20:06:40.38スクリプトのお勉強・自慢・腕試しなどにどうぞ。
□お約束
・特記なき場合はBourne Shell(/bin/sh)がデフォルトです。
bash/zsh/ksh/ashなどに依存する場合は明示しましょう。
Linuxユーザは/bin/shの正体がbashなので特に注意。
FreeBSDユーザは/bin/shの正体がashなので注意。
v7 shに一番近くて、現役のshは、OpenSolaris由来のheirloom sh。
http://src.opensolaris.org/source/xref/onnv/onnv-gate/usr/src/cmd/sh/
http://heirloom.sourceforge.net/sh.html
・csh/tcshのシェルスクリプトは推奨されません。
(理由は「csh-whynot」でググれ)
・UNIXにはシェルスクリプトに便利な小さなコマンドがいろいろあります。
manや参考リンクを見ましょう。
aproposないしはman -kでそれらしい単語による簡単な検索もできます。
・シェルスクリプトのことをシェルってゆーな
・シェルで使えるワイルドカード等は正規表現ではありません。
正規表現の話題はスレ違い(正規表現スレへ)
□初心者へのアドバイス:
・適した道具を判断するのも頭の重要な使い方。シェルスクリプトよりも
RubyやPerlの方が適した仕事には素直にそちらを使いましょう。
・知らないコマンドが出てきたらmanを引きましょう。
・思い通りに動かないときは、まずは sh -x でトレースしましょう。
前スレ
シェルスクリプト総合 その18
http://hibari.2ch.net/test/read.cgi/unix/1308195527/
次スレは >>970 で。
0640名無しさん@お腹いっぱい。
2012/03/22(木) 21:29:35.34かっこういい
0641名無しさん@お腹いっぱい。
2012/03/23(金) 22:05:53.20・ファイルを指定フォルダ内へコピー
・同名ファイルが存在する場合は、
既に存在するファイル名を「ファイル名 (1)」へ変更し、
既に「ファイル名 (1)」が存在しているのであれば、
それをさらに「ファイル名 (2)」へ変更し、、、(繰り返し)
というシェルスクリプトを書きたいんですが、
既に「ファイル名 (*)」が存在する場合に*の数字を
どのようにすれば知ることができますか?
*さえ知れたらexprを使えば出来るとは思うのですが...
アドバイスお願いします。
0642名無しさん@お腹いっぱい。
2012/03/23(金) 22:27:01.94逆に考える。
変数iとかに1を入れておいて、
"ファイル名($i)" が存在したら
iをインクリメントして "ファイル名($i)" にリネームする。
0643名無しさん@お腹いっぱい。
2012/03/23(金) 22:28:54.510644名無しさん@お腹いっぱい。
2012/03/23(金) 22:32:05.36i=1; while [ -f "file($i).txt" ]; do i=`expr $i + 1`; done; echo $i
0645名無しさん@お腹いっぱい。
2012/03/23(金) 22:52:22.200646名無しさん@お腹いっぱい。
2012/03/23(金) 22:56:33.590647名無しさん@お腹いっぱい。
2012/03/24(土) 01:18:46.160648名無しさん@お腹いっぱい。
2012/03/24(土) 10:30:46.53つまりアヌスを知りたいと……教えてあげよう
0649名無しさん@お腹いっぱい。
2012/03/24(土) 18:23:06.74実機が無いので試せないけど・・・
第一引数:コピー対象ファイル名(絶対パス指定)
第二引数:コピー先ディレクトリパス
#!/bin/ksh
FILE_NAMEW=`basename ${1}`
COUNT=1
TARGET_FILE=${2}/${FILE_NAME}
if [[ -e ${TARGET_FILE} ]]; then
while true
do
if [[ -e "${TARGET_FILE}(${COUNT})" ]]; then
COUNT=`expr ${COUNT} + 1`
else
cp ${1} ${TARGET_FILE}(${COUNT})
break
fi
done
else
cp ${1} ${TARGET_FILE}
fi
exit 0
0650名無しさん@お腹いっぱい。
2012/03/24(土) 18:40:54.370651名無しさん@お腹いっぱい。
2012/03/24(土) 18:48:18.47第1引数: コピー元ファイル(絶対/相対path関係なし)
第2引数: コピー先ファイル(ディレクトリではない)
#!/bin/sh
FILE=$2
if [ -f "$FILE" ]; then
i=1
while [ -f "$FILE($i)" ]; do
i=`expr $i + 1`
done
FILE="$FILE($i)"
fi
cp "$1" "$FILE"
0652641
2012/03/24(土) 21:11:20.76642さんの意見を参考に書いていたら、
651さんとほぼ同じようなものが書けました。
勉強になりました!
0653名無しさん@お腹いっぱい。
2012/03/24(土) 22:20:08.70みんなも勉強になるんだけどなぁ
0654名無しさん@お腹いっぱい。
2012/03/24(土) 22:34:45.96[ -e "$2" ]&&{ i=0; while [ -e "$2($((++i)))" ];do :;done; set "$1" "$2($i)";}
cp "$1" "$2"
0655名無しさん@お腹いっぱい。
2012/03/25(日) 10:39:31.72ってどういう風に解釈すれば良いんですか?
-- って正規表現ですか?
*はファイル名ですか?
0656名無しさん@お腹いっぱい。
2012/03/25(日) 10:59:50.45その * は glob。シェルが解釈する。
シェルが展開して rm に渡す。
-- は正規表現でも何でもなくて、ただの --。
シェルは特に何もしない。
そのまま rm に渡される。
0657名無しさん@お腹いっぱい。
2012/03/25(日) 11:00:45.800658名無しさん@お腹いっぱい。
2012/03/25(日) 11:28:39.15マッチしたときのための策だな。
0659名無しさん@お腹いっぱい。
2012/03/25(日) 11:51:49.520660名無しさん@お腹いっぱい。
2012/03/25(日) 12:14:56.30調べたらBSD系はgetopt(3)つこうてるから--大丈夫だよと書いてあったんだけど
他のUNIXはまた違うの?
0661名無しさん@お腹いっぱい。
2012/03/25(日) 12:24:48.030662名無しさん@お腹いっぱい。
2012/03/25(日) 13:38:40.680663名無しさん@お腹いっぱい。
2012/03/25(日) 13:56:37.10将来は -- が使えなくなるので注意、と書いてあるね。
0664名無しさん@お腹いっぱい。
2012/03/25(日) 18:55:04.62rm ./*
みたいにすれば引数の先頭が - になることはない。
--が使える保証がなければこっちで。
0665名無しさん@お腹いっぱい。
2012/03/25(日) 20:43:30.34で、-- が使えない ln で、-sという名前を指しているsymlinkを作ろうと、
ln -s ./-s hoge ってやると、hoge -> ./-s というsymlinkができて、
symlink自体に ./ が含まれてしまって美しくない。
-- が使えない条件で 、hoge -> -s は作れないものか。
0666名無しさん@お腹いっぱい。
2012/03/25(日) 22:24:57.35じゃあ全部スクリプトでやれよみたいな話になりかねんからダメかね
0667名無しさん@お腹いっぱい。
2012/03/25(日) 23:41:27.50昔からUnixやってる奴は普通そのやり方だよな
--指定してても、引数に./*でなく*を指定するのは違和感を感じる
0668名無しさん@お腹いっぱい。
2012/03/26(月) 06:20:04.43「◯◯◯.png」というファイルを見つけました
と表示させたいのですが、なるべくコンパクトにするにはどのようにすればいいでしょうか?
自分のやり方だとbasenameで失敗してうまく表示されません
あと、そのフォルダにpngファイルがなかった場合は何も表示しないようにしたいです。
find ${DIR}/ -name "*.png" -exec echo "「`basename {}`」というファイルを見つけました" \;
0669名無しさん@お腹いっぱい。
2012/03/26(月) 06:44:41.40find "$DIR" -name '*.png' -printf '「%f」というファイルを見つけました¥n'
0670名無しさん@お腹いっぱい。
2012/03/26(月) 07:06:33.81`basename {}`はfind実行より先に展開されてしまうので、
{}という文字列のbasenameが実行されて無意味になっているのが原因。
GNU findなら >>669 でよし。
0671名無しさん@お腹いっぱい。
2012/03/26(月) 07:46:41.98find "$DIR" -name '*.png' -exec sh -c 'for f;do echo 「`basename "$f"`」というファイルを見つけました;done' - {} +
0672名無しさん@お腹いっぱい。
2012/03/26(月) 14:20:40.93おかげさまでできました!
ありがとうございました!!
0673名無しさん@お腹いっぱい。
2012/03/27(火) 22:52:24.44開始日付と終了日付を指定したら、その間の日付を出力するスクリプトを作りたいです。
イメージとしては以下のような感じです。
$ ./hoge.sh -s 20120301 -e 20120303
20120301
20120302
20120303
以下の処理までは作ったのですが、ここから先が思いつきません。
---------------------------------------------
$ cat hoge.sh
#!/bin/sh
while getopts s:e: option
do
case $option in
"s")
start_date=$OPTARG
;;
"e")
end_date=$OPTARG
;;
*)
;;
esac
done
---------------------------------------------
よろしくご教示お願いします。
0674名無しさん@お腹いっぱい。
2012/03/27(火) 23:18:59.00perlとか使っちゃう。
0675名無しさん@お腹いっぱい。
2012/03/27(火) 23:30:44.14dateをつかってこんな感じかな
ステップ1. 今日の日付を20120301のように出力しよう
ステップ2. 20120301のような日付を、UTCエポックからの秒数に換算しよう
ステップ3. UTCエポックからの秒数を、20120301のような日付に変換しよう
ステップ4. 20120301のような日付を受け取り、その翌日の日付を出力しよう
ステップ5. ループを回して指定範囲の日付を全て出力しよう
※UTCエポックは 1970-1-1 00:00:00 UTC
0676名無しさん@お腹いっぱい。
2012/03/27(火) 23:42:28.28bash、GNU date依存
#!/bin/bash
while getopts s:e: opt; do
case "${opt}" in
's') stime=$(date -d "${OPTARG}" '+%s') ;;
'e') etime=$(date -d "${OPTARG}" '+%s') ;;
esac
done
t=${stime}
while (( ${etime} >= ${t} )); do
date -d @${t} '+%Y%m%d'
t=$(( ${t} + (60 * 60 * 24) ))
done
0677名無しさん@お腹いっぱい。
2012/03/28(水) 00:15:25.48bashならこっち
for (( t = stime; etime >= t; t += 60 * 60 * 24 )); do
date -d @${t} '+%Y%m%d'
done
0678676
2012/03/28(水) 00:53:33.920679名無しさん@お腹いっぱい。
2012/03/28(水) 06:47:17.46そんな中、何の解答にもヒントにもなってない >>675 って・・・
0680名無しさん@お腹いっぱい。
2012/03/28(水) 06:56:04.85わざわざepochからの秒数に直さなくても、日付だけで行けるよ。
#!/bin/bash
while getopts s:e: opt; do
case $opt in
s) stime=$OPTARG;;
e) etime=$OPTARG;;
esac
done
for ((t = stime; etime >= t; t=$(date -d "$t 1 day" '+%Y%m%d'))) {
date -d "$t" '+%Y%m%d'
}
0681名無しさん@お腹いっぱい。
2012/03/28(水) 07:04:16.81forループ中、dateが1回無駄になってる。せっかく$tに入ってるのでそれ使え。
for ((t = stime; t <= etime; t=$(date -d "$t 1 day" '+%Y%m%d'))) {
echo "$t"
}
0682680
2012/03/28(水) 07:10:47.880683名無しさん@お腹いっぱい。
2012/03/30(金) 14:34:15.36かっこいい方法ありますか?
0684名無しさん@お腹いっぱい。
2012/03/30(金) 14:51:40.880685名無しさん@お腹いっぱい。
2012/03/30(金) 14:55:44.65#!/bin/sh
while IFS= read -r line; do
echo "`date`: $line"
done
0686名無しさん@お腹いっぱい。
2012/03/30(金) 15:42:18.34ありがとうございます
0687名無しさん@お腹いっぱい。
2012/03/30(金) 15:44:45.94ヽ(・ω・)/ ズコー
\(.\ ノ
、ハ,,、  ̄
 ̄
0688名無しさん@お腹いっぱい。
2012/03/30(金) 16:39:16.720689名無しさん@お腹いっぱい。
2012/03/30(金) 16:44:05.09誰かパッチを送れよ。
0690名無しさん@お腹いっぱい。
2012/03/30(金) 17:35:02.11command > $(date +%Y%m%d)
って意味かとオモタ
0691名無しさん@お腹いっぱい。
2012/03/30(金) 18:02:04.91目的はロギングだな、と想像できれば、毎行にタイムスタンプつけたいんだな
ということもわかる
0692名無しさん@お腹いっぱい。
2012/03/30(金) 18:09:51.47目的はロギングだな、と想像できれば、ログファイル名毎に日付を入れたいんだな
と考えるのが自然
0693名無しさん@お腹いっぱい。
2012/03/30(金) 18:15:31.040694名無しさん@お腹いっぱい。
2012/03/30(金) 18:27:01.79touch -t YYMMDDhhmm /dev/stdout
って意味かとオモタ
0695名無しさん@お腹いっぱい。
2012/03/31(土) 01:35:03.440696名無しさん@お腹いっぱい。
2012/03/31(土) 04:56:25.930697名無しさん@お腹いっぱい。
2012/03/31(土) 17:23:45.97それだと、1900年とかそれ以前とか、32bit版の場合の2038年以降が動かないし、
GNU dateに依存するのもいやだし、
dateコマンドに頼らずに20120301等のの数字を直接操作して計算する方法ないですか?
0698名無しさん@お腹いっぱい。
2012/03/31(土) 17:42:14.02月の長さや閏年とか考えると、GNU dateかLL使うのがいいですよ。
コマンド組み合わせるのがシェルの得意なところだし。
0699名無しさん@お腹いっぱい。
2012/03/31(土) 17:47:43.69GNU date使っても epoch以前の日付には対応できないだろ、って言ってるのでは?
0700名無しさん@お腹いっぱい。
2012/03/31(土) 18:35:51.33calコマンドが表示してるのはイギリスの暦らしいね
$ cal 9 1752
September 1752
Su Mo Tu We Th Fr Sa
1 2 14 15 16
17 18 19 20 21 22 23
24 25 26 27 28 29 30
0701名無しさん@お腹いっぱい。
2012/03/31(土) 18:48:26.54江戸時代でもそれ以前でも年表は西暦で記述されてるから無問題。
1752年の件も含め、紀元1年以降対応できれば幸いです。
紀元前までは要りません。
0702名無しさん@お腹いっぱい。
2012/03/31(土) 19:29:56.450703名無しさん@お腹いっぱい。
2012/03/31(土) 20:51:17.87↓こういうデータがあって、Common Lisp, Mathematica, Javaのライブラリがある。
Calendrical Tabulations, 1900?2200
Edward M. Reingold, Nachum Dershowitz
http://www.cambridgejapan.org/academicproduct.html?isbn=9780521782531
けどまあ元のお題についてはGNU dateかLLでやるのがいいかと。
0704名無しさん@お腹いっぱい。
2012/04/01(日) 12:21:49.60曜日はどうなのよ・・・
0705名無しさん@お腹いっぱい。
2012/04/01(日) 12:24:17.1220120301 等の数字だけ処理できればいいので、曜日は無視でいいです。
0706名無しさん@お腹いっぱい。
2012/04/01(日) 13:39:02.95曜日以外はできたのか。じゃあ早く回答してやれよ。
0707名無しさん@お腹いっぱい。
2012/04/01(日) 14:01:13.24曜日以外は標準で対応し照るじゃん
0708名無しさん@お腹いっぱい。
2012/04/01(日) 14:02:48.07epoch以前でも?
0709名無しさん@お腹いっぱい。
2012/04/01(日) 14:05:53.142011年 4月 1日 金曜日 14:04:49 JST
$ date -d "100 years ago"
1912年 4月 1日 月曜日 14:04:55 CJT
$ date -d "200 years ago"
1812年 4月 1日 月曜日 14:05:08 CJT
できてんじゃん
0710名無しさん@お腹いっぱい。
2012/04/01(日) 14:09:36.73環境依存なので却下
0711名無しさん@お腹いっぱい。
2012/04/01(日) 14:16:22.13「すべての環境」に対応するのは骨が折れると思うが
期待しているぞ
0712名無しさん@お腹いっぱい。
2012/04/01(日) 14:21:28.89>>697 の言うように、20120301等の数字を直接演算すればそもそも環境依存しない。
その演算が場合分けとかでややこしいだけで。
0713名無しさん@お腹いっぱい。
2012/04/01(日) 14:36:54.89GNU date epoch以前の閏年処理おかしいよ。
$ date -d 16000229 ← 1600年は閏年
Tue Feb 29 00:00:00 LMT 1600
$ date -d 17000229 ← 1700年は閏年
date: invalid date `17000229' ← 駄目じゃん
$ cal 2 1700
February 1700
Su Mo Tu We Th Fr Sa
1 2 3
4 5 6 7 8 9 10
11 12 13 14 15 16 17
18 19 20 21 22 23 24
25 26 27 28 29 ← 閏年だよ
0714名無しさん@お腹いっぱい。
2012/04/01(日) 16:31:55.42グレゴリオ暦で1700年は平年です
$ man cal
> グレゴリオ暦への切り替えは 1752 年の 9 月 3 日に行われたと仮定している。
$ cal 9 1752
9月 1752
日 月 火 水 木 金 土
1 2 14 15 16
17 18 19 20 21 22 23
24 25 26 27 28 29 30
0715名無しさん@お腹いっぱい。
2012/04/01(日) 16:37:34.38gcal (GNU cal) 3.6
$ gcal 2 1600; gcal 2 1700 ; gcal 9 1752
February 1600
Su Mo Tu We Th Fr Sa
1 2 3 4 5
6 7 8 9 10 11 12
13 14 15 16 17 18 19
20 21 22 23 24 25 26
27 28 29
February 1700
Su Mo Tu We Th Fr Sa
1 2 3 4 5 6
7 8 9 10 11 12 13
14 15 16 17 18 19 20
21 22 23 24 25 26 27
28
September 1752
Su Mo Tu We Th Fr Sa
1 2
3 4 5 6 7 8 9
10 11 12 13 14 15 16
17 18 19 20 21 22 23
24 25 26 27 28 29 30
0716名無しさん@お腹いっぱい。
2012/04/01(日) 16:47:02.181700年はまだグレゴリオ暦に切り替わってないのだから、
1700年は閏歳が正しい。
別の例として、
1500年は閏歳なのだが、
calでは閏歳(正しい)
GNU dateでは平年(間違い)
いずれにしても GNU dateでこの年代の日付処理はできない。
0717名無しさん@お腹いっぱい。
2012/04/01(日) 17:25:06.621700年以前なら単純にすべて閏年にする。
1800年以降の場合は、400の倍数でない年のみ平年に戻す。
で桶?
case文バリバリでだれかシェルで組んでくれ。
0718名無しさん@お腹いっぱい。
2012/04/01(日) 17:54:49.44あまりしゃかりきにならんでええ。
日本は1872年かな。
0719名無しさん@お腹いっぱい。
2012/04/01(日) 18:04:26.310720名無しさん@お腹いっぱい。
2012/04/01(日) 18:12:40.42できた気がする。思ったより場合分け簡単だった。
1752年9月も対応
s_date=20120227 # 仮
e_date=20120305 # 仮
t=$s_date
while [ "$t" -le "$e_date" ]; do
echo "$t"
t=`expr "$t" + 1`
case $t in
????0[13578]32|????1032) t=`expr "$t" + 69`;;
????1232) t=`expr "$t" + 8869`;;
????0[469]31|????1131) t=`expr "$t" + 70`;;
????0230) t=`expr "$t" + 71`;;
17520903) t=17520914
esac
case $t in
????0229)
y=`expr "$t" / 10000`
case `expr "$y" % 4` in
0)
[ "$y" -le 1700 ] && continue
[ `expr "$y" % 400` = 0 ] && continue
[ `expr "$y" % 100` != 0 ] && continue
;;
esac
t=`expr "$t" + 72`;;
esac
done
0721名無しさん@お腹いっぱい。
2012/04/01(日) 19:24:47.68西暦999年以前がダメ。頭に0付け加える処理が必要。
0722名無しさん@お腹いっぱい。
2012/04/03(火) 07:17:19.37ありがとうございました
0723名無しさん@お腹いっぱい。
2012/04/03(火) 20:05:15.60[ -f /tmp/unko ]
よりかっこいい方法募集
0724名無しさん@お腹いっぱい。
2012/04/03(火) 20:06:28.950725名無しさん@お腹いっぱい。
2012/04/03(火) 20:20:14.400726名無しさん@お腹いっぱい。
2012/04/03(火) 20:44:46.90まず入ってるであろうコマンドのことだよ
coreutils と読み替えてもらっても結構
0727名無しさん@お腹いっぱい。
2012/04/03(火) 20:46:12.150728名無しさん@お腹いっぱい。
2012/04/03(火) 20:52:43.650729名無しさん@お腹いっぱい。
2012/04/03(火) 21:13:29.780730名無しさん@お腹いっぱい。
2012/04/03(火) 21:18:54.77< /tmp/unko 2> /dev/null && echo ある
0731名無しさん@お腹いっぱい。
2012/04/03(火) 22:18:53.410732名無しさん@お腹いっぱい。
2012/04/04(水) 10:03:23.150733名無しさん@お腹いっぱい。
2012/04/04(水) 10:04:50.22] はコマンドじゃないから
0734名無しさん@お腹いっぱい。
2012/04/04(水) 10:07:01.640735名無しさん@お腹いっぱい。
2012/04/04(水) 10:52:50.01$ test -f /tmp/unko ]
-bash: test: /tmp/unko: binary operator expected
なんかエラーでた
0736名無しさん@お腹いっぱい。
2012/04/04(水) 10:54:52.950737名無しさん@お腹いっぱい。
2012/04/04(水) 10:59:35.40なんで [ じゃだめなの?
0738名無しさん@お腹いっぱい。
2012/04/04(水) 12:05:16.44うぉぉ マジだ。 test.c ってこんなことやってんのか。
はぁー 無駄にすげぇーなー
0739名無しさん@お腹いっぱい。
2012/04/04(水) 12:15:50.420740名無しさん@お腹いっぱい。
2012/04/04(水) 12:21:38.54■ このスレッドは過去ログ倉庫に格納されています