shell base
这里写文章的前言:
一个简单的开头,简述这篇文章讨论的问题、目标、人物、背景是什么?并简述你给出的答案。
可以说说你的故事:阻碍、努力、结果成果,意外与转折。
📝 主旨内容
https://www.runoob.com/linux/linux-shell.htmlhttps://yelog.org/2018/09/08/shell-command/
高级语言–>执行–>解释器–>机器代码
编程逻辑处理方式:
顺序执行、循环执行、选择执行
shell编程:过程式、解释执行
编程语言的基本结构:
各种系统命令的组合
数据存储:变量、数组
表达式:a + b
语句:if
shell脚本的用途有:
- 自动化常用命令
- 执行系统管理和故障排除
- 创建简单的应用程序
- 处理文本或文件
格式要求:首行shebang机制
创建shell脚本
1、使用编辑品来创建,首行声明#!/bin/bash
2、运行脚本,需要有执行权限
- 相对/绝对路径
- 直接运行解释器,将脚本作为解释程序的参数 bash sh ./file.sh
脚本规范
1、第一行一般为调用使用的语言
2、程序名,避免更改文件名为无法找到正确的文件
3、版本号
4、更改后的时间
5、作者相关信息
6、该程序的作用,及注意事项
7、最后是各版本的更新简要说明
脚本的基本结构
脚本调试
注意:当脚本写好之后先检查语法在调试执行过程建议初学者可以使用shellcheck检查脚本
执行shell前 先检查语法
检查过程,调试执行,可以看到详细的执行步骤
变量
- Shell中变量命名法则: 1、不能使程序中的保留字:例如if, for 2、只能使用数字、字母及下划线,且不能以数字开头 3、见名知义 4、统一命名规则:驼峰命名法
- Shell中命名建议规则: 1、变量名大写 2、局部变量小写 3、函数名小写 4、用英文名字,并体现出实际作用
根据变量的生效范围等标准划分下面变量类型
局部变量:生效范围为当前shell进程;对当前shell之外的其它shell进程,
包括当前shell的子shell进程均无效
环境变量:生效范围为当前shell进程及其子进程
本地变量:生效范围为当前shell进程中某代码片断,通常指函数
位置变量:$1, $2, …来表示,用于让脚本在脚本代码中调用通过命令行传
递给它的参数
特殊变量:
$?, $0, $*, $@, $#,$$
变量赋值:name=’value’
可以使用引用value
(1) 可以是直接字串:name=”root”
(2) 变量引用:name=”$USER”
(3) 命令引用:name=`COMMAND` name=$(COMMAND)
变量引用:${name} 或者 $name
“ “ 弱引用,其中的变量引用会被替换为变量值
‘ ‘ 强引用,其中的变量引用不会被替换为变量值,而保持原字符串
显示已定义的所有变量:set
删除变量:unset name练习
1、编写脚本/root/bin/systeminfo.sh,显示当前主机系统信息,包括主机名,IPv4地址,操作系统版本,内核版本,CPU型号,内存大小,硬盘大小
2、编写脚本/root/bin/backup.sh,可实现每日将/etc/目录备份到/root/etcYYYY-mm-dd中
3、编写脚本/root/bin/disk.sh,显示当前硬盘分区中空间利用率最大的值
4、编写脚本/root/bin/links.sh,显示正连接本主机的每个远程主机的IPv4地址和连接数,并按连接数从大到小排序
环境变量
- 变量声明、赋值: export name=VALUE declare -x name=VALUE
- 变量引用:$name, ${name}
- 显示所有环境变量: env printenv export declare -x
- 删除变量:unset name
- bash内建的环境变量
只读和位置变量
只读变量:只能声明,但不能修改和删除
- 声明只读变量: readonly name declare -r name
- 查看只读变量: readonly -p
为什么要只有只读变量
#比如 $UID 如果UID可以赋值,那UID=0 是管理员就危险了
位置变量:在脚本代码中调用通过命令行传递给脚本的参数
$1, $2, … 对应第1、第2等参数,shift \[n\]换位置
$0 命令本身
$\* 传递给脚本的所有参数,全部参数合为一个字符串
$@ 传递给脚本的所有参数,每个参数为独立字符串
$# 传递给脚本的参数的个数
注意:$@ $\* 只在被双引号包起来的时候才会有差异
set – 清空所有位置变量
退出状态
- 进程使用退出状态来报告成功或失败 0 代表成功,1-255代表失败 $? 变量保存最近的命令退出状态
- 例如:
ping -c1 -W1 host &> /dev/null echo $?
退出状态码
bash自定义退出状态码
exit \[n\]:自定义退出状态码
注意:脚本中一旦遇到exit命令,脚本会立即终止;终止退出状态取决于exit命令后面的数字
注意:如果未给脚本指定退出状态码,整个脚本的退出状态码取决于脚本中执行的最后一条命令的状态码
若执行失败,$?返回的值是非0 false
算术运算
- bash中的算术运算:help let
+, -, *, /, %取模(取余), **(乘方),乘法符号有些场景中需要转义 实现算术运算: (1) let var=算术表达式 (2) var=$\[算术表达式\] (3) var=$((算术表达式)) (4) var=$(expr arg1 arg2 arg3 …) (5) declare –i var = 数值 (6) echo ‘算术表达式’ | bc
- bash有内建的随机数生成器变量:$RANDOM(0-32767)
示例:
生成 0 - 49 之间随机数 echo $[$RANDOM%50]
赋值
- 增强型赋值:
+=, -=, *=, /=, %=
- let varOPERvalue 例如:let count+=3 自加3后自赋值
- 自增,自减: let var+=1 let var++ let var-=1 let var–
练习
1、编写脚本/root/bin/sumid.sh,计算/etc/passwd文件中的第10个用户和第20用户的ID之和
2、编写脚本/root/bin/sumspace.sh,传递两个文件路径作为参数给脚本,计算这两个文件中所有空白行之和
3、编写脚本/root/bin/sumfile.sh,统计/etc, /var, /usr目录中共有多少个一级子目录和文件
逻辑运算
true,false 1,0
与运算(相乘)和0相与为0 两个1相与为1
1 与 1 = 1
1 与 0 = 0
0 与 1 = 0
0 与 0 = 0
或运算(相加)和1相或为1 两个0相或为0
1 或 1 = 1
1 或 0 = 1
0 或 1 = 1
0 或 0 = 0
非: ! 取反
!1=0 !true
!0=1 !false
短路运算
短路与
第一个为0,结果必定为0
第一个为1,第二个必须要参与运算
短路或
第一个为1,结果必定为1
第一个为0,第二个必须要参与运算
异或:^
异或的两个值,相同为假,不同为真
条件测试
- 判断某需求是否满足,需要由测试机制来实现 专用的测试表达式需要由测试命令辅助完成测试过程
- 评估布尔声明,以便用在条件性执行中
- 若真,则返回0
- 若假,则返回1
- 测试命令:
- test EXPRESSION
[ EXPRESSION ][[ EXPRESSION ]]
注意:EXPRESSION前后必须有空白字符
条件性的执行操作符
- 根据退出状态而定,命令可以有条件地运行
- && 代表条件性的AND THEN
- || 代表条件性的OR ELSE
- 例如:
test命令
- 长格式的例子:
test "$A" = "$B" && echo "Strings are equal"test “$A”-eq “$B” && echo "Integers are equal"
- 简写格式的例子:
[ "$A" = "$B" ] && echo "Strings are equal"[ "$A" -eq "$B" ] && echo "Integers are equal"
bash的数值测试
\-v VAR
变量VAR是否设置
bash的字符串测试
注意:用于字符串比较时的用到的操作数都应该使用引号
练习
1、编写脚本/root/bin/argsnum.sh,接受一个文件路径作为参数;如果参数个数小于1,则提示用户“至少应该给一个参数”,并立即退出;如果参数个不小于1,则显示第一个参数所指向的文件中的空白行数
2、编写脚本/root/bin/hostping.sh,接受一个主机的IPv4地址做为参数,测试是否可连通。如果能ping通,则提示用户“该IP地址可访问”;如果不可ping通,则提示用户“该IP地址不可访问”
3、编写脚本/root/bin/checkdisk.sh,检查磁盘分区空间和inode使用率,如果超过80%,就发广播警告空间将满
bash文件测试
存在性测试
bash文件权限测试
文件权限测试:
r FILE:是否存在且可读 -w FILE: 是否存在且可写 -x FILE: 是否存在且可执行文件特殊权限测试:u FILE:是否存在且拥有suid权限 -g FILE:是否存在且拥有sgid权限 -k FILE:是否存在且拥有sticky权限
bash 文件属性测试
文件大小测试:
-s FILE: 是否存在且非空
文件是否打开:-t fd: fd 文件描述符是否在某终端已经打开 -N FILE:文件自从上一次被读取之后是否被修改过 -O FILE:当前有效用户是否为文件属主 -G FILE:当前有效用户是否为文件属组
双目测试:FILE1 -ef FILE2: FILE1是否是FILE2的硬链接 FILE1 -nt FILE2: FILE1是否新于FILE2(mtime) FILE1 -ot FILE2: FILE1是否旧于FILE2
组合测试
第一种方式:COMMAND1 && COMMAND2 并且 COMMAND1 || COMMAND2 或者 ! COMMAND 非 如:[ -f "$FILE" ] && [[ "$FILE"=~ .*\\.sh$ ]]
第二种方式:EXPRESSION1 -a EXPRESSION2 并且 EXPRESSION1 -o EXPRESSION2 或者 ! EXPRESSION 必须使用测试命令进行,[[ ]] 不支持
示例:[ -z “$HOSTNAME” -o $HOSTNAME "=="localhost.localdomain" ] && hostname node1 [ -f /bin/cat -a -x /bin/cat ] && cat /etc/fstab练习
1、编写脚本per.sh,判断当前用户对指定参数文件,是否不可读并且不可写
2、编写脚本excute.sh ,判断参数文件是否为sh后缀的普通文件,如果是,添加所有人可执行权限,否则提示用户非脚本文件
3、编写脚本nologin.sh和login.sh,实现禁止和允许普通用户登录系统
使用read命令来接受输入
使用read来把输入值分配给一个或多个shell变量
鸡兔同笼
🤗 总结归纳
总结文章的内容
📎 参考文章
- 一些引用
- 引用文章
有关Notion安装或者使用上的问题,欢迎您在底部评论区留言,一起交流~
Loading...
keepalived