shell的循环控制语句

-  continue:提前结束某次循环,重新开始下一次

-  break:提前结束某层循环

范例:

#求100以内的奇数和
#!/bin/bash
sum=0
for i in `seq 100`;do
if [ $[i%2] -ne 0 ];then
continue
else
let sum+=$i
fi
done
echo $sum
#实现100以内的奇数和
#!/bin/bash
sum=0
for i in `seq 100`;do #``:命令替换
if [ $[i%2] -eq 0 ];then #$[]:shell的算数语法
let sum+=$i #let工具需要借助一个变量存储计算后的值
fi
done echo $sum

shell的循环控制 shift 命令

每执行一次shift命令,都会左移一次。

#比如有三个参数$1 $2 $3,执行一次shift命令后,$1就删除了,$2变成$1,一次类推。
#!/bin/bash
echo $*
shift
echo $*
shift
echo $*
shift
#输出结果:
[root@Centos8 ~]# bash shift.sh 1 2 3 4 5
1 2 3 4 5
2 3 4 5
3 4 5
#利用shift的方式批量创建账号
#!/bin/bash if [$# -eq 0];then #判断位置参数是否是0
echo "请输入需要创建的用户"
exit
fi while [ "$1" ] ;do #[ "$1" ]:判断$1是否为空,只写字符串就表示字符串不为空,为真
if id $1 &> /dev/null; then #如果id命令执行的结果为真($?=0)就表示用户存在了
echo "$1 is exist"
else
useradd $1
echo "$1 is create"
fi
shift
done
echo "all user is create " #使用for的形式创建账号
if [$# -eq 0];then #判断位置参数是否是0
echo "请输入需要创建的用户"
exit
fi for user in $*;do
if id $user &> /dev/null; then #如果id命令执行的结果为真($?=0)就表示用户存在了
echo "$user is exist"
else
useradd $user
echo "$user is create"
fi
done
echo "all user is create "

while 特殊用法 while read

文本的逐行处理。

格式

while read line; do
循环体
done < /PATH/FROM/SOMEFILE
while read line ;do echo $line; done #标准输入,逐行处理

while read line ; do xxxx; done < filename #可以利用标准输入重定向把文件传给它

cat /etc/issue | while read line ;do echo $line; done #用管道

#使用while read判断磁盘分区磁盘的利用率
#!/bin/bash while true ; do #一直执行
df | sed -nr '/^\/dev\/sd/ s#([^ ]+).*([0-9]+)%.*#\1 \2#p' | while read DEV USE;do
if [ $USE -gt 80 ];then
echo "$DEV will be full, USE: $USE"
else
echo "$DEV is health, USE: $USE"
fi
done
sleep 2 #每两秒执行一次
done #sed -nr '/^\/dev\/sd/ s#(^[^ ]+).*([0-9]+)%.*#\1 \2#p' | while read DEV USE;do
#/^\/dev\/sd/ --- 以为/dev/sd开头的(非空格的内容)
#[^ ]+ :取出分区的设备,非空格,一个以上的字符
#\1 \2 --- 后项替换
#查看/sbin/nologin的shell类型的用户名和UID
#!/bin/bash
while read line ; do #通过输入重定向的方式逐行处理内容
if [[ $line =~ /sbin/nologin$ ]];then # [[ =~ ]]:会把会把右边的字符串当成正则
echo $line | cut -d: -f1,3 # -d:指定分隔符 -f:取那几列
fi
done < /etc/passwd

循环与菜单 select

帮助我们生成菜单的作用。

格式:

#语法和for的语法很类似
select 变量 in 列表 ;do
所作的操作
done #PS3用于输出提示信息,效果等同于 read -p
#输入的信息保存到置变量REPLY中
#select 是个无限循环,因此要用 break 命令退出循环 #PS1:是影响我们当前提符的
#PS2:影响多行多定向(<<)提示符的 echo $PS2--> >
#列表就是用来生成菜单的
PS3="请选择功能(1-5):" #PS3:更改菜单的提示信息
select list in 升级软件 备份数据库 回滚软件 删库跑路 ; do
echo $REPLY;
done
#!/bin/bash

PS3="请输入对应选项(1-5): "

select menu in 安装 升级 配置 卸载;do
case $REPLY in
1)
echo "install"
;;
2)
echo "update"
;;
3)
echo "config"
;;
4)
echo "remove"
;;
*)
echo "No this select"
;;
esac
done

