shell脚本案例分析

#!/bin/sh

### GLOBALS
IMG_EXT="{png,jpg,gif}"
SQL_FILE="my_images_mysql.sql"
SQL_INS="INSERT INTO images VALUES ("
SQL_IMAGEID_RANGE=0
SQL_IMAGETYPE=1
SQL_NAME=""
SQL_IMAGE=""

### ERROR
NORMAL=0
ERR_ARGS=1
ERR_NO_DIR=2
ERR_NO_FILE=3
RETVAL=$NORMAL

########################################################################
### Actual Main
########################################################################
main() {
    local dir=$1
    local num=$2
    
    # check the number of command argument
    if [ $# -lt 2 ]; then
        return $ERR_ARGS
    fi
    
    # check target dir
    [ ! -d $dir ] && return $ERR_NO_DIR

    # check target file
    check_image_file $dir || return $?

    # make sql file
    make_sql_file $num

    return $RETVAL
}

########################################################################
### Check image files existence
########################################################################
check_image_file() {

    local dir=$1
    local file_num=0

    file_num=$(eval ls $dir/*.$IMG_EXT 2>/dev/null | wc -l)
    [ $file_num -eq 0 ] && return $ERR_NO_FILE

    return $RETVAL
}

########################################################################
### Make SQL file to insert image files
########################################################################
make_sql_file() {
    
    local f=

    [ -f $SQL_FILE ] && rm -f $SQL_FILE
    
    SQL_IMAGEID=$1
    for f in $(eval ls $dir/*.$IMG_EXT 2>/dev/null)
    do
        SQL_NAME=$(basename $f | cut -d. -f1)
        SQL_IMAGE="0x$(od -tx1 $f | awk '{for(i=2; i<=NF; i++) printf("%s", toupper($i))}')"
        echo "$SQL_INS $SQL_IMAGEID, $SQL_IMAGETYPE, '$SQL_NAME', $SQL_IMAGE);" >> $SQL_FILE
        SQL_IMAGEID=$(($SQL_IMAGEID + 1))
        echo -n "." # in progress 
    done

    echo -e "\ncompleted"
    return $RETVAL
}


########################################################################
### Check error and display error message
########################################################################
check_error() {

    local result=$1

    case $result in
        $ERR_ARGS)
            usage
            ;;
        $ERR_NO_DIR)
            echo "cannot find target dir"
            ;;
        $ERR_NO_FILE)
            echo "cannot find \"*.$IMG_EXT\" files"
            ;;
        *)
            echo "unknown error"
            ;;
    esac

    return $result
}

########################################################################
### Usage
########################################################################
usage() {
    echo "Usage: make_img_insert_sql.sh <dir> <start_imageid>"
}

########################################################################
### Script Main
########################################################################
main "$@" || check_error $?

 

1、各种$变量:

$0 这个程式的执行名字
$n 这个程式的第n个参数值,n=1..9
$* 这个程式的所有参数,此选项参数可超过9个。
$# 这个程式的参数个数
$$ 这个程式的PID(脚本运行的当前进程ID号)
$! 执行上一个背景指令的PID(后台运行的最后一个进程的进程ID号)
$? 执行上一个指令的返回值 (显示最后命令的退出状态。0表示没有错误,其他任何值表明有错误)
$- 显示shell使用的当前选项,与set命令功能相同
$@ 跟$*类似,但是可以当作数组用

2、eval的用法:eval命令将会首先扫描命令行进行所有的置换,然后再执行该命令。该命令适用于那些一次扫描无法实现其功能的变量。该命令对变量进行两次扫描。这些需要进行两次扫描的变量有时被称为复杂变量。不过这些变量本身并不复杂。eval命令也可以用于回显简单变量,不一定是复杂变量。

1 eval命令也可以用于回显简单变量,不一定是复杂变量  
例如: 

[neau@mail ~]$ NAME=VALUEBANK  
[neau@mail ~]$ eval echo $NAME  
VALUEBANK  
[neau@mail ~]$ echo $NAME  
VALUEBANK

 
2 执行含有字符串的命令  
    首先我们首先创建一个名为test的小文件,在这个小文件中含有一些文本。接着,将cat test赋给变量myfile,现在我们e cho该变量,看看是否能够执行上述命令。  

[neau@mail ~]$ vi test  
[neau@mail ~]$ cat test  
Hello World!!!  
I am a chinese Boy!

 
将cat testf赋给变量myfile  
[neau@mail ~]$ myfile="cat test"  
如果我们echo该变量,我们将无法列出test 文件中的内容。  

[neau@mail ~]$ echo $myfile  
cat test

让我们来试一下eval命令,记住eval命令将会对该变量进行两次扫瞄。  

[neau@mail ~]$ eval $myfile  
Hello World!!!  
I am a chinese Boy!

 
从上面的结果可以看出,使用eval命令不但可以置换该变量,还能够执行相应的命令。第一次扫描进行了变量置换,  
即eval cat test,第二次扫描执行了该字符串中所包含的命令cat test。  
3命令还可以用来显示出传递给脚本的最后一个参数  

[neau@mail ~]$ cat test1  
#!/bin/bash  
echo "Total of the arguments passed $#"  
echo "The process Id is $$"  
echo "Last argument os "$(eval echo \$$#)""  
[neau@mail ~]$ ./test1 value bank test last  
Total of the arguments passed 4  
The process Id is 21545  
Last argument os last

 
在上面的脚本中, 反斜杠表示转义字符的意识,所以第一个$在第一次替换时只作为普通‘$’符,eval命令首先把$#解析为当前shell的参数个数“$(eval echo $4)”,然后在第二次扫描时eval 将执行echo命令,并且外层“”有解析功能,会把$4用值(第4个参数)代替,所以最终是执行echo last,得出最后一个参数last。

4给每个值一个变量名  
可以给一个值一个变量名。下面我对此做些解释,假定有一个名为test2的文件,  
你希望该文件中的第一列成为变量名,第二列成为该变量的值,这样就可以:  

[neau@mail ~]$ cat test2  
COMMANY TQ  
LANGUE ENGLISH  
LIKE YES  
[neau@mail ~]$ cat test3  
#!/bin/bash  
while read NAME VALUE  
do  
eval "${NAME}=${VALUE}"  
done  
echo "$COMMANY $LANGUE $LIKE"  
[neau@mail ~]$ ./test3  
TQ ENGLISH YES

补充:  
eval:的作用其实还是命令解析的作用,“${NAME}=${VALUE}”中的‘=’是一个复制过程。  
read:从保准输入读取一行参数,并将参数按分隔符分段付给参数。这里read的不是stdin二是test2文件,因为存在重定向done

你可能感兴趣的:($?,$#,eval,$1,$*)