Bash Shell脚本学习小结
今天需要写一个Shell脚本。很简单,判断一个日志文件是否大大于2G,如果大于2G则删除。久了没有写bash,竟然一点都想不起来写了。于是复习了一下,下面对今天的学习做个小结:
1.字符截断:
如果是一般路径的字符截断可以用 basename和 dirname这两个工具:
basename可以从一个文件路径中截一个文件名
例如:
dirname可以从一个文件路径中截到一个目录路径
例如:
不使用外部工具进行字符截断
bash有自带的功能来对变量进行字符截断,一般使用"##", "#", "%%", "%", "*" 组合来实现。例如:
" #"表示从字符开始部分除去,一旦匹配则立即除去
" ##"表示从字符开始部分除去,会搜整个字符串最长的和的匹配来除去
" %"表示从字符结束的部分除去,一旦匹配成公则立即除去
" %%"表示从字符结束的部分开始除去,会搜寻整个字符穿中最长的匹配来除去
" *"统配符,一般与“##”或"#"联用时放在搜索字符串的左边,例如:${String#*sh}(在sh的左边),与"%%"或"%"联用时会放在匹配字符串的右边,例如:${String%%sh*}
常用技巧:
接收来自命令行传入的参数,第一个参数用$1表示,第二个参数$2表示,。。。以此类推。注意:$0表示脚本文件名。另外一个在shell编程中经常用到的是“$@”这个代表所有的参数,。你可以用一个循环来遍历这个参数。如果用java来类比的话,可以把$@看作是man函数中定义的那个数组
3.if语句:
格式:
conditon测试类型对照表
感觉bash 中的if相比其他的一些语言智能多了,在bash中,测试一个文件的存在跟比较两个数字的大小没有什么两样 ;)
4.for语句
bash里的语句总是那么的人性化,十分的接近自然语言,在for语句中几乎可以迭代任何类似与集合的数据类型(或许这样个说法不对,但我确实想不到更好的词来代替)。
看一个例子:
for更经常用到的是遍历目录,下面的例子用于列出当前目录下的所有文件和文件夹的名称
对于文件遍历,更有趣的是,你可以在 “in” 后面接上多个表达式。也就是说,你可以一次在遍历多个目录。
下面这段代码能把当前目录下go文件夹和do文件夹里的文件复制到fo文件夹下
1.字符截断:
如果是一般路径的字符截断可以用 basename和 dirname这两个工具:
basename可以从一个文件路径中截一个文件名
例如:
$ basename
/
home
/
file
.
tar
file . tar
file . tar
dirname可以从一个文件路径中截到一个目录路径
例如:
$ dirname
/
home
/
file
.
tar
/ home
/ home
不使用外部工具进行字符截断
bash有自带的功能来对变量进行字符截断,一般使用"##", "#", "%%", "%", "*" 组合来实现。例如:
$ string
=
hellowbashshell
$ echo ${string # #*sh}
ell
$ echo ${string # *sh}
shell
$ echo ${string % %sh * }
hellowba
$ echo ${string %sh * }
hellowbash
$ echo ${string # #*sh}
ell
$ echo ${string # *sh}
shell
$ echo ${string % %sh * }
hellowba
$ echo ${string %sh * }
hellowbash
" #"表示从字符开始部分除去,一旦匹配则立即除去
" ##"表示从字符开始部分除去,会搜整个字符串最长的和的匹配来除去
" %"表示从字符结束的部分除去,一旦匹配成公则立即除去
" %%"表示从字符结束的部分开始除去,会搜寻整个字符穿中最长的匹配来除去
" *"统配符,一般与“##”或"#"联用时放在搜索字符串的左边,例如:${String#*sh}(在sh的左边),与"%%"或"%"联用时会放在匹配字符串的右边,例如:${String%%sh*}
常用技巧:
在路径中取文件名:${path##*/}(与basename相同功能)2.自变量的接收
在路径中取目录路径:${path%/*}(与dirname相同功能)
取文件的扩展名:${path##*.}
接收来自命令行传入的参数,第一个参数用$1表示,第二个参数$2表示,。。。以此类推。注意:$0表示脚本文件名。另外一个在shell编程中经常用到的是“$@”这个代表所有的参数,。你可以用一个循环来遍历这个参数。如果用java来类比的话,可以把$@看作是man函数中定义的那个数组
3.if语句:
格式:
if
[ condition ]
then
action
fi
then
action
fi
注意:“if”和“[”之间需要空格,如果你不空格,shell会报告语法错误的。我就被这个浪费了好一阵时间
conditon测试类型对照表
运算符 | 描述 | 示例 |
文件比较运算符 | ||
-e filename | 如果 filename存在,则为真 | [ -e /var/log/syslog ] |
-d filename | 如果 filename为目录,则为真 | [ -d /tmp/mydir ] |
-f filename | 如果 filename为常规文件,则为真 | [ -f /usr/bin/grep ] |
-L filename | 如果 filename为符号链接,则为真 | [ -L /usr/bin/grep ] |
-r filename | 如果 filename可读,则为真 | [ -r /var/log/syslog ] |
-w filename | 如果 filename可写,则为真 | [ -w /var/mytmp.txt ] |
-x filename | 如果 filename可执行,则为真 | [ -L /usr/bin/grep ] |
filename1-nt filename2 | 如果 filename1比 filename2新,则为真 | [ /tmp/install/etc/services -nt /etc/services ] |
filename1-ot filename2 | 如果 filename1比 filename2旧,则为真 | [ /boot/bzImage -ot arch/i386/boot/bzImage ] |
字符串比较运算符 (请注意引号的使用,这是防止空格扰乱代码的好方法) | ||
-z string | 如果 string长度为零,则为真 | [ -z "$myvar" ] |
-n string | 如果 string长度非零,则为真 | [ -n "$myvar" ] |
string1= string2 | 如果 string1与 string2相同,则为真 | [ "$myvar" = "one two three" ] |
string1!= string2 | 如果 string1与 string2不同,则为真 | [ "$myvar" != "one two three" ] |
算术比较运算符 | ||
num1-eq num2 | 等于 | [ 3 -eq $mynum ] |
num1-ne num2 | 不等于 | [ 3 -ne $mynum ] |
num1-lt num2 | 小于 | [ 3 -lt $mynum ] |
num1-le num2 | 小于或等于 | [ 3 -le $mynum ] |
num1-gt num2 | 大于 | [ 3 -gt $mynum ] |
num1-ge num2 | 大于或等于 | [ 3 -ge $mynum ] |
感觉bash 中的if相比其他的一些语言智能多了,在bash中,测试一个文件的存在跟比较两个数字的大小没有什么两样 ;)
4.for语句
bash里的语句总是那么的人性化,十分的接近自然语言,在for语句中几乎可以迭代任何类似与集合的数据类型(或许这样个说法不对,但我确实想不到更好的词来代替)。
看一个例子:
#
!/bin/bash
for args in $@
do
echo $args
done
把上面这段代码录入保存为showargs.sh设置为可执行(chmod +x showargs.sh)执行:
for args in $@
do
echo $args
done
$
./
showargs
.
sh arg1 arg2 arg3 arg4
arg1
arg2
arg3
arg4
这个例子中,我们用到了之“$@”,它代表了所有的命令行参数。在这里用for对其进行遍历,系统迭代地从$@中取出命令行参数把他放到args中,最后使用echo $args进行输出。
arg1
arg2
arg3
arg4
for更经常用到的是遍历目录,下面的例子用于列出当前目录下的所有文件和文件夹的名称
$
for
file
in
*
> do
> echo $file
> done
这里用*代表当前目录,列出的是所有的文件和文件夹的名称,在这里,文件夹和文件你是分不出来的,如果你需要,你应该用if [-d ${file}]来做一下判断。
> do
> echo $file
> done
对于文件遍历,更有趣的是,你可以在 “in” 后面接上多个表达式。也就是说,你可以一次在遍历多个目录。
下面这段代码能把当前目录下go文件夹和do文件夹里的文件复制到fo文件夹下
#
!/bin/bash
for args in ./ go /* ./ do /*
do
cp ${args} ./ fo
echo " copying ${args} to ./fo/${args} "
done
for args in ./ go /* ./ do /*
do
cp ${args} ./ fo
echo " copying ${args} to ./fo/${args} "
done