函数:function

函数定义格式:

函数名 ()
{
函数体
}

function 函数
{
函数体
}
#写了function关键词,可以省略函数名后面的括号

调用函数:直接通过函数名调用

函数名出现的地方,会被自动替换为函数代码

[root@centos8 ~]#dir() {
> ls -l
> }
[root@centos8 ~]#dir #调用函数
total 4

函数文件:只存放函数的文件

调用函数文件
 . filename 或 source  filename #.和source等价

函数返回值

# 使用echo等命令进行输出

# 函数体中调用命令的输出结果

函数的参数

也是用$n等来实现的。

函数退出状态码

#1. 取决于最后一天命令的状态码

#2. 自定义状态码: retuen num #0:无错误返回,1-255:有错误返回

函数的变量

普通变量:变量名="值" #只能当前进程使用

环境变量:export 变量名="值" #当前进程和子进程都能使用

本地变量: local 变量名="值" #限定了变量在函数中有效,对外部的不影响

范例:

#判断操作系统类型安装软件
#!/bin/bash
os_type
{
if cat /etc/os-release | grep -i -q "ubuntu" ; then # -q:不显示查找的结果,不管找到与否
echo "ubuntu"
elif cat /etc/os-release | grep -E "centos" ;then
echo "centos"
else
echo "no find ubuntu or centos"
fi
}
if [ `os_type` = centos ] ;then #字符串用= != < > 等 (判断符号两边需要加上空格)
yum install httpd -y
elif [ `os_type` = ubuntu ] ; then
apt install httpd -y
else
echo "nofind centos or ubuntu"
#粗略判断ip格式是否合法
##/bin/bash
is_ip ()
{
if [[ "$1" =~ ^([0-9]{1,3}\.){3}[0-9]{1,3}$ ]] ;then
echo "$1 is valid"
else
echo "$1 is not valid"
fi
}
is_ip $1 #给函数传递参数

shell脚本的常用工具

shell脚本信号的捕捉:trap

trap:陷阱,用于捕获信号的

信号的类型:

使用trap -l可以查看

CTRL+C:发送的就是2(SIGINT)信号,程序收到这个信号以后就会退出。

shell脚本捕获信号

通过trap指令来完成:

#格式:
#通过捕获信号来修改信号的原有功能
trap '捕获到信号以后要做的事' 信号 #信号可以是全称、缩写或者前面的数字标号 例如:
trap '' 信号 #忽略信号操作
trap '-' 信号 #恢复信号原有的功能

shell脚本创建临时文件:mktemp

希望创建的临时文件和现有文件不冲突

#创建临时文件的格式
mktemp [选项] [模板] #模板就是临时文件的文件名
#模板格式: filenameXXX,文件前缀+三个随机字符(X(大写)至少要出现三个) 例如;mktemp fileXXX --->file90X
#选项: -d:创建临时目录文件
-p:指定创建的位置

范例

#通过创建临时文件,实现文件垃圾箱(把不用的文件移动到一个目录里面)
DIR=`mktemp -d /tmp/trash-$(date +%F_%H-%M-%S)XXXXXX` #创建一个临时目录文件
mv $* $DIR # $*:表示所有位置的参数
echo $* is move to $DIR

安装复制文件 install

install命令可以改权限、拷贝、创建文件夹等

#编译安装make install:把make命令执行的编译的结果复制到对应的文件中。make install 调用了install命令
#install 的用法(和cp命令很类似) #选项
-m MODE,默认755,修改权限
-o OWNER,指定所有者
-g GROUP,指定所有组
-d DIRNAME 目录,创建目录文件 ##例如
install -m 770 -d /testdir/installdir #创建一个权限为770的文件夹(install -d 相当于 mkdir -p) install -m 700 -o tom -g root /etc/issue /etc/xx.txt #复制一个文件到指定位置,复制的时候更改了权限

expect

把交互式的命令变成非交互式的。(espect通过对指定命令的监控,出现对应关键字的时候执行对应操作)

