結構簡単なbashスクリプトを作っているが、細かい点を忘れがちでここに記す
もはやbashに関係ないものも書いているが..

findで見つけたファイルをtarで固める

tar . -name \( '*.html' -o -name '*.htm' \) -print0 |  tar czf find.tar.gz --null -T -

繰り返して実行する

while true ;do echo `date '+%y%m%d %H%M%S'` $(sensors | head -n +3 | tail -n 1 ); sleep 10; done;
 
while true; do echo `date '+%y%m%d %H:%M:%S'`  $(smartctl -d scsi -a /dev/nvme0n1 -T permissive | grep "Current Drive"|awk '{print $4}'); sleep 10 ;done;

○行目以降を表示

qstatの表示で上二行を省いて表示

qstat | tail -n +3

ヒアドキュメント

cat << _EOF_  > README.1st
どうして
こうなった?
_EOF_

awk

awkをbashで使ったので
unixユーザで、uidが1000以上のユーザリストを取得する

[foo@c ~]$ getent passwd |awk -F: '$3>1000{print $1}'

「-F」はデミリタ、「$3」でUIDの部分を示してそれが1000より大きいのを持って来て、「{print $1}」でアカウント名を表示

標準出力

標準出力、標準エラー共に一つのファイルに入れる. 「2>1&」を使う

[foo@c ~]$ ./bash.cmd  > log  2>&1

他にも「./bash.cmd >& log」、「./bash.cmd &> log」

共に出力しないなら「2>1&」と「/dev/null」を使って

[foo@c ~]$ ./bash.cmd  > /dev/null 2>&1

どちらとも別々のファイルに入れたい. log1(標準出力)、log2(標準エラー)

[foo@c ~]$ ./bash.cmd  1> log1  2> log2

標準エラーは画面に出しつつファイルにも書いて欲しい。標準出力はいらね.
teeコマンドは標準出力をファイルに落としてくれるが、標準エラーは見ない。
なので一旦エラーの識別子(2)を(1)に変更して、パイプ(|)で渡す

[foo@c ~]$ { ./bash.cmd   3>&2  2>&1  1>&3 | tee log1; }  2> /dev/null

留意 先頭の「{」の後は空白が必要
「3>&2 2>&1 1>&3」の考え方だけど、1と2を入れ替えただけ。イメージは下記のような感じ
2017y08m23d_190145715.png 最終的に1と2が入れ替わった

for文

よく使います. フォルダ内のファイルを処理したい場合

#!/bin/bash
IFS=$'\n'               # IFS="$'\n'$'\t' " デフォルト値だと改行、タブ、スペースが区切りと見做される
for f in `ls /tmp`; do
  echo $f
done

そのフォルダが複数ある場合

#!/bin/bash
etc=`ls -d /etc/*`       #/etc直下のファイルをフルパスで表記
tmp=`ls -d /tmp/*`       #同/tmp直下も
for f in $etc $tmp; do   # 2つの変数を同時に扱う
  echo $f
done

特定のファイルで

#!/bin/bash
mrcs=$(find /data4 -type f -name "*.mrcs" | grep -e "[0-9]\{4\}_[0-9]\{2\}") # "2012_06"とかを含むmrcsファイル
for f in $mrcs; do
  echo $f
done

連番のファイルにしたい

#!/bin/bash
mrcs=$(find /data4 -type f -name "*.mrcs")
i=1
for f in $mrcs; do
  cmd="ln -s $f $(printf "%04d" $i).mrcs"   #一旦コマンドラインに整形して、次の行で実行
  $cmd
  i=$(expr $i + 1)
done

既に連番を持ったファイルで偶数番号のファイルだけを拾いたい

#!/bin/bash
tif=$(find . -name "*.tif")
i=1
for f in $tif; do  #aaaa
   num=$(echo $f | cut -c6-9|bc)   # ファイル名の連番部分を切り出して(cut)、数字にします(bc)
   if [ $(($num%2)) == 0 ]; then   # 2で割った余りで偶数/奇数を見る
     echo $f
   fi
done

超簡易並列

多数のディレクトリに対して処理を施したいが、どうせなら複数のジョブを流して対処したい
っで作ってみた

#!/bin/bash

cat <<'_EOF_'>run.$$.sh            # 個々のディレクトリで動くスクリプト
#!/bin/bash
echo start:`date '+%y%m%d %H:%M:%S'`
expr `cat max |bc` - 1 > max

<目的の処理>

expr `cat max |bc` + 1 > max
echo end:`date '+%y%m%d %H:%M:%S'`
_EOF_
chmod +x run.$$.sh

echo 8 > max.                  # 同時実行数

for d in `find . -maxdepth 2  -mindepth 2 -type d`; do   # ターゲットのディレクトリを取得
  ./run.$$.sh $d &             # 「&」でバックグラウンド処理へ回す
  sync                         # 「max」ファイルの同期に必須
  if [ $(expr `cat max`) -eq 0 ]; then
     while :
     do
       sleep 10
       echo "break:" `cat max`
       if [ $(expr `cat max`) -gt 0 ]; then
         break
       fi
     done
  fi
done
rm -f run.$$.sh max

トップ   編集 添付 複製 名前変更     ヘルプ   最終更新のRSS
Last-modified: 2017-12-14 (木) 10:54:31 (3d)