Shell编程规范及变量

一、什么是shell

1.1

Shell是一种命令解释器,它在操作系统的最外层,负责直接与用户进行对话,把用户的输入解释给操作系统,并处理各种各样的操作系统的输出结果,输出到屏幕反馈给用户。这种对话方式可是交互也可以是非交互式的,我们所输入的命令计算机是不识别的,这时就需要一种程序来帮助我们进行翻译,变成计算机能识别的二进制程序,同时又把计算机生成的结果返回给我们。

1.2shell的作用

1.2.1

Linux系统中的Shell是一个特殊的应用程序,它介于操作系统内核与用户之间,充当了一个“命令解释器”的角色,负责接收用户输入的操作指令(命令)并进行解释,将需要执行的操作传递给内核执行,并输出执行结果。 
常见的 Shell 解释器程序有很多种,使用不同的 Shell 时,其内部指令、命令行提示符等方面会存在一些区别。通过/etc/shells 文件可以了解当前系统所支持的 Shell 脚本种类。

[root@admin ~]# cat /etc/shells
/bin/sh
/bin/bash
/sbin/nologin
/usr/bin/sh
/usr/bin/bash
/usr/sbin/nologin
/bin/tcsh
/bin/csh

这里来解释一下 cat /etc/shells输出的命令结果

/bin/sh  #是bash命令的软链接(已经被/bin/bash所替换)

/bin/bash  #基于GNU框架下发展出的Shell。

/usr/bin/sh  #已经被bash所替换

/usr/bin/bash  #centos和redhat系统默认使用bash shell

/bin/tcsh   #csh的增强版,与csh完全兼容,整合了csh,提供了更多功能

/bin/csh  #已经被/bin/bash所替换

注意:nologin:奇怪的shell,这个shell可以让用户无法登录主机

bash(/bin/bash)是目前大多数Linux版本采用的默认Shell

拓展问题

  1. 为什么我们的系统上合法的shell要写入/etc/shells这个文件?

答:这是因为系统某些服务在运行过程中,会去检查用户能够使用的shells,而这些shell的查询就是借由/etc/shells这个文件。说简单点,能够使用的shells全部在/etc/shells这个文件里,如果文件没有shells命令你的程序就运行不了,所以要写入进去。

2.用户户什么时候可以取得shell来工作?还有我这个默认会取得哪一个shell?

答:当我登录的时候,系统就会给我shell让我来工作,而这个登录取得的shell就记录在/etc/passwd这个文件内。

1.2.2shell脚本是什么

shell脚本就是说我们把原来 linux 命令或语句放在一个文件中,然后通过这个程序文件去执行时,我们就说这个程序为 shell 脚本或 shell 程序;我们可以在脚本中输入一系统的命令以及相关的语法语句组合,比如变量,流程控制语句等,把他们有机结合起来就形成了一个功能强大的 shell 脚本。

总结:将需要执行的命令保存到一个文件中,按照顺序执行,它不需要编译,它是解释型的

1.2.3shell脚本能干什么

自动化完成软件的安装部署,如安装部署LAMP架构服务
自动化完成系统的管理,如批量添加用户
自动化完成备份,如数据库定时备份
自动化的分析处理,如网站访问量

1.2.4shell脚本使用场景

在需要完成大量复杂、重复性的工作时,不需要在命令行重复执行命令,直接运行shell脚本即可,大大的节省了时间提高了效率

二、shell脚本的构成

2.1

第一行为“#!/bin/bash”,脚本申明(默认解释器):表示此行以下的代码语句是通过/bin/bash程序来执行。还有其他类型的解释器。

比如#!/usr/bin/python、#!/usr/bin/expect
注释信息:以“#”开头的语句表示为注释信息,被注释的语句在脚本运行时不会被执行
可执行语句:如echo命令,用于输出“ ”之间的字符串。

示例:

[root@admin MyScript]# vi chongdx.sh 
#!/bin/bash
#这个脚本是练习重定向的
touch a.txt
echo "123 456 789" > a.txt
echo -n   "a.txt的信息是:"
cat a.txt
read -p "请输入你想替代a.txt文件内容的信息:" a
echo $a > a.txt
cat a.txt
[root@admin MyScript]# chmod +x chongdx.sh 
[root@admin MyScript]# ./chongdx.sh 
a.txt的信息是:123 456 789
请输入你想替代a.txt文件内容的信息:

