目 录
一、Shell脚本概述
概念:
- 将要执行的命令按顺序保存到一个文本文件
- 给该文件可执行权限
- 可结合何种shell控制语句以完成更复杂的操作
应用场景:
- 重复性操作、交互性任务、批量事务处理、服务运行状态监控、定时任务执行等。
①Shell的作用
1.什么是shell?
- 就是与内核沟通的界面、应用程序、KDE等等。比如你要播放音乐,你的计算机通过你在shell输入的打开音乐的命令,shell在告诉操作系统的内核用户希望打开音乐,内核在通过CPU调度、内存管理、磁盘输入等工作,将硬件调动,这时硬件声卡才会工作,硬件才是实际的工作者。
- shell是一个特殊的应用程序,它介于操作系统内核和用户之间,充当了一个“命令解释器”的角色,负责接收用户输入的操作指令并进行解释,将需要执行的操作传递给内核执行,并输出执行结果。
2.Linux中有哪些shell?
- 检查一下/etc/shells这个文件,里面就是Linux所包含的shell。
- 最常用的就是bash、tcsh、csh、nologin这些shell。这些shell都是依据Linux发展者的不同创造出来的不同版本。
bash:基准于GNU的框架下发展出的Shell。
csh:语法有点类似于c语言的Shell.
tcsh:整合了csh,提供更多的功能。
sh:已经被bash所替换。
nologin:奇怪的shell, 这个shell可以让用户无法登录主机。
注:bash(bin/bash)是目前大多是Linux版本采用的默认shell
3.为什么系统上合法的Shel1要写入/etc/she1ls这个文件呢?
- 这是因为系统某些服务运行过程中,会去检查用户能够使用的Shell,而这些shell的查询就是借由/etc/shells这个文件。
4.用户什么时候可以取得shell来工作?用户默认会取得哪一个shell?
- 当用户登录的时候,系统就会给我一个shell让我来用,而这个登录取得的Shell就记录在/etc/passwd这个文件内
②Shell编程规范
Shell脚本的构成
-
脚本申明(解释器) :若第一行为“!/bin/bash",表示此行以下的代码语句是通过/bin/bash程序来解释执行,#!/bin/bash为默认解释器。还有其它类型的解释器,比如# !/usr/bin/python. #! /usr/bin/expect.
-
注释信息:以“#"开头的语句表示为注释信息,被注释的语句在运行脚本时不会被执行。
-
可执行语句:比如echo命令,用于输出”"之间的字符串
vim /root/first. sh
#! /bin/bash
#! This is my first Shell-Script.
cd /boot
echo "当前的目录位于:"
pwd
echo "其中以vml开头的文件包括:"
ls -lh vml*
Shell脚本的执行
方法一:指定路径的命令,要求文件必须有x权限。
chmod +x / root/first. sh
指定绝对路径: /root/first.sh
指定相对路径: ./first.sh
方法二:指定Shell来解释脚本,不要求文件必须有x权限。
sh脚本路径: sh first. sh
source脚本路径: . first.sh 或者source first.sh
执行完此条命令后,此时所在的目录即是脚本里面切换的目录
③重定向与管道
1.管道
管道操作 |
将管道符号“|"左侧的命令输出的结果,作为右侧命令的输入(处理对象),同一行命令中可以使用多个管道.
ps aux I wc-1
echo “abc123” | passwd --stdin zhangsan
注:管道符号配合xargs用法。xargs命令可以通过管道接受字符串,并将接收到的字符串通过空格分割成许多参数(默认情况下是通过空格分割) 然后将参数传递给其后面的命令,作为后面命令的命令行参数
2.重定向
交互式硬件设备
类型 | 设备文件 | 文件描述编号 | 默认设备 |
---|---|---|---|
标准输入 | /dev/stdin | 0 | 键盘 |
标准输出 | /dev/stdout | 1 | 显示器 |
标准错误输出 | /dev/stderr | 2 | 显示器 |
重定向操作
类型 | 操作符 | 用途 |
---|---|---|
重定向输入 | < | 从指定文件读取数据 |
重定向输出 | > | 将标准输出结果保存到指定的文件,并且覆盖原有文件 |
>> | 将标准输出追加到指定的文件的尾部,不覆盖原有内容 | |
标准错误输出 | 2> | 将错误信息保存到指定文件,并且覆盖原有文件 |
2>> | 将错误信息追加到指定文件的尾部,不覆盖原有内容 | |
混合输出 | &> 和 2>&1 | 将标准输出,标准错误保存到同一文件中 |
举例
echo “123456”>1.txt
cat 1.txt
echo “654321”>1.txt
cat 1.txt
echo “123456”>>1.txt
cat 1.txt
ls -lh > log.txt 2>&1 等同于 ls -lh &> log.txt
将1.txt里面的内容删光
> (将标准输出结果保存到指定的文件)
2>&1(将标准错误输出重定向到标准输出)
&>(将标准输出、标准错误输出保存到同一文件)
二、Shell脚本变量
1.变量=会变化的值
2.常量=不会变化的值
变量的作用:
- 用来存放系统和用户需要使用的特定参数(值)
- 变量名:使用固定的名称,由系统预设或用户定义
- 变量值:能够根据用户设置、系统环境的变化而变化
变量类型: 自定义变量和特殊变量
①自定义变量
由用户自己定义、修改和使用 |
定义新变量
变量名=变量值 #变量名以字母或下划线开头,区分大小写,建议全大写
echo $变量名 #查看变量的值
等号(=)用来给变量赋值
等号(=)运算符左边是一个变量名,等号(=)运算符右边是存储在变量中的值
赋值使用引号及其每种引号的作用
- 双引号:允许通过$符号引用其他变量值
- 单引号:禁止引用其他变量值, $视为普通字符
- 反撇号:命令替换,提取命令执行后的输出结果,``和$(…)作用相同
从键盘输入内容为变量赋值
方法① read命令获取输入内容
read -p 提示信息 变量名
echo $变量名
方法② 在脚本中编写,在命令行实现read获取
vim GAME.sh
#!/bin/bash
read -p “请输入你玩的游戏名字” GAME
echo "--------"
read -p “请输入他玩的游戏名字” GAME2
echo "--------"
echo "你玩的游戏名字是"
echo $GAME
echo "他玩的游戏名字是"
echo $GAME2
执行结果:
设置变量的作用范围
我们使用pstree查看shell环境,输入bash命令进入第二层子shell环境,以此内推进入好多层shell环境。
退出使用exit或者ctrl+d。
使用bash进入子shell环境:
默认下,新定义的变量只在当前的shell环境下有效,我们称之为局部变量,若我们想要在其他的shell环境下有效,我们需要将新定义的变量设置成全局变量。
可以通过内部命令export将指定的新定义的变量变成全局变量。
格式1:export 变量名
格式2:export 变量名=变量值
整数变量的算术计算
运算符:+ 加法、- 减法、* 乘法、/ 除法、% 取余
常见表达式:
expr 变量1 运算符 变量2 (expr是整数运算)
i=$(expr 12 \* 5)
i=$((10 * 5))
i=$[10 * 4]
let i=10*3
i++ 相当于 i=$[$i+1]
i-- 相当于 i=$[$i-1]
i+=2 相当于 i=$[$i+2]
②特殊变量
环境变量(只读变量、位置变量、预定义变量) |
环境变量:
1 由系统提前创建,用来设置用户的工作环境
2 配置文件:/etc/profile(全局生效)、~/.bash_profile(当前用户环境)
常见环境变量
- 使用 env 命令可以查看到当前工作环境下的环境变量
- 变量USER表示用户名称,HOME表示用户的宿主目录,LANG表示语言和字符集,PWD表示当前所在的工作目录,变量PATH表示可执行程序的默认搜索路径
Ⅰ:PATH
echo $PATH #查看当前搜索路径
PATH="$PATH:/root" #将/root目录添加到搜索路径
export PATH="$PATH:/root" #输出为全局环境变量
first.sh
Ⅱ:只读变量
用户的变量值不允许被修改
readonly 命令设置只读变量
readonly PRODUCT #设置为只读变量
echo $PRODUCT
PRODUCT=Python #只读变量不可以被重新赋值
unset PRODUCT #只读变量不可以被删除,unset 命令用于删除变量,但无法删除只读变量,只能重启系统解决
Ⅲ:位置变量
- 当执行命令行操作时,第一个字段表示命令名或脚本程序名,其余的字符串参数按照从左到右的顺序依次赋值给位置变量。
- $n:n为数字,$0代表命令本身,1~9代表第一个到第九个参数,十以上的参数需要使用大括号表示,比如第十个参数为 ${10}
vim lic.sh
#!/bin/bash
echo $1
echo $2
echo $1 + $2
Ⅳ:预定义变量
$*、$@:表示命令或脚本要处理的参数。
$*:把所有参数看成以空格分隔的一个字符串整体,代表"$1 $2 $3 $4"。
$@:把各个参数加上双引号分隔成n份的参数列表,每个参数是独立 的,代表"$1" "$2" "$3" "$4"。
$0:表示当前执行的脚本或命令的名称。
$#:表示命令或脚本要处理的参数的个数。
$?:表示前一条命令或脚本执行后的返回状态码,返回值为0表示执行正确,返回任何非0值均表示执行出现异常。
也常被用于Shell脚本中return退出函数并返回的退出值。
举例:
vim mybak.sh
#!/bin/bash
time=backup-`date +%F`.tar.gz
tar zcf $time $* &> /dev/null #/dev/null表示的是一个黑洞文件,通常用于丢弃不需要的数据输出
echo "已执行 $0 脚本,"
echo "共完成 $# 个对象的备份"
echo "具体内容包括: $*"
chmod +x mybak.sh
./mybak.sh /etc/passwd /etc/shadow