可以说,shell使用的熟练程度反映了用户对Unix/Linux使用的熟练程度。
如果你运行的是 Unix 或 Linux 系统,例如 Ubuntu,Red Hat,SUSE Linux,还有macOS,都是内置了 bash shell 的,所以你不需要额外配置所谓的开发环境。
Shell有两种执行命令的方式:
交互式:用户输入一条命令,shell就解释一条命令。
批处理:事先写一个脚本,其中有很多条命令,shell一次性把这些命令执行完,不必一次次地敲命令。
shell脚本是解释执行的,不需要编译,shell程序从脚本中一行行读取并执行这些命令,相当于一个用户把脚本中的命令一行行敲到shell提示符下执行。
1 查询shell版本:
echo $BASH_VERSION
2 type 根据输入单词来显示命令的类型:
别名
方法
shell内置命令
关键字
文件
type cd 查看cd指令基本信息
type -a cd 查看更加详细的信息
3 PATH命令:
export PATH=$PATH 把当前目录加到PATH环境下
4 创建脚本:
文件名为hello.sh
#!/bin/bash
echo "Hello World"
exit 0
#!/bin/bash:通常情况下,脚本的默认第一行代码就是它。“#!”又被成为shebang。它用来告诉系统的解释器来执行脚本。除了 bash,我们还可以PHP,Perl等其他脚本。
echo "Hello World"
:echo 是一个内置的命令,用来表示标准输出。
exit 0:表示脚本结束退出。
5 运行脚本的方法:
作为可执行程序
将上面的代码保存为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告诉系统说,就在当前目录找。
通过这种方式运行bash脚本,第一行一定要写对,好让系统查找到正确的解释器。
作为解释器参数
可以在命令行下,进入到脚本文件存在的目录,也可以是在任意目录下,但是,在
执行的时候,文件的路径就应该为绝对路径。
bash /home/sunyubo/vim/hello.sh
6 shell变量
定义变量时,变量名不加美元符号($),如:
variableName="value"
- 首个字符必须为字母(a-z,A-Z)。
- 中间不能有空格,可以使用下划线(_)。
- 不能使用标点符号。
- 不能使用bash里的关键字(可用help命令查看保留关键字)。
myUrl="http://see.xidian.edu.cn/cpp/linux/" myNum=100
your_name="mozhiyan" echo $your_name echo ${your_name}
变量名外面的花括号是可选的,加不加都行,加花括号是为了帮助解释器识别变量的边界,比如下面这种情况:
for skill in Ada Coffe Action Java do echo "I am good at ${skill}Script" done
如果不给skill变量加花括号,写成echo "I am good at skillScript当成一个变量(其
值为空),代码执行结果就不是我们期望的样子了。
推荐给所有变量加上花括号,这是个好的编程习惯。
重新定义变量
已定义的变量,可以被重新定义,如:myUrl="http://see.xidian.edu.cn/cpp/linux/" echo ${myUrl} myUrl="http://see.xidian.edu.cn/cpp/shell/" echo ${myUrl}
只读变量
使用 readonly 命令可以将变量定义为只读变量,只读变量的值不能被改变。
下面的例子尝试更改只读变量,结果报错:
#!/bin/bash myUrl="http://see.xidian.edu.cn/cpp/shell/" readonly myUrl myUrl="http://see.xidian.edu.cn/cpp/danpianji/"
/bin/sh: NAME: This variable is read only.
删除变量
使用 unset 命令可以删除变量。语法:unset variable_name
变量被删除后不能再次使用;unset 命令不能删除只读变量。
举个例子:
#!/bin/sh myUrl="http://see.xidian.edu.cn/cpp/u/xitong/" unset myUrl echo $myUrl
上面的脚本没有任何输出。
变量类型
运行shell时,会同时存在三种变量:1) 局部变量
局部变量在脚本或命令中定义,仅在当前shell实例中有效,其他shell启动的程序不能访问局部变量。2) 环境变量
所有的程序,包括shell启动的程序,都能访问环境变量,有些程序需要环境变量来保证其正常运行。必要的时候shell脚本也可以定义环境变量。3) shell变量
shell变量是由shell程序设置的特殊变量。shell变量中有一部分是环境变量,有一部分是局部变量,这些变量保证了shell的正常运行
7 shell特殊变量
前面已经讲到,变量名只能包含数字、字母和下划线,因为某些包含其他字符的
变量有特殊含义,这样的变量被称为特殊变量。
$echo $$
变量 | 含义 |
---|---|
$0 | 当前脚本的文件名 |
$n | 传递给脚本或函数的参数。n 是一个数字,表示第几个参数。例如,第一个参数是2。 |
$# | 传递给脚本或函数的参数个数。 |
$* | 传递给脚本或函数的所有参数。 |
$@ | 传递给脚本或函数的所有参数。被双引号(" ")包含时,与 $* 稍有不同,下面将会讲到。 |
$? | 上个命令的退出状态,或函数的返回值。 |
$$ | 当前Shell进程ID。对于 Shell 脚本,就是这些脚本所在的进程ID。 |
命令行参数
运行脚本时传递给脚本的参数称为命令行参数。命令行参数用 1 表示第一个参
数,$2 表示第二个参数,依次类推。
请看下面的脚本:
#!/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 : $#"
$./test.sh Zara Ali File Name : ./test.sh First Parameter : Zara Second Parameter : Ali Quoted Values: Zara Ali Quoted Values: Zara Ali Total Number of Parameters : 2
@ 的区别
@ 都表示传递给函数或脚本的所有参数,不被双引号(" ")包含时,都以"2" … "$n"
的形式输出所有参数。
但是当它们被双引号(" ")包含时,"1 n"的形式输出所有参数;"1" "n" 的形式输出
所有参数。
下面的例子可以清楚的看到 @ 的区别:
#!/bin/bash echo "\$*=" $* echo "\"\$*\"=" "$*" echo "\$@=" $@ echo "\"\$@\"=" "$@" echo "print each param from \$*" for var in $* do echo "$var" done echo "print each param from \$@" for var in $@ do echo "$var" done echo "print each param from \"\$*\"" for var in "$*" do echo "$var" done echo "print each param from \"\$@\"" for var in "$@" do echo "$var"
- done
$*= a b c d "$*"= a b c d $@= a b c d "$@"= a b c d print each param from $* a b c d print each param from $@ a b c d print each param from "$*" a b c d print each param from "$@" a b c d
退出状态
$? 可以获取上一个命令的退出状态。所谓退出状态,就是上一个命令执行后的返回结果。退出状态是一个数字,一般情况下,大部分命令执行成功会返回 0,失败返回 1。
不过,也有一些命令返回其他值,表示不同类型的错误。
下面例子中,命令成功执行:
$./test.sh Zara Ali File Name : ./test.sh First Parameter : Zara Second Parameter : Ali Quoted Values: Zara Ali Quoted Values: Zara Ali Total Number of Parameters : 2 $echo $? 0 $