这个示例就基本上把创建shell脚本的过程展示出来了。

创建shell脚本的具体过程:

第一步:创建一个包含命令和控制结构的文件。
第二步:修改这个文件的权限使它可以执行     # 使用 chmod +x  文件名
第三步:检测语法错误           
第四步:执行 ./文件名

2.2shell脚本执行

shell 脚本的执行通常有以下几种方式
方法一:当前路径(决定路径与相对路径)下执行脚本(要有执行权限)
/home/first.sh 或者 ./first.sh

方法二:sh 、bash脚本文件路劲(这种方式可以不对脚本文件添加执行权限)
bash first.sh 或 sh first.sh

方法三:source 脚本文件路劲(可以没有执行权限)
source first.sh 

方法四:其他方法
sh < first.sh 或者 cat first.sh |sh(bash)

三、重定向与管道操作

3.1什么是重定向

将原本要输出到屏幕的数据信息,重新定向到某个指定的文件中。 比如:每天凌晨定时备份数据,希望将备份数据的结果保存到某个文件中。这样第二天通过查看文件的内容就知道昨天备份的数据是成功还是失败。

简单来说,就是不输出到默认位置,而是输出到指定位置。

3.2重定向的三种交互设备文件

标准输入(STDIN):默认的设备是键盘,设备文件/dev/stdin,文件编号为 0,命令将从标准输入文件中读取在执行过程中需要的输入数据。 
标准输出(STDOUT):默认的设备是显示器,设备文件/dev/stdout,文件编号为 1,命令将执行后的输出结果发送到标准输出文件。 
标准错误(STDERR):默认的设备是显示器,设备文件/dev/stderr 文件编号为 2,命令将执行期间的各种错误信息发送到标准错误文件。

从键盘接收用户输入的各种命令字串、辅助控制信息,并将命令结果输出到屏幕上;如果命令执行出错,也会将错误信息反馈到屏幕上

简单来说就是:

- 标准输入:从该设备接收用户输入的数据
- 标准输出:通过该设备向用户输出数据
- 标准错误:通过该设备报告执行出错信息

3.3重定向的四种类型操作符

类型操作符用途
重定向输入< 从指定的文件读取数据,而不是从键盘输入    
重定向输出  1>将输出结果保存到指定的文件(覆盖原有内容)
>>将输出结果追加到指定的文件尾部
标准错误输出2> 将错误信息保存到指定的文件(覆盖原有内容)
2>> 标准错误输出结果追加到指定的文件尾部
混合输出&>无论对错都可以重定向    将标准输出、标准错误的内容保存到同一个文件中

3.3.1重定向输出

若重定向输出的目标文件不存在,则会新建该文件,然后将前面命令的输出结果保存到该文件中;若目标文件已经存在,则将输出结果覆盖或追加到文件中。

简单来说: >意思是当原来文件有内容的话,原来的内容会被覆盖

>>意思是当原来文件中有内容的话,新的内容会追加到里面而不会覆盖原来的内容

示例1:首先uname  -p  命令查看主机处理器类型,并把信息给文件

然后cat text.txt命令就是查看文件内容,可以看到原来的内容没有被覆盖

[root@admin wangam]# uname -p >text1.txt 
[root@admin wangam]# cat text1.txt 
x86_64

示例2:首先uname -r 命令就是内核发行版本号

然后查看文件内容,可以看到只是追加到原来的内容后面,并没有覆盖

[root@admin wangam]# uname -r >>text1.txt 
[root@admin wangam]# cat text1.txt 
x86_64
3.10.0-693.el7.x86_64

3.3.2重定向输入

重定向输入指的是将命令中接收输入的途径由默认的键盘改为指定的文件,而不是等待 从键盘输入。重定向输入使用“<”操作符。 通过重定向输入可以使一些交互式操作过程能够通过读取文件来完成。例如,使用 passwd 命令为用户设置密码时,每次都必须根据提示输入两次密码字串,非常烦琐,若改 用重定向输入将可以省略交互式的过程,而自动完成密码设置(结合 passwd 命令的“--stdin” 选项来识别标准输入)。
 

示例1:创建一个文件并在此之前把内容放到文件里,显示1.txt文件,其实第三个命令和第二个命令是一样的。唯一不同的是1.txt是作为输入设备

