Linux-流程控制语法-函数-属组
1、流程控制
1.1 条件选择
1.1.1选择执行if语句
格式:
if COMMANDS; then COMMANDS; [ elif COMMANDS; then COMMANDS; ] ... [ else COMMANDS; ] fi
单分支
if 判断条件;then 条件为真的分支代码 fi
双分支
if 判断条件; then 条件为真的分支代码 else 条件为假的分支代码 fi
多分枝
if 判断条件1; then 条件1为真的分支代码 elif 判断条件2; then 条件2为真的分支代码 elif 判断条件3; then 条件3为真的分支代码 ... else 以上条件都为假的分支代码 fi
- 多个条件时,逐个条件进行判断,第一次遇为 ‘ 真 ’ 条件时,执行其分支,而后结束整个if语句
- if语句可嵌套
1.1.2条件判断case语句
格式:
case WORD in [PATTERN [| PATTERN]...) COMMANDS ;;]... esac
case 变量引用 in PAT1) 分支1 ;; PAT2) 分支2 ;; ... *) 默认分支 ;; esac
case支持glob风格的通配符
* 任意长度任意字符 ? 任意单个字符 [] 指定范围内的任意单个字符 | 或者,如: a|b
1.2循环
1.2.1 for循环
格式1
for NAME [in WORDS ... ] ; do COMMANDS; done #方式1 for 变量名 in 列表;do 循环体 done #方式2 for 变量名 in 列表 do 循环体 done
- 依次将列表中的元素赋值给“变量名”; 每次赋值后即执行一次循环体; 直到列表中的元素耗尽,循环 结束
- 如果省略 [in WORDS ... ] ,此时使用位置参数变量 in "$@"
格式2
双小括号方法,即((…))格式,也可以用于算术运算,双小括号方法也可以使bash Shell实现C语言风格 的变量操作 I=10;((I++))
for ((: for (( exp1; exp2; exp3 )); do COMMANDS; done for ((控制变量初始化;条件判断表达式;控制变量的修正表达式)) do 循环体 done
- 控制变量初始化:仅在运行到循环代码段时执行一次
- 控制变量的修正表达式:每轮循环结束会先进行控制变量修正运算,而后再做条件判断
1.2.2 while循环
格式:
while COMMANDS; do COMMANDS; done
while CONDITION; do 循环体 done
- CONDITION:循环控制条件;进入循环之前,先做一次判断;每一次循环之后会再次做判断;条件为 “true”,则执行一次循环;直到条件测试状态为“false”终止循环,因此:CONDTION一般应该有循环控 制变量;而此变量的值会在循环体不断地被修正
- 进入条件:CONDITION为 true
- 退出条件:CONDITION为 false
无限循环
while true; do 循环体 done while : ; do 循环体 done
1.2.3 until循环
格式
until COMMANDS; do COMMANDS; done
until CONDITION; do 循环体 done
- 进入条件: CONDITION 为false
- 退出条件: CONDITION 为true
1.2.4 continue循环控制语句
continue [N]:提前结束第N层的本轮循环,而直接进入下一轮判断;最内层为第1层
格式:
while CONDITION1; do CMD1 ... if CONDITION2; then continue fi CMDn ... done
1.2.5 break循环控制语句
break [N]:提前结束第N层整个循环,最内层为第1层
格式:
while CONDITION1; do CMD1 ... if CONDITION2; then break fi CMDn ... done
1.2.6 shift命令 循环控制
shift [n] 用于将参量列表 list 左移指定次数,缺省为左移一次。
参量列表 list 一旦被移动,最左端的那个参数就从列表中删除。while 循环遍历位置参量列表时,常用 到 shift
1.2.7 while特殊用法 while read
while 循环的特殊用法,遍历文件或文本的每一行
格式:
while read line; do 循环体 done < /PATH/FROM/SOMEFILE
- 依次读取/PATH/FROM/SOMEFILE文件中的每一行,且将行赋值给变量line
1 2 3 4 5 6 7 8 9 10 | 范例:查看/sbin/nologin的shell类型的用户名和UID< br > cat while_read_passwd.sh #!/bin/bash # while read line ;do if [[ "$line" =~ /sbin/nologin$ ]] ;then echo $line | cut -d: -f1,3 fi done < /etc/passwd |
1.2.8 select 循环于菜单
格式:
select NAME [in WORDS ... ;] do COMMANDS; done select variable in list ;do 循环体命令 done
- select 循环主要用于创建菜单,按数字顺序排列的菜单项显示在标准错误上,并显示 PS3 提示 符,等待用户输入
- 用户输入菜单列表中的某个数字,执行相应的命令
- 用户输入被保存在内置变量 REPLY 中
- select 是个无限循环,因此要用 break 命令退出循环,或用 exit 命令终止脚本。也可以按 ctrl+c 退出循环
- select 经常和 case 联合使用
- 与 for 循环类似,可以省略 in list,此时使用位置参量
2、函数function
函数介绍
函数function是由若干条shell命令组成的语句块,实现代码重用和模块化编程
它与shell程序形式上是相似的,不同的是它不是一个单独的进程,不能独立运行,而是shell程序的一 部分
函数和shell程序区别
- Shell程序在子Shell中运行
- 函数在当前Shell中运行。因此在当前Shell中,函数可对shell中变量进行修改
管理函数
函数由两部分组成:函数名和函数体
帮助参看:help function
2.1.1 定义函数
#语法一:
func_name (){
...函数体...
}
#语法二:
function func_name {
...函数体...
}
#语法三:
function func_name () {
...函数体...
}
2.1.2查看函数
#查看当前已定义的函数名 declare -F #查看当前已定义的函数定义 declare -f #查看指定当前已定义的函数名 declare -f func_name #查看当前已定义的函数名定义 declare -F func_name
2.1.3删除函数
unset func_name
2.2函数的调用方式
- 可在交互式环境下定义函数
- 可将函数放在脚本文件中作为它的一部分
- 可放在只包含函数的单独文件中
调用:函数只有被调用才会执行,通过给定函数名调用函数,函数名出现的地方,会被自动替换为函数代码
函数的生命周期:被调用时创建,返回时终止
函数在使用前必须定义,因此应将函数定义放在脚本开始部分,直至shell首次发现它后才能使用,调用 函数仅使用其函数名即可
2.2.1使用函数文件
可以将经常使用的函数存入一个单独的函数文件,然后将函数文件载入shell,再进行调用函数函数文件名可任意选取,但最好与相关任务有某种联系,例如:functions
一旦函数文件载入shell,就可以在命令行或脚本中调用函数。可以使用delcare -f 或set 命令查看所有 定义的函数,其输出列表包括已经载入shell的所有函数
若要改动函数,首先用unset命令从shell中删除函数。改动完毕后,再重新载入此文件
实现函数文件的过程:
- 创建函数文件,只存放函数的定义
- 在shell脚本或交互式shell中调用函数文件,格式如下:
. filename
source filename
2.3函数返回值
函数的执行结果返回值:
- 使用echo等命令进行输出
- 函数体中调用命令的输出结果
函数的退出状态码:
- 默认取决于函数中执行的最后一条命令的退出状态码
- 自定义退出状态码,其格式为:
return 从函数中返回,用最后状态命令决定返回值
return 0 无错误返回
return 1-255 有错误返回
2.4环境函数
类拟于环境变量,也可以定义环境函数,使子进程也可使用父进程定义的函数
定义环境函数:
export -f function_name
declare -xf function_name
查看环境函数:
export -f
declare -xf
2.5函数参数
函数可以接受参数:
- 传递参数给函数:在函数名后面以空白分隔给定参数列表即可,如:testfunc arg1 arg2 ...
- 在函数体中当中,可使用$1, $2, ...调用这些参数;还可以使用$@, $*, $#等特殊变量
2.6函数变量
变量作用域:
- 普通变量:只在当前shell进程有效,为执行脚本会启动专用子shell进程;因此,本地变量的作用 范围是当前shell脚本程序文件,包括脚本中的函数
- 环境变量:当前shell和子shell有效
- 本地变量:函数的生命周期;函数结束时变量被自动销毁
注意:
- 如果函数中定义了普通变量,且名称和局部变量相同,则使用本地变量
- 由于普通变量和局部变量会冲突,建议在函数中只使用本地变量
在函数中定义本地变量的方法
local NAME=VALUE
2.7函数递归
函数递归:函数直接或间接调用自身,注意递归层数,可能会陷入死循环
递归特点:
- 函数内部自已调用自已
- 必须有结束函数的出口语句,防止死循环
范例: 无出口的递归函数调用
[root@centos8 ~]#func () { echo $i;echo "run func";let i++; func; } [root@centos8 ~]#func
3、属组 array
属组介绍
变量:存储单个元素的内存空间
数组:存储多个元素的连续的内存空间,相当于多个变量的集合
数组名和索引
- 索引的编号从0开始,属于数值索引
- 索引可支持使用自定义的格式,而不仅是数值格式,即为关联索引,bash 4.0版本之后开始支持
- bash的数组支持稀疏格式(索引不连续)
3.1声明数组
#普通数组可以不事先声明,直接使用 declare -a ARRAY_NAME #关联数组必须先声明,再使用 declare -A ARRAY_NAME
注意:两者不可相互转换
3.2数组赋值
数组元素的赋值
- 一次只赋值一个元素
ARRAY_NAME[INDEX]=VALUE
示例:
weekdays[0]="Sunday" weekdays[4]="Thursday"
- 一次赋值全部元素
ARRAY_NAME=("VAL1" "VAL2" "VAL3" ...)
title=("ceo" "coo" "cto") num=({0..10}) alpha=({a..g}) file=( *.sh )
- 只赋值特定元素
ARRAY_NAME=([0]="VAL1" [3]="VAL2" ...)
- 交互式数组值对赋值
read -a ARRAY
3.3显示所有数组
显示所有数组
declare -a
3.4引用数组
引用特定的数组元素
${ARRAY_NAME[INDEX]}
#如果省略[INDEX]表示引用下标为0的元素
引用数组所有元素
${ARRAY_NAME[*]}
${ARRAY_NAME[@]}
数组的长度,即数组中元素的个数
${#ARRAY_NAME[*]}
${#ARRAY_NAME[@]}
3.5删除数组
删除数组中的某元素,会导致稀疏格式
unset ARRAY[INDEX]
删除整个数组
unset ARRAY
3.6数组数据处理
数组切片:
${ARRAY[@]:offset:number}
offset #要跳过的元素个数
number #要取出的元素个数
#取偏移量之后的所有元素
{ARRAY[@]:offset}
[root@centos8 ~]#num=({0..10}) [root@centos8 ~]#echo ${num[*]:2:3} 2 3 4 [root@centos8 ~]#echo ${num[*]:6} 6 7 8 9 10
向数组中追加元素:
ARRAY[${#ARRAY[*]}]=value
ARRAY[${#ARRAY[@]}]=value
[root@centos8 ~]#num[${#num[@]}]=11 [root@centos8 ~]#echo ${#num[@]} 12 [root@centos8 ~]#echo ${num[@]} 0 1 2 3 4 5 6 7 8 9 10 11
3.7关联数组
declare -A ARRAY_NAME ARRAY_NAME=([idx_name1]='val1' [idx_name2]='val2‘...)
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· DeepSeek 开源周回顾「GitHub 热点速览」
· 物流快递公司核心技术能力-地址解析分单基础技术分享
· .NET 10首个预览版发布:重大改进与新特性概览!
· AI与.NET技术实操系列(二):开始使用ML.NET
· .NET10 - 预览版1新功能体验(一)