【Shell】Shell变量
在 Shell 编程中,变量是用于存储数据值的名称。
定义变量时,变量名不加美元符号($,PHP语言中变量需要),如:
your_name="runoob"
变量的命名规则:
只包含字母、数字和下划线: 变量名可以包含字母(大小写敏感)、数字和下划线 _,不能包含其他特殊字符。
不能以数字开头: 变量名不能以数字开头,但可以包含数字。
避免使用 Shell 关键字: 不要使用Shell的关键字(例如 if、then、else、fi、for、while 等)作为变量名,以免引起混淆。
使用大写字母表示常量: 习惯上,常量的变量名通常使用大写字母,例如 PI=3.14。
避免使用特殊符号: 尽量避免在变量名中使用特殊符号,因为它们可能与 Shell 的语法产生冲突。
避免使用空格: 变量名中不应该包含空格,因为空格通常用于分隔命令和参数。
注意,变量名和等号之间不能有空格
等号两侧避免使用空格:
# 正确的赋值
variable_name=value# 有可能会导致错误
variable_name = value
除了显式地直接赋值,还可以用语句给变量赋值,如:
for file in `ls /etc`
或
for file in $(ls /etc)

使用变量
使用一个定义过的变量,只要在变量名前面加美元符号即可,如
your_name="qinjx"
echo $your_name
echo ${your_name}
变量名外面的花括号是可选的,加不加都行,加花括号是为了帮助解释器识别变量的边界,比如下面这种情况:
for skill in Ada Coffe Action Java; doecho "I am good at ${skill}Script"
done
如果不给skill变量加花括号,写成echo "I am good at skillScript",解释器就会把skillScript",解释器就会把skillScript",解释器就会把skillScript当成一个变量(其值为空),代码执行结果就不是我们期望的样子了。
推荐给所有变量加上花括号,这是个好的编程习惯。
已定义的变量,可以被重新定义,如:
your_name="tom"
echo $your_name
your_name="alibaba"
echo $your_name
只读变量
使用 readonly 命令可以将变量定义为只读变量,只读变量的值不能被改变。
下面的例子尝试更改只读变量,结果报错:
#!/bin/bashmyUrl="https://www.google.com"
readonly myUrl
myUrl="https://www.runoob.com"
删除变量
使用 unset 命令可以删除变量。语法:
unset variable_name
变量被删除后不能再次使用。unset 命令不能删除只读变量。
#!/bin/shmyUrl="https://www.runoob.com"
unset myUrl
echo $myUrl
以上实例执行将没有任何输出。
特殊变量
有一些特殊变量在 Shell 中具有特殊含义,例如 $0表示当前脚本(或命令)的名称 / 路径,$1, $2, 等表示脚本的参数。
$#表示传递给脚本的参数数量,$? 表示上一个命令的退出状态等
Shell字符串
字符串是shell编程中最常用最有用的数据类型(除了数字和字符串,也没啥其它类型好用了),字符串可以用单引号,也可以用双引号,也可以不用引号。
str='this is a string'
单引号字符串的限制:
单引号里的任何字符都会原样输出,单引号字符串中的变量是无效的;
单引号字符串中不能出现单独一个的单引号(对单引号使用转义符后也不行),但可成对出现,作为字符串拼接使用。
双引号
your_name="runoob"
str="Hello, I know you are \"$your_name\"! \n"
echo -e $str

echo 命令使用了 -e 选项,该选项会开启转义字符的解析功能,因此 \n 被识别为换行符,最终输出时会在句尾换行
没有 -e 时,\n 会被当作普通字符输出(显示为 \n)。
内部的\"是 Shell 语法中的转义(属于变量定义阶段的解析),作用是告诉 Shell:这里的双引号是普通字符,不要当作字符串的结束符。
这个转义过程发生在 变量赋值阶段(脚本执行到 str=… 这一行时),与后续的 echo 命令无关。无论 echo 是否加 -e," 都会被解析为普通双引号,保存在变量 str 中。
#!/bin/bashyour_name="runoob"
str="Hello, I know you are '$your_name'! \n"
echo -e $str

单引号的 “屏蔽变量解析” 功能,仅针对被单引号直接包裹的内容
在该脚本中,单引号是双引号字符串里的普通字符,变量实际在双引号范围内,因此能正常解析。
这是 Shell 字符串中 “引号嵌套” 的典型场景,外层引号决定了整体解析规则,内层引号仅作为普通字符存在
获取字符串长度
string="abcd"
# 这里要用花括号
echo ${#string}
变量为字符串时,${#string} 等价于 ${#string[0]}
Shell注释
单行注释
以 # 开头的行就是注释,会被解释器忽略。
通过每一行加一个 # 号设置多行注释,像这样:
#--------------------------------------------
# 这是一个注释
# author:菜鸟教程
# site:www.runoob.com
# slogan:学的不仅是技术,更是梦想!
#--------------------------------------------
##### 用户配置区 开始 #####
#
#
# 这里可以添加脚本描述信息
#
#
##### 用户配置区 结束 #####
如果在开发过程中,遇到大段的代码需要临时注释起来,过一会儿又取消注释,怎么办呢?
每一行加个#符号太费力了,可以把这一段要注释的代码用一对花括号括起来,定义成一个函数,没有地方调用这个函数,这块代码就不会执行,达到了和注释一样的效果。
多行注释
使用Here文档
:<<EOF
注释内容...
注释内容...
注释内容...
EOF
以上例子中,: 是一个空命令,用于执行后面的 Here 文档,<<‘EOF’ 表示开启 Here 文档,COMMENT 是 Here 文档的标识符,在这两个标识符之间的内容都会被视为注释,不会被执行。
EOF 也可以使用其他符号:


或者


或者


不建议使用'


' 是 Shell 中的字符串分隔符,当它作为 here-document 的标记时,Shell 会认为需要一个非单引号的标记(或对'进行转义)。简单说,用'作为起始标记时,Shell 很难正确识别对应的结束标记,容易触发语法歧义。
也可以直接使用 : 号
我们也可以使用了冒号 : 命令,并用单引号 ' 将多行内容括起来。由于冒号是一个空命令,这些内容不会被执行。
格式为:: + 空格 + 单引号。