[root@admin MyScript]# echo "1234" > 1.txt
[root@admin MyScript]# cat 1.txt
1234
[root@admin MyScript]# cat <1.txt
1234

示例2:0作为分界符,只要不输入0就会一直输入数据从而显示到屏幕上

[root@admin MyScript]# cat <<0
> 1
> 2
> 2
> 0
1
2
2

示例3:可以引申一下,和重定向输出结合起来,将键盘上输入的内容输出到文件里

[root@admin MyScript]# cat << 0 > 1.txt
> 121
> djdj
> 0
[root@admin MyScript]# cat 1.txt 
121
djdj

3.3.3 错误重定向

错误重定向指的是将执行命令过程中出现的错误信息(如选项或参数错误等)保存到指 定的文件,而不是直接显示在屏幕上。错误重定向使用“2>”操作符

2个作用

1.在实际应用中,错误重定向可用来收集程序执行的错误信息,为排错提供依据
2.还可以将无关紧要的错误信息重定向到空文件/dev/null 中,以保持脚本输出的简洁

注意:

使用“2>”操作符时,会像使用“>”操作符一样覆盖目标文件的内容,若要追加内容而不是覆盖文件,则应改用“2>>”操作符

当命令输出的结果可能既包括标准输出(正常执行)信息,又包括错误输出信息时,可 以使用操作符“>”“2>”将两类输出信息分别保存到不同的文件,也可以使用“&>”操作符将两类 输出信息保存到同一个文件

/dev/null

/dev/null:把它看作"黑洞",所有写入它的内容都会永远丢失. 而尝试从它那儿读取内容则什么也读不到. 然 而 /dev/null 对命令行和脚本都非常的有用

原本的信息是不会改动的,而且这里想看/dev/null也什么都看不到

&>和>&

&是等同于的意思

1>&2把标准输出重定向到标准错误

2>&1把标准错误重定向到标准输出

示例1:简单来说就是:把出现前面错误的这个信息,重新发给1.txt

本来1.txt的内容是www.txt给的,但是后面又重新把这个标准错误给了前面的>,然后给了1.txt(个人理解)

示例2:这里改了1.txt的内容。我理解的执行顺序是:本身没有文件,重定向到1.txt是个错误,这个信息给了1.txt ,1.txt在重定向输出给了2>,2>再把错误信息给了2.txt

四、管道操作

管道(pipe)操作为不同命令之间的协同工作提供了一种机制,位于管道符号“|”左侧的命令输出的结果,将作为右侧命令的输入(处理对象),同一行命令中可以使用多个管道。 
在 Shell 脚本应用中,管道操作通常用来过滤所需要的关键信息。

五、shell变量及赋值

5.1shell的变量

变量是用来临时保存数据的,并且该数据时可以变化的,任何一个语言都离不开变量,如果某个内容需要多次使用并且会重复出现,这样就可以使用变量了,如果需要修改直接修改变量就可以了
常见 Shell 变量的类型包括自定义变量、环境变量、只读变量、位置变量、预定义变量

5.1.1自定义变量

Bash中的变量操作相对比较简单,不像其他高级编程语言(如C/C++、Java等)那么复杂。在定义一个新的变量时,一般不需要提前进行声明,而是直接指定变量名称并赋给初始值(内容)即可
格式:变量名=变量值

变量名:暂时存储数据的地方

变量值:暂时可变化的数据

要求

1.等号两边,不能有空格,变量名称要以字母或者下划线开头,名称中不能包含特殊字符(+,-,*,!等)

2.查看变量值:echo $变量名 ——用echo命令,当然也要在变量名前面加$号,也可以查多个变量

3.取消变量名:unset 变量名

示例1

[root@admin MyScript]# qqq=1
[root@admin MyScript]# www
bash: www: command not found...
[root@admin MyScript]# www=2
[root@admin MyScript]# echo $qqq $www
1 2
[root@admin MyScript]# 

示例2:如果变量和其他字符串连在一起,计算机容易混淆,会无法确定正确的变量名称,会显示空值,所以要加大括号

[root@admin MyScript]# www=200
[root@admin MyScript]# echo ${www}RMB
200RMB
[root@admin MyScript]# 

echo

echo -n //不换行输出

echo -e  //使用转义字符,可以将转义字符的内容给显示出来

常用的转义字符有:

\n 换行  

\t转义后表示插入tab,即制表符

