文章目录
- 1. 脚本执行
- 2. 变量
- shell变量
- 获取当前时间并格式化
- 3. 传递参数- $
- 4. 表达式
- 5.other
shell脚本一般以 *.sh 的方式进行命名,第一行为
#!/bin/bash
。
一般情况下,人们并不区分 Bourne Shell 和 Bourne Again Shell,所以,像 #!/bin/sh
,它同样也可以改为#!/bin/bash
。
#!
告诉系统其后路径所指定的程序即是解释此脚本文件的Shell程序。
shell脚本扩展名为sh(sh代表shell),扩展名并不影响脚本执行,见名知意就好。
1. 脚本执行
- 作为可执行程序(直接使用文件执行) ,以执行 test.sh 为例
-
./test.sh
:一定要写成./test.sh,而不是test.sh,运行其它二进制的程序也一样,直接写test.sh,linux系统会去PATH里寻找有没有叫test.sh的,而只有/bin, /sbin, /usr/bin,/usr/sbin等在PATH里,你的当前目录通常不在PATH里,所以写成test.sh是找不到命令的,要用./test.sh告诉系统说,就在当前目录找。 -
sh test.sh
:如果在当前路径下执行,可以使用sh test.sh
执行 test.sh脚本
- 作为解释器参数执行
直接执行解释器,后面添加shell脚本作为解释器参数。eg:/bin/zsh 1_hello.sh
,这么指定后不需要在第一行中指定解释器信息了(#!
字段),写了也不会执行。
2. 变量
shell变量
- 区分大小写
- 给变量赋值时不能有空格,如下图中的18行的空格就不能正常识别。
- 定义变量时,变量名不加$
符号,使用的时候加上$
符号,使用的时候加不加{}
都可以,不过为了上面这种给变量赋值的情况也不用加$
。推荐给所有变量加上花括号,这是个好的编程习惯。
```shell
Name="orig max"
echo $Name
name="orig"
echo $name
echo ${name}
name="changed"
echo $name
echo ${name}
echo '$namewhere'
echo '${name}where'
echo "$namewhere" #输出空,因为没有这个变量,此时需要加{}才能正常识别
echo "${name}where"
```
- 字符串
- 字符串可以用单、双引号,也可以不用引号。单双引号的区别跟PHP类似。
- 单引号字符串的限制:
- 单引号中的任何字符都将原样输出,包括其中的变量
- 单引号中不能再出现单引号,即使使用转义字符也不行
- 双引号的优点:
- 双引号中可以有变量
- 可以出现转义字符
- 字符串拼接
greet="hello, "${name}"!"
echo $greet
- 获取字符串长度:使用
#
,eg:echo ${#greet}
- 提取子字符串:eg:
sub=${greet:5:10}
- 数组
- 在Shell中,用括号来表示数组,数组元素用"空格"符号分割开。如果使用
,
,将会赋值给第一个元素。array2=(1 2 3 4 5 6 )
才能正常输出。 - 使用**@符号**可以获取数组中的所有元素,eg:
echo ${array2[@]}
# 数组
echo "array"
array=(1,2,3,4,5)
for i in {0..5}
do
echo ${array[i]}
done
echo "array2"
array2=(1 2 3 4 5 6 )
for i in {0..5}
do
echo ${array2[i]}
done
- 获取数组的长度的方法和字符串的一样,使用
#
,只不过这里需要足以使用的是哪种。
echo ${#array[@]} #1 获得数组元素个数
echo ${#array[0]} #9 数组中第一个元素的长度,等价于echo ${#array}
echo ${#array} #9
通过上面的输出结果,可以获知,数组中使用,
,shell将会把,
当成变量的一部分,指导遇到空格或者结束符)
- Shell 注释:shell中只能通过
#
添加注释,没有多行注释,每行一个。如果要注释多行,使用编辑器或者定义成一个函数,这样就只用注释调用函数的地方。
获取当前时间并格式化
time=$(date "+%Y.%m.%d-%H%M%S")
echo ${time}
# logs文件根据时间重命名
mv logs logs_${time}
- date后面有一个空格,否则无法识别命令,shell对空格还是很严格的。
参数含义:
- Y显示4位年份,如:2018;
- y显示2位年份,如:18。
- m表示月份;M表示分钟;
- d表示日期(几号);而D则表示当前日期,如:1/18/18(也就是2018.1.18)
- H表示小时;而h显示月份的英文,如
Oct
表示 10月 - S显示当前秒钟,单位为秒。
- s显示当前秒钟,单位为毫秒;
refer:shell获取系统当前时间并格式化
3. 传递参数- $
在执行 Shell 脚本时,向脚本传递参数,脚本内获取参数的格式为:$n
。n 代表一个数字,1 为执行脚本的第一个参数,2 为执行脚本的第二个参数,$0
为shell脚本名称。其中不受执行方式的影响,即,/bin/sh 3_transf_param.sh
和 ./3_transf_param.sh
的效果一样,其中para 0还是文件名。
还有几个特殊字符用来处理参数:
echo '$# = ' $# #传递到脚本的参数个数,不包括文件名
echo '$* = ' $* #以一个单字符串显示所有向脚本传递的参数。
echo '$@ = ' "$@" #与$*相同,但是使用时加引号,并在引号中返回每个参数。
echo '$$ = ' "$$" #脚本运行的当前进程ID
echo '$? = ' "$?" #显示最后命令的退出状态。0表示没有错误,其他任何值表明有错误。
echo `$n` #n 代表一个数字,1 为执行脚本的第一个参数,2 为执行脚本的第二个参数,`$0`为shell脚本名称。
- $* 与 $@ 区别:
相同点:都是引用所有参数。
不同点:只有在双引号中体现出来。假设在脚本运行时写了三个参数 1、2、3,,则 " * " 等价于 “1 2 3”(传递了一个参数),而 “@” 等价于 “1” “2” “3”(传递了三个参数)。
如下代码:
for i in "$*"; do
echo '$*' $i
done
for i in "$@"; do
echo '$@' $i
done
./3_transf_param.sh 123 456 sdf
输出结果为:
4. 表达式
脚本 5_operation.sh
- 注意点
- 乘号(*)前边必须加反斜杠()才能实现乘法运算;
- 表达式和运算符之间要有空格
- if和[之间也要有空格,for循环一定有空格
- 但是变量定义的时候变量和=之前不能有空格
- 疑问
- 下面的shell中${val6}不输出任何东西?
# ${val6}在这里不输出任何东西?
val6=`expr 5 == 6`
echo 'expr 5 == 6 ?' ${val6}
- 以下语句导致溢出?
# 应该使用单反引号,以下语句导致栈溢出?
val2='expr 5 * 6'
echo 'error: expr 5 * 6 = ' ${val2}
5.other
var=0
# bash 里面可以用 (( )) 执行 C 风格的算术表达式。
# 如果你接下来还会读 if 那一段的话,你还会知道这玩意的返回和 C 的非零真假判断一致。
(( var += 1 )) # 这是一种,现在 var 是 1
(( var++ )) # 这也是一种自增,2
(( var = var * var )) # 怎么来乘法了!var 现在是 4。
let 'var = var / 3' # 还是只有 bash 才能用的拓展。除法是整数除法,向 0 舍入,1。
# 来一点不一定要 bash 的方法吧,毕竟 sh 还有很多种,总不能全报错了吧。
# $(( )) 会展开成为这个 C 表达式求值的结果。以前 bash 有个 $[ ] 一样,但是别去用。
echo $((var += 2)) # echo 出 3,var 现在是 3。
var=$((var-1)) # 展开成 var=2,var 现在是……还用说吗,2。
shell调试模式:sh -x strangescript
shell只检查语法,不执行脚本:sh -n your_script
在计算领域中,Shebang(也称为 Hashbang )是一个由井号和叹号构成的字符序列 #!
,其出现在文本文件的第一行的前两个字符。 在文件中存在 Shebang 的情况下,类 Unix 操作系统的程序加载器会分析 Shebang 后的内容,将这些内容作为解释器指令,并调用该指令,并将载有 Shebang 的文件路径作为该解释器的参数[1]。