shell脚本之函数详细解释及运用
什么是函数
在 Shell 脚本中,函数是组织和复用代码的核心机制。下面从基础到高级全面解析 Shell 函数的用法:
一、基础语法
1. 定义函数的三种方式
标准语法
function 函数名() {命令序列return 返回值 # 可选,默认返回最后命令的退出状态(0-255)
}简化写法1: 没有参数
function 函数名 {
指令
return
}简化写法2: 省略function关键字
函数名() {
指令
return
}
2. 调用函数
函数名 参数1 参数2 ... # 直接写函数名,无需括号
二、函数参数与变量
1. 传递参数
- 在函数内部,通过
$1
,$2
, ... 访问参数 $0
仍表示脚本本身$#
表示参数数量$@
或$*
表示所有参数
示例:计算两数之和
sum() {local result=$(( $1 + $2 )) # local 声明局部变量echo $result
}# 调用函数
result=$(sum 10 20)
echo "结果: $result" # 输出: 结果: 30
2. 局部变量
- 使用
local
关键字声明局部变量(仅在函数内可见) - 若未声明,则默认为全局变量
global_var=10myfunc() {local local_var=20global_var=100 # 修改全局变量echo "函数内: local_var=$local_var, global_var=$global_var"
}myfunc
echo "函数外: global_var=$global_var" # 输出: 100
echo "函数外: local_var=$local_var" # 输出: 空(变量不存在)
三、返回值与退出状态
1. return
语句
- 返回一个整数(0-255)作为函数的退出状态
- 常用于表示成功(0)或失败(非 0)
- 使用
$?
获取返回值
check_file() {if [ -f "$1" ]; thenreturn 0 # 文件存在,返回成功elsereturn 1 # 文件不存在,返回失败fi
}check_file "/etc/passwd"
echo "退出状态: $?" # 输出: 0check_file "/nonexistent"
echo "退出状态: $?" # 输出: 1
2. 通过 echo
返回值
- 函数可通过
echo
输出结果,外部使用$(函数名)
捕获
get_username() {echo "$(id -un)" # 返回当前用户名
}user=$(get_username)
echo "当前用户: $user" # 输出: 当前用户: root
四、函数的作用域
1. 全局变量
- 函数可直接访问和修改全局变量
count=0increment() {((count++)) # 直接修改全局变量
}increment
echo "计数: $count" # 输出: 计数: 1
2. 局部变量
- 使用
local
声明局部变量,避免污染全局环境
total=0calculate() {local num1=$1local num2=$2total=$((num1 + num2)) # 修改全局变量local result=$((num1 * num2)) # 局部变量echo $result
}product=$(calculate 5 3)
echo "乘积: $product, 和: $total" # 输出: 乘积: 15, 和: 8
五、函数的高级用法
1. 递归函数
- 函数可调用自身(需设置终止条件)
# 计算阶乘
factorial() {local n=$1if [ $n -le 1 ]; thenecho 1elselocal prev=$(factorial $((n-1)))echo $((n * prev))fi
}echo "5的阶乘: $(factorial 5)" # 输出: 120
2. 函数库
- 将常用函数保存到独立文件,通过
source
引入
# utils.sh 函数库
#!/bin/bashlog_info() {echo "$(date '+%Y-%m-%d %H:%M:%S') [INFO] $1"
}log_error() {echo "$(date '+%Y-%m-%d %H:%M:%S') [ERROR] $1" >&2
}# 在主脚本中使用
source utils.sh
log_info "开始处理数据"
log_error "文件不存在!"
3. 函数重载(有限支持)
- Shell 不支持真正的函数重载,但可通过参数数量模拟
process() {if [ $# -eq 1 ]; thenecho "处理单个参数: $1"elif [ $# -eq 2 ]; thenecho "处理两个参数: $1 和 $2"elseecho "参数数量错误"return 1fi
}process "apple" # 输出: 处理单个参数: apple
process "apple" "banana" # 输出: 处理两个参数: apple 和 banana
六、常见应用场景
1. 脚本模块化
- 将复杂逻辑拆分为多个函数,提高可读性
#!/bin/bash# 初始化环境
init() {echo "初始化配置..."
}# 检查依赖
check_deps() {command -v curl >/dev/null || { echo "需要安装 curl"; exit 1; }
}# 主函数
main() {initcheck_depsecho "开始主任务..."
}# 执行主函数
main
2. 错误处理
- 封装错误处理逻辑
error_exit() {echo "错误: $1" >&2exit ${2:-1} # 默认退出状态为1
}validate_file() {[ -f "$1" ] || error_exit "文件不存在: $1" 2
}validate_file "/etc/passwd" # 正常
validate_file "/nonexistent" # 报错并退出详细解释:error_exit() 函数
error_exit() {echo "错误: $1" >&2 # 输出错误信息到标准错误(stderr)exit ${2:-1} # 以指定状态码退出(默认1)
}
参数:
$1:错误消息内容
$2:可选的退出状态码(默认 1)
功能:
显示错误信息并终止脚本,适用于各种检查失败的场景。2. validate_file() 函数
validate_file() {[ -f "$1" ] || error_exit "文件不存在: $1" 2 # 检查文件是否存在
}参数:
$1:要验证的文件路径
功能:
使用 [ -f "$1" ] 检查文件是否存在
若不存在,调用 error_exit 输出错误并以状态码 2 退出
exit{2:-1详解}
一、基础语法解析
1. ${parameter:-word} 结构
作用:
如果变量 parameter 存在且不为空,则返回其值;否则返回 word。
示例:
bash
# 变量存在时
name="Alice"
echo ${name:-"默认值"} # 输出: Alice# 变量不存在时
unset age
echo ${age:-18} # 输出: 182. exit 命令
作用:
终止当前脚本或 shell 进程,并返回一个退出状态码(范围 0-255)。
约定:
0 表示成功,非零表示失败。
常见错误码:1(通用错误)、2(误用 shell 命令)、127(命令未找到)等。
二、exit ${2:-1} 的具体含义
1. 参数映射
$2:函数的第二个参数($1 是第一个,依此类推)。
示例:
bash
error_exit "文件不存在" 2 # $1="文件不存在", $2=2
error_exit "权限不足" # $1="权限不足", $2 未提供(为空)2. 执行逻辑
当 $2 存在时:
bash
exit ${2:-1} # 等价于 exit $2示例:
bash
error_exit "文件不存在" 2 # 脚本退出状态为 2当 $2 未提供或为空时:
bash
exit ${2:-1} # 等价于 exit 1示例:
bash
error_exit "权限不足" # 脚本退出状态为 1(默认值)
七、注意事项
-
函数必须先定义后使用
Shell 是逐行解释执行的,调用函数前必须确保已定义。 -
参数传递方式
参数通过位置传递($1
,$2
),没有类型检查,需自行验证。 -
返回值限制
return
只能返回 0-255 的整数,复杂结果需通过echo
输出。 -
命名冲突
函数名不能与内置命令或其他函数重名,建议使用前缀(如myapp_
)。
八、总结
Shell 函数是组织脚本的关键工具,通过合理使用函数可以:
- 提高代码复用性
- 使脚本结构更清晰
- 简化错误处理
- 实现复杂逻辑
掌握函数的定义、参数传递、返回值和作用域,是编写高质量 Shell 脚本的基础
相关文章:
shell脚本之函数详细解释及运用
什么是函数 通俗地讲,所谓函数就是将一组功能相对独立的代码集中起来,形成一个代码块,这个代码可 以完成某个具体的功能。从上面的定义可以看出,Shell中的函数的概念与其他语言的函数的 概念并没有太大的区别。从本质上讲&#…...
校平机的原理、应用及发展趋势
一、校平机的定义与作用 校平机(Leveling Machine)是一种用于矫正金属板材、带材或卷材表面平整度的工业设备。其核心功能是通过机械作用消除材料内部残余应力,修正材料在加工、运输或存储过程中产生的弯曲、波浪形、翘曲等缺陷,…...
NFM算法解析:如何用神经网络增强因子分解机的特征交互能力?
在推荐系统和广告点击率预测等场景中,特征交叉(Feature Interaction)是提升模型效果的关键。传统的因子分解机(FM)通过二阶特征交互取得了显著效果,但其线性建模方式和有限阶数限制了模型的表达能力。今天&…...
Python人工智能算法 模拟退火算法:原理、实现与应用
模拟退火算法:从物理启发到全局优化的深度解析 一、算法起源与物理隐喻 模拟退火算法(Simulated Annealing, SA)起源于20世纪50年代的固体退火理论,其核心思想可追溯至Metropolis等人提出的蒙特卡罗模拟方法。1983年,…...
服务器网络配置 netplan一个网口配置两个ip(双ip、辅助ip、别名IP别名)
文章目录 问答 问 # This is the network config written by subiquity network:ethernets:enp125s0f0:dhcp4: noaddresses: [192.168.90.180/24]gateway4: 192.168.90.1nameservers:addresses:- 172.0.0.207- 172.0.0.208enp125s0f1:dhcp4: trueenp125s0f2:dhcp4: trueenp125…...
FTP与NFS服务详解
一、FTP服务 (一)Linux下FTP客户端管理工具 1. ftp工具 安装命令:yum install ftp -y连接服务器:ftp 服务器IP,输入账号密码登录。常用命令: 命令说明ls查看远程目录文件put上传单个文件到远程服务器get…...
算法中的数学:欧拉函数
1.相关定义 互质:a与b的最大公约数为1 欧拉函数:在1~n中,与n互质的数的个数就是欧拉函数的值 eg: n1时,欧拉函数的值为1,因为1和1是互质的 n2是,值为2,因为1和2都是互质的 积性函数&…...
如果有三个服务实例部署在三台不同的服务器上,这三个服务实例的本地缓存,是存储一模一样的数据?还是各自只存一部分?
✅ 答案是:通常每个服务实例都会独立地缓存它自己访问过的数据,这些数据可能是相同的,也可能是不同的,取决于请求的内容。 📌 举个例子说明 假设你有一个商品详情页的服务,部署了 3 个服务实例(…...
Coze工作流-选择器的用法
上集回顾 上集教程我们学习了什么是变量以及变量类型的用法。即什么时候用什么变量类型 教程简介 本教程将带大家学习工作流的选择和问答模块 工作流类型选择 在Coze中,工作流是智能体的核心逻辑单元。根据任务复杂度,可选择两种模式: 类…...
《AI工程技术栈》:三层结构解析,AI工程如何区别于ML工程与全栈工程
每周跟踪AI热点新闻动向和震撼发展 想要探索生成式人工智能的前沿进展吗?订阅我们的简报,深入解析最新的技术突破、实际应用案例和未来的趋势。与全球数同行一同,从行业内部的深度分析和实用指南中受益。不要错过这个机会,成为AI领…...
uni-app使用大集
1、手动修改页面标题 uni.setNavigationBarTitle({title: 修改标题 }); 2、单选 不止有 radio-group,还有 uni-data-checkbox 数据选择器 <!-- html部分 --> <uni-data-checkbox v-model"sex" :localdata"checkboxList"></u…...
链表day3
链表定义 struct ListNode{int val;ListNode *next; //next是一个指针变量,存储的是地址,是ListNode类型的地址ListNode(int x) : val(x),next(nullptr){} //也就是说ListNode必须接受一个int x,next指针默认为nullptr,值由外部指…...
C++23关联容器的异质擦除重载 (P2077R2)介绍
文章目录 一、基本概念二、原理重载机制类型转换 三、优势提高查找效率提升程序整体性能避免不必要的初始化确保系统实时性 四、应用场景高性能计算大型对象管理实时系统 五、代码示例六、相关图片材料结构与微观图像半导体研究图示与图表科学图表芯片与电路板 一、基本概念 在…...
Flink架构概览,Flink DataStream API 的使用,FlinkCDC的使用
一、Flink与其他组件的协同 Flink 是一个分布式、高性能、始终可用、准确一次(Exactly-Once)语义的流处理引擎,广泛应用于大数据实时处理场景中。它与 Hadoop 生态系统中的组件可以深度集成,形成完整的大数据处理链路。下面我们从…...
AI加速芯片全景图:主流架构和应用场景详解
目录 一、为什么AI芯片如此重要? 二、主流AI芯片架构盘点 三、不同芯片在训练与推理中的部署逻辑 四、真实应用案例解读 五、AI芯片发展趋势预测 AI芯片的选择,是AI系统能否高效运行的关键。今天笔者就从架构角度出发,带你系统了解主流AI加速芯片的种类、优劣对比及实际…...
Ubuntu22.04 系统安装Docker教程
1.更新系统软件包 #确保您的系统软件包是最新的。这有助于避免安装过程中可能遇到的问题 sudo apt update sudo apt upgrade -y 2.安装必要的依赖 sudo apt install apt-transport-https ca-certificates curl software-properties-common -y 3.替换软件源 原来/etc/apt/s…...
更新ubuntu软件源遇到GPG error
BUG背景 执行sudo apt update后遇到类似下列报错: E: The repository https://download.docker.com/linux/ubuntu bionic Release no longer has a Release file. N: Updating from such a repository cant be done securely, and is therefore disabled by defau…...
vue调后台接口
1.1 什么是 axios Axios 是一个基于 promise 的 HTTP 库,可以用来发送网络请求。它可以在浏览器和 node.js 中使用,本质上是对原生 XMLHttpRequest 的封装,符合最新的 ES 规范,支持 Promise API,能够拦截请求和响应&am…...
Ubuntu学习记录
冷知识补充 1.VMware官网安装后,会有两个软件,一个收费(pro)(功能更多,可以一次运行多个虚拟机)(尽管2024年最新版本的也免费了)一个免费(player)。 2.ubuntu打开终端快捷键:ctrlal…...
【音频】如何解析mp3文件
解析和播放MP3文件涉及两个主要步骤:解码(将MP3压缩数据转换为原始PCM音频)和播放(将PCM数据通过音频设备输出)。以下是不同平台和编程语言的实现方法: 一、MP3文件结构基础 MP3文件由多个**帧(Frame)**组成,每帧包含固定时长的音频数据(通常为26ms)。每个帧包含:…...
学习笔记:黑马程序员JavaWeb开发教程(2025.4.9)
12.16 异常处理 定义一个类,加上注解RestControllerAdvice,即定义了一个全局异常处理器 再方法上加上注解ExceptionHandler,通过注解当中的value属性来指定捕获那个类型的异常 完成Filter、interceptor、异常处理代码实操 Filter Filter里…...
【音频】wav文件如何解析编码格式(压缩格式)?
要确定一个WAV文件的编码格式,可以通过以下几种方法实现,包括使用操作系统自带工具、专业音频软件或编程解析文件头信息。以下是详细说明: 一、通过文件属性查看(Windows/macOS) 1. Windows系统 步骤: 右…...
【Django系统】Python+Django携程酒店评论情感分析系统
Python Django携程酒店评论情感分析系统 项目概述 这是一个基于 Django 框架开发的酒店评论情感分析系统。系统使用机器学习技术对酒店评论进行情感分析,帮助酒店管理者了解客户反馈,提升服务质量。 主要功能 评论数据导入:支持导入酒店…...
OpenCv高阶(十六)——Fisherface人脸识别
文章目录 前言一、Fisherface人脸识别原理1. 核心思想:LDA与Fisher准则2. 实现步骤(1) 数据预处理(2) 计算类内散布矩阵 SW对每个类别(每个人)计算均值向量 μi:(3) 计算类间散布矩阵 SB(4) 求解投影矩阵 W(5) 降维与分类 3. Fish…...
数据库与Redis数据一致性解决方案
在写数据时保证 Redis 和数据库数据一致,可采用以下方案,需根据业务场景权衡选择: 1. 先更新数据库,再更新 Redis 步骤: 写入 / 更新数据库数据。删除或更新 Redis 缓存。适用场景:读多写少,对缓存一致性要求不高(短暂不一致可接受)。风险:若第二步失败,导致缓存与…...
Python面试题
Python面试题 Python面试题回答1. Python面向对象的三个特征?多态如何实现和使用2. is 和 的区别?3. GIL了解吗?说说4. 可变类型和不可变类型?5. yield用法?6. 深拷贝和浅拷贝区别?7. Python中的线程8. 生…...
力扣周赛置换环的应用,最少交换次数
置换环的基本概念 置换环是排列组合中的一个概念,用于描述数组元素的重排过程。当我们需要将一个数组转换为另一个数组时,可以把这个转换过程分解为若干个 “环”。每个环代表一组元素的循环交换路径。 举个简单例子 假设原数组 A [3, 2, 1, 4]&…...
差分数组 - 对区间内元素的统一操作
目录 概念 题单 1 拼车 2 将区间分为最少组数 3 字母移位 4 使数组中的所有元素都等于零 5 零数组变换Ⅰ 6 最大化城市的最小电量 概念 差分数组,顾名思义,就是由原数组的相邻元素作差而得到的差值组成的新的数组。 对于原数组 a [ 1 , 3 , 5 …...
线上问题排查
一:CPU飙高问题排查过程 遇到这种问题,首先是登录到服务器,看一下具体情况。 定位进程:top命令,查看CPU占用情况定位线程:top -Hp 1893命令,查看各个线程的CPU使用情况定位代码:pr…...
计及可再生能源不确定性的经济优化调度方法
目前,计及可再生能源不确定性的经济调度方法主要有随机优化、鲁棒优化和区间优化。 随机优化:可再生能源输出被定义为一个已知概率分布的随机变量。 难以同时保证计算精度和效率。 1-场景法 场景生成 基于随机变量概率分布进行采样:蒙特…...
支持向量机(SVM):分类与回归的数学之美
在机器学习的世界里,支持向量机(Support Vector Machine,简称 SVM)是一种极具魅力且应用广泛的算法。它不仅能有效解决分类问题,在回归任务中也有着出色的表现。下面,就让我们深入探索 SVM 如何在分类和回归…...
用户刷题记录日历——签到表功能实现
MySQL实现 在数据库中设计一张签到表,记录用户每次签到的日期及其他相关信息。然后通过时间范围查询得到用户的签到记录。 CREATE TABLE user_sign_in (id BIGINT AUTO_INCREMENT PRIMARY KEY, -- 主键,自动递增userId BIGINT NOT NULL, …...
C语言中的内存函数
目录 1 memcpy()函数的基本信息及功能(1) void * destination(2) const void * source(3) size_t num 1.2 memcpy()函数实战演练1.3 memcpy()函数的模拟实现1.3.1 my_memcpy()函数定义及参数1.3.2 my_memcp…...
本特利内华达330103-00-03-05-02-05毫米接近传感器
描述 3300 XL 8 mm近程传感器系统包括:一个3300 XL 8 mm探头、一根3300 XL延长电缆1和一个3300 XL近程传感器2。 该系统提供的输出电压与探针尖端和观察到的导电表面之间的距离成正比,可以测量静态(位置)和动态(振动)值。该系统的主要应用是流体膜轴承机器的振动和位…...
啤酒游戏与系统思考
今天,与上海地产集团的伙伴们一同体验经典的系统思考沙盘模拟——“啤酒游戏”。虽然大家身处房地产行业,但也会惊讶地发现,啤酒游戏的核心理念对任何行业都适用,尤其是站在全局的角度,做出精准决策。 每次进行啤酒游戏…...
id分页遍历数据漏行问题
令入参id为0 while(true){ select * from table where id>#{id} order by id asc limit 100; 取结果集中最大id作为下次查询的入参 其他操作 } 这个算法一般没问题,但在主从数据系统中,主库写,查询从库遍历数据时,出现了…...
【Vue3】Vue3工程的创建 及 开发者工具的安装
目录 一、创建Vue3工程的方式 方法一 方法二 二、区分Vue3 和 Vue2的构建 观察main.js vue3不向下兼容,也就是说Vue3不支持Vue2的写法! JavaScript 的模块导入有两种常见写法: 三、安装Vue3的开发者工具 总结不易~本章节对我有很大的…...
docker exec -it abc bash
当然可以!让我们详细解析一下 docker exec -it abc bash 这个命令的各个部分及其作用。 命令概述 docker exec -it abc bash这个命令用于在已经运行的 Docker 容器 abc 中启动一个新的交互式终端会话。具体来说,它会执行容器内的 bash 命令,…...
基于AI大语言模型的历史文献分析在气候与灾害重建中的技术-以海南岛千年台风序列重建为例
随着人工智能技术的飞速发展,大语言模型如GPT、BERT等在自然语言处理领域取得了显著成果。这些模型不仅提高了文本数据的处理和理解效率,还为历史灾害研究提供了全新的视角和方法。本文将深入探讨基于AI大语言模型的历史文献分析在气候与灾害重建领域中的…...
【最细】自动化测试-解决日志问题,一文贯通...
目录:导读 前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结(尾部小惊喜) 前言 持续集成的自动化…...
PromptIDE:一款强大的AI提示词优化工具
今天向大家推荐一款专业且免费的AI提示词工具——PromptIDE。作为AI领域从业者,我们都深知提示词质量对模型输出的重要性,而这款工具正是为解决这一痛点而生。 核心功能解析 1、提示词优化 简单输入你的需求描述,点击AI生成提示词…...
f-string 高效的字符串格式化
f-string,称为格式化字符串常量(formatted string literals),是Python3.6新引入的一种字符串格式化方法,该方法源于PEP 498 – Literal String Interpolation,主要目的是使格式化字符串的操作更加简便。 p…...
Powershell及命令行文章合集(不定期更新)
一、Powershell: 1.Powershell中常用命令和常用属性:https://blog.csdn.net/humors221/article/details/147978718 2.Powershell数值应用讲解:https://blog.csdn.net/humors221/article/details/142897029 3.PowerShell 抓取网络日志:https://blog.csdn.net/humors221/artic…...
leetcode hot100刷题日记——8.合并区间
class Solution { public:vector<vector<int>> merge(vector<vector<int>>& intervals) {if(intervals.empty()){//复习empty函数啊,日记1有的return {};}// 按照区间的起始位置进行排序sort(intervals.begin(), intervals.end());vect…...
基于moonshot模型的Dify大语言模型应用开发核心场景
基于moonshot模型的Dify大语言模型应用开发核心场景学习总结 一、Dify环境部署 1.Docker环境部署 这里使用vagrant部署,下载vagrant之后,vagrant up登陆,vagrant ssh,在vagrant 中使用 vagrant centos/7 init 快速创建虚拟机 安装…...
系统设计应优先考虑数据流还是控制流?为什么优先考虑数据流?数据流为主、控制流为辅的架构原则是什么?控制流优先会导致哪些问题?
在当代软件工程的复杂演化中,每个现代系统,不论是处理金融交易的平台、智能家居系统,还是自动驾驶系统,都面临同一个核心问题:设计者该以“数据流”为主导,还是以“控制流”为主导? 在系统设计过程中,工程师所面对的核心问题不仅是代码的堆叠与组织,更是信息流动模式…...
Redis Cluster动态扩容:架构原理与核心机制解析
一、哈希槽的数学本质与拓扑重构 核心图示:哈希槽分配演变 #mermaid-svg-YmcBfipoPA8LvxYF {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-YmcBfipoPA8LvxYF .error-icon{fill:#552222;}#mermaid-svg-Y…...
高考之外,英国国际大一:开启本科留学新征程
在竞争激烈的教育赛道上,高考一直被视为通向高等学府的重要关卡。然而,当千军万马挤在国内升学这座独木桥时,越来越多有远见的学生和家长将目光投向海外,英国本科留学凭借其灵活的录取机制和多元的升学路径,成为众多学…...
UML 图的细分类别及其应用
统一建模语言(UML,Unified Modeling Language)是一种用于软件系统建模的标准化语言,广泛应用于软件工程领域。UML 图分为多种类别,每种图都有其特定的用途和特点。本文将详细介绍 UML 图的细分类别,包括 类…...
Android10如何设置ro.debuggable=1?
说明:仅供学习使用,请勿用于非法用途,若有侵权,请联系博主删除 作者:zhu6201976 目录 一、背景 二、如何解决? 三、操作步骤 一、背景 Android 10 开始的限制:ro.debuggable 是只读属性 从 …...