示例3:我们可以看到,转义字符要加双引号或者单引号

[root@admin MyScript]# echo -e \t\tdbajkaf
ttdbajkaf
[root@admin MyScript]# echo -e "\t\tdbajkaf"
		dbajkaf
[root@admin MyScript]# echo -e "\tdbajkaf"
	dbajkaf
[root@admin MyScript]# echo -e '\tdbajkaf'
	dbajkaf

5.1.2交互式定义变量

5.1.2.1read命令

用来提示用户输入信息,从而实现简单的交互过程。执行时将从标准输入设备(键盘)读入 一行内容,并以空格为分隔符,将读入的各字段依次赋值给指定的变量(多余的内容赋值给 最后一个变量)。若指定的变量只有一个,则将整行内容赋值给此变量。
 

示例1:有点奇怪,我既想交互,又想控制时长,最后命令行又出现了我输入的东西,只用选项-p是没有在出现我输入的内容的,之后我发现是-t导致的。

交互式定义变量(read)
-p 提示用户的信息
-n 定义字符数
-s 不显示用户输入的内容,常用于输入密码  read -s -p "input your password:" pass
-t 定义超时时间,超过多长时间没输自动退出

5.1.3变量的作用范围

通常,新定义的变量只能在当前shell环境中有效,因此被称为局部变量,当进入另一个环境时,局部变量就无法使用。例如使用bash

5.1.4全局变量的设定

为了使用户定义的变量在所有的子 Shell 环境中能够继续使用,减少重复设置工作,可以通过内部命令 export 将指定的变量导出为全局变量。用户可以同时指定多个变量名称作为参数(无须使用“$”符号),变量名之间以空格分隔

格式:export  变量名

示例:

5.2双引号,单引号,反撇号

5.2.1双引号

双引号主要起界定字符串的作用,特别是当要赋值的内容中包含空格时,必须以双引号括起来;其他情况下双引号通常可以省略

示例:变量可以等于变量,但不能等于变量值加变量名的组合

[root@admin MyScript]# www=world
[root@admin MyScript]# qqq="hello $www"
[root@admin MyScript]# echo $qqq 
hello world
[root@admin MyScript]# 

[root@admin MyScript]# qqq=hello $www
bash: world: command not found...

5.2.2单引号

当要赋值的内容中包含$、“、\等具有特殊含义的字符时,应使用单引号括起来。
****在单引号的范围内,将无法引用其他变量的值,任何字符均作为普通字符看待。输入什么就显示什么****
但赋值内容中包含单引号(‘)时,需使用\’符号进行转义,以免冲突。

实例:

[root@admin MyScript]# echo $qqq 
world
[root@admin MyScript]# echo '$qqq '
$qqq 
[root@admin MyScript]# 

这个地方变量加上单引号就得不到想要的变量值了,只能当做普通字符串来用。只要记住,单引号是给特殊字符用的

5.2.3

反撇号主要用于命令替换,允许将执行某个命令的屏幕输出结果赋值给变量。
反撇号括起来的范围内必须是能够执行的命令行,否则将会出错

示例:这里就是先通过which useradd命令查找出useradd命令的程序位置,然后再根据查找结果列出文件属性

示例2:主要用途就是将能够显示在屏幕上操作的命令换一个名字

注意:使用反撇号难以在一行命令中实现嵌套命令替换操作,这时可以改用 “$()”来代替反撇号操作,以解决嵌套的问题
rpm -qc $(rpm -qf $(which useradd))

示例3:

5.2.4大扩号(花括号)

大括号里面的内容不能用空格隔开,空格只能用转义字符。

对大括号中的以逗号分割的文件列表进行扩展,如touch{a,b} .txt 结果为a.txt b.txt

对大括号中以点点(……)分割的顺序文件列表七起拓展作用,如touch{a...d} 结果就是四个文件

六、变量运算

在 Bash Shell 环境中,只能进行简单的整数运算,不支持小数运算
整数值的运算主要通过内部命令 expr 进行
运算符与变量之间必须有至少一个空格。

运算内容:加(+)、减(-)、乘(*)、除(/)、取余(%)
运算符号:$(()) 和$[]
运算命令:expr和let
运算工具:bc(系统自带)

6.1expr命令(不仅能运算,还支持输出到屏幕上)