#expect中相关命令:
spawn 监控某一个命令的执行
expect 捕获特定的字符串
send 捕获到这个特定字符串后,发送对应信息
interact 允许用户交互
exp_continue 匹配多个字符串在执行动作后加此命令

范例

#匹配到hi后,会输出“you said hi”,并换行
expect "hi" {send "You said hi\n"} #捕获屏幕上是否有hi,如果有就自动打印xx

shell脚本的数组

数组:多个变量的集合

bash的数组支持稀疏格式 #索引不连续就叫做稀疏格式

数据的申明

建议先声明再使用。
#声明数组:
declare -a 数据名 #申明为普通数组
declare -A 数组名 #申明为关联数组 #管理数组必须申明后再使用,且关联和普通不能转换 #声明变量
既然所有变量的默认类型是字符串型,那么只要我们把变量声明为整数型就可以进行运算了,
# 定义变量b并赋值为3,具有整型属性。 declare -i x=3

数组赋值

#单个元素赋值 数组名[索引]=值
#例如:weekdays[0]="Sunday" #一次全部赋值:数组名=(v1,v2...vn)
#范例:title=("ceo" "coo" "cto") #使用空格作为分隔符,而不是逗号

使用数组

${数组名[索引]}

#引用数组所有元素
${ARRAY_NAME[*]}或${ARRAY_NAME[@]}

删除数组

#删除数组中的某一个元素
unset 数组名[索引] #删除整个数组
unset 数组名

关联数组

下标不是默认的数字0..n,而是自定义的。

格式
数组名[自定义的下标]=值
#例如: aa[name1]=bob --- echo ${aa[name1]}--->bob

范例

#随机生成十个数字,找出最大的和最小的
#!/bin/bash
max=0
min=0
declare -a arr #定义一个数组用于存储生成的是个随机数
i=0
while [ $i -le 10 ];do
arr[$i]=$RANDOM #给数组赋值
if [ $i -eq 0 ];then #把第一个数作为基准来判断后续数的大小
max=${arr[$i]}
min=${arr[$i]}
else
if [ ${arr[$i]} -gt $max ];then
max=${arr[$i]}
fi
if [ ${arr[$i]} -lt $min ];then
min=${arr[$i]}
fi
fi
let i++ #let工具需要一个变量来接受变量的值
done
echo "max is : $max , min is $min"

shell的字符串处理

shell字符串切片

取出某个字符串中的部分内容。

