shell变量、参数传递($)、脚本执行等常见操作总结

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]。

  • 5
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值