常用的 几种运算符如下所述。  
+:加法运算。 
-:减法运算。 
\*:乘法运算,注意不能仅使用“*”符号,否则将被当成文件通配符。  
/:除法运算。 
%:求模运算,又称为取余运算,用来计算数值相除后的余数。

示例:注意,乘号要转义

6.2$(())和$[]

这两个比较特殊,这两个也能运算,但是不能显示结果,所以要配合echo使用

这里的乘号可以不用转义,而且$也可以省略

6.3let命令

这个命令相比于上面的命令,它的升级之处在于,它可以改变本身值的结果,但是和$[],$()一样,不能显示出来,也要配合echo

6.4bc命令


使用bc进行运算,支持小数运算,但在脚本中不可直接使用否则会进入交互界面,可以用echo结合管道使用,也通常会跟scale使用,scale是表示后面又多少小数点的。
 

要注意格式。

七、环境变量

环境变量指的是出于运行需要而由 Linux 系统提前创建的一类变量,主要用于设置用户的工作环境,包括用户宿主目录、命令查找路径、用户当前目录、登录终端等。
环境变量的值由 Linux 系统自动维护,会随着用户状态的改变而改变。 

使用 env 命令可以查看到当前工作环境下的环境变量,对于常见的一些环境变量应了解其各自的用途

7.1常用的环境变量

变量 USER 表示用户名称,

HOME 表示用户的宿主目录

,LANG 表示语言和字符集,

PWD 表示当前所在的工作目录,

PATH 表示命令搜索路径等、

RANDOM表示随机数,会返回0-32767的整数,

USER表示当前账户的账户名称等,一般都用全大写定义,注意和自定义变量区分

示例:

7.1.1如果自己有脚本,但是不在$PATH目录里面,系统是无法识别的,也就无法使用,所以需要跟上绝对路径使用该脚本

方法一:将脚本的目录加入$PATH

可以看到后面又root了。之前是没有的。这个root是临时的,要永久生效需要编辑/etc/profile文件。

方法二:将你自己写的脚本放到$PATH中的某一个目录

在 Linux 系统中,环境变量的全局配置文件为/etc/profile,在此文件中定义的变量作用 于所有用户。除此之外,每个用户还有自己的独立配置文件(~/.bash_profile)。
修改完了要重新登陆才能生效,如果想立即生效,可以使用source

注意:修改$PATH需要慎重操作,如果找不到了会影响命令的使用!!!

7.2只读变量

Shell 变量中有一种特殊情况,一经设定,其值是不可改变的,这种变量被称为只读变 量。在创建变量的时候可将其设置为只读属性,也可以将已存在的变量设置为只读属性,只 读变量主要用于变量值不允许被修改的情况
只读变量不可以改变值也不可以被删除

示例

7.3位置变量

当执行命令行操作时,第一个字段表示命令名或脚本程序名,其余的字符串参数按照从左到右的顺序依次赋值给位置变量。 
位置变量也称为位置参数,使用$1、$2、$3、…、$9 表示
命令或脚本本身的名称使用“$0”表示
 

示例:编写一个简单创建用户及设置密码的操作

7.4预定义变量

预定义变量是由 Bash 程序预先定义好的一类特殊变量,用户只能使用预定义变量,而 不能创建新的预定义变量,也不能直接为预定义变量赋值。预定义变量使用“$”符号和另一个符号组合表示
$#:表示命令行中位置参数的个数。  
$*:表示所有位置参数的内容,这些内容当做一个整体
$@:表示列出所有位置参数,但是是以单个的形式的列出 
$?:表示前一条命令执行后的返回状态,返回值为 0 表示执行正确,返回任何非 0 值均表示执行出现异常。 
$0:表示当前执行的脚本或程序的名称
$$:表示返回当前进程的进程号
$!:返回最后一个后台进程的进程号
 

示例:

理解$*和$@的区别
$*、$@:表示命令或脚本要处理的参数。
$*:把所有参数看成以空格分隔的一个字符串整体(单字符串)返回,代表"$1 $2 $3 $4"。
$@:把各个参数加上双引号分隔成n份的参数列表,每个参数作为一个字符串返回,代表"$1""$2""$3""$94"。


总结:
$*是将参数全部当做一个整体
$@是将参数每一个都当做单独的个体
 

set:查看系统所有的变量,包括环境变量和自定义变量(没有单独查看自定义变量的命令,可以set管道过滤)

  • 9
    点赞
  • 24
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值