Shell基本介绍
**Shell **
是一个用 C 语言编写的程序,它是用户使用** Linux **
的桥梁。**Shell **
既是一种命令语言,又是一种程序设计语言。**Shell **
是指一种应用程序,这个应用程序提供了一个界面,用户通过这个界面访问操作系统内核的服务。**Ken Thompson **
的** sh **
是第一种** Unix Shell**
,**Windows Explorer **
是一个典型的图形界面**Shell**
。
Shell脚本说明
Shell 脚本(shell script),是一种为 shell 编写的脚本程序。
注意:业界所说的 shell 通常都是指 shell 脚本,但读者朋友要知道,shell 和 shell script 是两个不同的概念。
Shell运行环境
Shell 编程跟 JavaScript、php 编程一样,只要有一个能编写代码的文本编辑器和一个能解释执行的脚本解释器就可以了。
Linux 的 Shell 种类众多,常见的有:
Bourne Shell(/usr/bin/sh或/bin/sh)
Bourne Again Shell(/bin/bash)
C Shell(/usr/bin/csh)
K Shell(/usr/bin/ksh)
Shell for Root(/sbin/sh)
……
本教程关注的是 Bash,也就是** Bourne Again Shell**
,由于易用和免费,Bash 在日常工作中被广泛使用。同时,Bash 也是大多数Linux 系统默认的 Shell。
在一般情况下,人们并不区分** Bourne Shell **
和 Bourne Again Shell,所以,像 **#!/bin/sh**
,它同样也可以改为 **#!/bin/bash**
。
Shell脚本示例:
#!/bin/bash
user_name=zhangsan
user_pwd='112233'
echo -e "Hello World ! userName=${user_name},userPwd=${user_pwd}!"
echo -e "lao zhi di yi chi qiao shell jiaoben geidian mianzhi"
echo "hello,${user_name},${user_pwd}" > userinfo.txt
**#!**
告诉系统其后路径所指定的程序即是解释此脚本文件的 Shell 程序。
Shell执行方式
运行 Shell 脚本有两种方法:
1、作为可执行程序
将上面的代码保存为 test.sh,并 cd 到相应目录:
chmod +x ./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**
告诉系统说,就在当前目录找。
2、作为解释器参数
这种运行方式是,直接运行解释器,其参数就是 shell 脚本的文件名,如:
/bin/sh test.sh
/bin/php test.php
这种方式运行的脚本,不需要在第一行指定解释器信息(就是**#!/bin/bash **
),写了也没用。
Shell变量
定义变量
定义变量时,变量名不加美元符号($,PHP语言中变量需要),如:
user_name="zhangsan"
user_name="112233445566"
注意,变量名和等号之间不能有空格,这可能和你熟悉的所有编程语言都不一样。同时,变量名的命名须遵循如下规则:
命名只能使用英文字母,数字和下划线,首个字符不能以数字开头。
中间不能有空格,可以使用下划线(_)。
不能使用标点符号。
不能使用bash里的关键字(可用help命令查看保留关键字)。
有效的 Shell 变量名示例如下:
RUNOOB="778899"
LD_LIBRARY_PATH="456455"
_var="556644"
var2="112233"
无效的变量命名:
# 你就按照JAVA的编程习惯来
?var=123
user*name=runoob
除了显式地直接赋值,还可以用语句给变量赋值,如:
# 别看了,你现在看不懂的
for file in `ls /etc`
或
for file in $(ls /etc)
以上语句将 /etc 下目录的文件名循环出来。
使用变量
使用一个定义过的变量,只要在变量名前面加美元符号即可,如:
your_name="zhangsan"
# echo是打印的意思,增强版是printf
echo $your_name
echo ${your_name}
注意:变量名外面的花括号是可选的,加不加都行,加花括号是为了帮助解释器识别变量的边界,处于严谨的话最好变量都加一个,便于区分
已定义的变量,可以被重新定义,如:
your_name="tom"
echo $your_name
your_name="alibaba"
echo $your_name
这样写是合法的,但注意,第二次赋值的时候不能写**$your_name="alibaba"**
,使用变量的时候才加美元符($)。
设置为只读(类似Java的final)
使用 readonly 命令可以将变量定义为只读变量,只读变量的值不能被改变。
下面的例子尝试更改只读变量,结果报错:
#!/bin/bash
myUrl="https://www.google.com"
readonly myUrl
myUrl="https://www.runoob.com"
运行脚本,结果如下:
/bin/sh: NAME: This variable is read only.
删除变量(你写东西应该不会考虑回收的)
使用 unset 命令可以删除变量。语法:
unset variable_name
变量被删除后不能再次使用。unset 命令不能删除只读变量。
实例
#!/bin/sh
myUrl="https://www.runoob.com"
unset myUrl
echo $myUrl
以上实例执行将没有任何输出。
变量类型
运行shell时,会同时存在三种变量:
1) 局部变量 局部变量在脚本或命令中定义,仅在当前shell实例中有效,其他shell启动的程序不能访问局部变量。
2) 环境变量 所有的程序,包括shell启动的程序,都能访问环境变量,有些程序需要环境变量来保证其正常运行。必要的时候shell脚本也可以定义环境变量。
3) shell变量 shell变量是由shell程序设置的特殊变量。shell变量中有一部分是环境变量,有一部分是局部变量,这些变量保证了shell的正常运行
特殊变量
前面已经讲到,变量名只能包含数字、字母和下划线,因为某些包含其他字符的变量有特殊含义,这样的变量被称为特殊变量。
变量 | 含义 |
---|---|
$0 | 当前脚本的文件名 |
$n | 传递给脚本或函数的参数。n 是一个数字,表示第几个参数。例如,第一个参数是$1,第二个参数是$2。 |
$# | 传递给脚本或函数的参数个数。 |
$* | 传递给脚本或函数的所有参数。 |
$@ | 传递给脚本或函数的所有参数。被双引号(“ “)包含时,与$*稍有不同 |
$? | 上个命令的退出状态,或函数的返回值。 |
$$ | 当前Shell进程ID。对于 Shell 脚本,就是这些脚本所在的进程ID。 |
示例:
test.sh内容:
#!/bin/bash
echo "File Name: $0"
echo "First Parameter : $1"
echo "First Parameter : $2"
echo "Quoted Values: $@"
echo "Quoted Values: $*"
echo "Total Number of Parameters : $#"
输出内容:
注意:在为shell脚本传递的参数中如果包含空格,应该使用单引号或者双引号将该参数括起来,以便于脚本将这个参数作为整体来接收。
$*
和 $@
的区别(别看了,你现在也看不懂这个)
$*
和 $@
都表示传递给函数或脚本的所有参数,不被双引号(“ “)包含时,都以"$1" "$2" … "$n"
的形式输出所有参数。但是当它们被双引号(“ “)包含时,”$*
“ 会将所有的参数作为一个整体,以”$1 $2 … $n
“的形式输出所有参数;”$@
“ 会将各个参数分开,以"$1" "$2" … "$n"
的形式输出所有参数。
退出状态**$?**
的用法
**$?**
可以获取上一个命令的退出状态。所谓退出状态,就是上一个命令执行后的返回结果。
示例:
if[[ $?!=0]];then
echo "error"
exit1;
fi
退出状态是一个数字,一般情况下,大部分命令执行成功会返回 0,失败返回 1。不过,也有一些命令返回其他值,表示不同类型的错误。
监听redis(默认端口6379)的监控脚本
这里给一个监听redis(默认端口6379)的监控脚本:
#!/bin/bash
ListeningPort=$(netstat -an | grep ":6379" | awk '$1 == "tcp" && $NF == "LISTEN" {print $0}' | wc -l)
if [ $ListeningPort -eq 0 ]; then
{
echo "Redis has stopped, restart..."
# 如果6379端口down了,重启服务
/etc/init.d/redis restart
}
else
{
echo "6379端口正常"
}
fi