#数组中显示数组元素的个数:
${#数组名}
#获取字符串的长度
${#var}
例如:str=jkfjafasfa --->10 #跳过字符串左边的几个
${var:num} #num:表示需要跳过几个
#跳过字符串左边的几个,保留多少个
${var:offset:number} #number表示要保留的长度

根据模式取字符串

#删左留右边
${var#*word} #从左边开始找,一直找到word后,就把他们都删除掉(懒惰模式)
${var##*word} #从左边开始找,一直找到最右边的word后,就把他们都删除掉(贪婪模式)
#范例:
file="var/log/messages" ---echo ${file#*/}-->log/messages

字符串的查找和替换

#替换第一次匹配到的
${var/pattern/substr} #替换所有匹配到的
${var//pattern/substr}

删除指定的字符串

#删除掉第一次找到的
${var/pattern} #删除所有找的的
${var/pattern}

Linux shell脚本进阶使用的更多相关文章

  1. Linux shell脚本编程(三)

    Linux shell脚本编程 流程控制: 循环语句:for,while,until while循环: while CONDITION; do 循环体 done 进入条件:当CONDITION为“真” ...

  2. Linux shell脚本编程(二)

    Linux shell脚本编程(二) 练习:求100以内所有偶数之和; 使用至少三种方法实现; 示例1: #!/bin/bash # declare -i sum=0 #声明一个变量求和,初始值为0 ...

  3. Linux shell脚本编程(一)

    Linux shell脚本编程: 守护进程,服务进程:启动?开机时自动启动: 交互式进程:shell应用程序 广义:GUI,CLI GUI: CLI: 词法分析:命令,选项,参数 内建命令: 外部命令 ...

  4. Linux Shell 脚本入门

    linux shell 脚本格式 #!/bin/sh#..... (注释)命令...命令... 使用vi 创建完成之后需设置权限 chmod +x filename.sh 执行命令: ./filena ...

  5. Linux Shell脚本入门--cut命令

    Linux Shell脚本入门--cut命令 cut cut 命令可以从一个文本文件或者文本流中提取文本列. cut语法 [root@www ~]# cut -d'分隔字符' -f fields &l ...

  6. Linux Shell脚本攻略 读书笔记

    Linux Shell脚本攻略 读书笔记 这是一本小书,总共253页,但内容却很丰富,书中的示例小巧而实用,对我这样总是在shell门前徘徊的人来说真是如获至宝:最有价值的当属文本处理,对这块我单独整 ...

  7. 阿里Linux Shell脚本面试25个经典问答

    转载: 阿里Linux Shell脚本面试25个经典问答 Q:1 Shell脚本是什么.它是必需的吗? 答:一个Shell脚本是一个文本文件,包含一个或多个命令.作为系统管理员,我们经常需要使用多个命 ...

  8. Linux Shell脚本教程

    v\:* {behavior:url(#default#VML);} o\:* {behavior:url(#default#VML);} w\:* {behavior:url(#default#VM ...

  9. Linux shell 脚本攻略之统计文件的行数、单词数和字符数

    摘自:<Linux shell 脚本攻略>

随机推荐

  1. python数据处理-matplotlib入门(2)-利用随机函数生成变化图形2

    鉴于上一篇中最后三个问题: 1.上述程序是否能进行优化(比如功能相同的) 2.创建三个3个实例,用了3个语句,能否建一个函数,只输入一个数n,就自动创建n个实例?同时,每个实例的num_times随机 ...

  2. .NET性能优化-你应该为集合类型设置初始大小

    前言 计划开一个新的系列,来讲一讲在工作中经常用到的性能优化手段.思路和如何发现性能瓶颈,后续有时间的话应该会整理一系列的博文出来. 今天要谈的一个性能优化的Tips是一个老生常谈的点,但是也是很多人 ...

  3. keil工程当中实现printf重定向串口打印

    之前是完全不知道printf可以重定向设置 最近才发现还有这等好事,可以让printf直接实现串口打印 在网上找了很多资料,终于实现了我想要的效果 原理:printf是通过调用底部的fputc来实现打 ...

  4. [ubuntu18.04 python3.6] 清华源 CondaHTTPError: HTTP 000 CONNECTION

    问题 嫌官网源安装jupyter notebook太慢,所以尝试修改为清华源,但每次在Solving environment的时候就报错如下: 解决方法,修改conda配置信息: vim ~/.con ...

  5. GO 语言入门(一)

    GO 语言入门(一) 本文写于 2020 年 1 月 18 日 Go 由 Google 工程师 Robert Griesemer,Rob Pike 和 Ken Thompson 设计的一门编程语言,第 ...

  6. spring 事务传播(Propagation)

    propagation 一共有以下几种选项: 1. REQUIRED(默认): 使用当前的事务,如果当前没有事务,则自己新建一个事务,子方法必须运行在一个事务中:如果当前存在事务,则加入这个事务,成为 ...

  7. 组织:IEEE

    电气和电子工程师协会(IEEE,全称是Institute of Electrical and Electronics Engineers)是一个美国的电子技术与信息科学工程师的协会,是世界上最大的非营 ...

  8. 安装vsFTP到CentOS(YUM)

    运行环境 系统版本:CentOS Linux release 7.3.1611 (Core) 软件版本:vsftpd-3.0.2 硬件要求:无 安装过程 1.安装YUM-EPEL存储库 YUM-EPE ...

  9. C#与SQL Server连接时,如何编写连接字符串?

    一.Windows身份验证时: String conStr = "Data Source=数据库服务器地址;Initial Catalog=数据库名称;Integrated Security ...

  10. Navicat 连接 MySQL

    目录 简述 新建连接 常见错误 简述 Navicat 是一套快速.可靠和全面的数据库管理工具,专门用于简化数据库管理和降低管理成本.Navicat 图形界面直观,提供简便的管理方法,设计和操作 MyS ...