【Linux】函数
一、函数
1、创建函数
如果定义了同名函数,则新定义的函数就会覆盖原先的定义的函数,而且在运行时不会报错。
创建函数的语法:
方法1:使用关键字function
function name {
commands
}
shell脚本中的函数名不能重复
方法2:
name() {
commands
}
函数名后的空括号name()表明正在定义的是一个函数。
2、使用函数
不能在函数定义之前调用函数,否则会报错。
#!/bin/bashfunction test {echo "测试123456"
}count=1
while [ $count -le 5 ]
dotestcount=$[ $count + 1 ]
done
echo "循环结束!"
test
echo "脚本运行结束!"
exit
3、函数返回值
使用退出状态码
默认情况下,函数的退出状态码是函数中最后一个命令返回的退出状态码。函数执行结束后,可以使用标准变量 $? 来确定函数的退出状态码。
#!/bin/bashfunc1() {echo "尝试展示不存在的文件。"ls -l badfile
}echo "测试函数:"
func1
echo "退出状态码是:$?"
如下所示,退出状态码是2,因为函数中的最后一个命令执行失败了,但 echo 命令却执行成功了。 所以,使用退出状态码的方法是一种危险的方法。
使用return命令
bash shell会使用return命令以特定的退出状态码退出函数;return命令允许指定一个整数值作为函数的退出状态码。
#!/bin/bashfunc2() {read -p "请输入一个数值:" valueecho "将输入的数值翻倍。"return $[ $value * 2 ]
}func2
echo "新的值是 $?"
exit
当用此方法获取返回值时,需注意两点:
函数执行一结束就立刻读取返回值:如果在用$?变量提取函数返回值之前执行了其它命令,那么函数返回值就会丢失。$?变量保存的是最后执行的那个命令的退出状态码。
退出状态码必须在0~255范围内:因为退出状态码必须小于256,因此函数结果也必须是一个小于256的整数值,如果大于256就会产生错误的值。
使用函数输出
#!/bin/bashfunc3() {read -p "请输入一个数值:" valueecho "将输入的数值翻倍。"echo $[ $value * 2 ] # 注意不是return
}result=$(func3)
echo "新的值是$result"
exit
4、向函数传递参数
函数可以使用标准的位置变量来表示命令行中传给函数的任何参数,如函数名保存在$0变量中,函数参数依次保存在$1、$2等变量中。也可以使用特殊变量$#来确定传给函数的参数数量。
在shell脚本中调用函数时,必须将参数和函数名放在同一行。
用位置变量获取参数值
#!/bin/bashfunction func4 {if [ $# -eq 0 ] || [ $# -gt 2 ]thenecho -1elif [ $# -eq 1 ]thenecho $[ $1 + $1 ]elseecho $[ $1 + $2 ]fi
}echo -n "传递两个参数:"
value=$(func4 10 15)
echo $valueecho -n "只传递一个参数:"
value=$(func4 20)
echo $valueecho -n "不传递参数:"
value=$(func4)
echo $valueecho -n "传递3个参数:"
value=$(func4 10 20 36)
echo $value
要想在函数中使用脚本的命令行参数,必须在调用函数时手动将其传入。
func5() {echo $[ $1 * $2 ]
}if [ $# -eq 2 ]
thenvalue=$(func5 $1 $2)echo "结果是$value."
elseecho "错误。"
5、函数中的变量
全局变量
全局变量在shell脚本内任何地方都有效的变量。
在默认情况下,在脚本中定义的任何变量都是全局变量。在函数外定义的变量可在函数内正常访问。
#!/bin/bashfunction func1 {value=$[ $value *2 ]
}read -p "请输入一个值:" value
func1
echo "新的值是 $value"
局部变量
任何在函数内部使用的变量都可以被声明为局部变量,只需在变量声明之前加上关键字 local 即可:local var
也可在变量赋值语句中使用 local 关键字:local var=$[ $value + 7 ]
local关键字保证了变量仅在函数中有效,如果函数外有同名变量,那么shell会保持这两个变量的值互不干扰,意味着可以轻松地将函数变量和脚本变量分离开。
#!/bin/bashfunction func2 {local temp=$[ $value + 10 ]result=$[ $temp * 2 ]
}temp=5
value=8func2
echo "结果是$result"
if [ $temp -gt $value ]
thenecho "temp更大"
elseecho "value更大"
fi
6、向函数传递数组
将数组变量当做单个参数传递的话,是不会起作用的。如果想要将数组变量作为函数参数进行传递,那么函数只会提取数组变量的第一个元素。
#!/bin/bashfunction func3 {echo "参数是:$@"thisarray=$1echo "接收到的数组是 ${thisarray[*]}"
}myarray=(1 2 3 4 5 6)
echo "原始数组是:${myarray[*]}"
func3 $myarray
传递数组变量时必须先将数组变量拆解成多个数组元素,然后将这些数组元素作为函数参数传递,最后在函数内部,将所有的参数重新组合成一个新的数组变量。
#!/bin/bashfunction func5 {local newarraynewarray=(`echo "$@"`)echo "新数组是:${newarray[*]}"
}myarray=(1 2 3 4 5 6)
echo "原始数组是:${myarray[*]}"
func5 ${myarray[*]}
#!/bin/bashfunction addarray {local sum=0local newarraynewarray=(`echo "$@"`)for value in ${newarray[*]}dosum=$[ $sum + $value ]doneecho $sum
}myarray=(1 2 3 4 5 6)
echo "原始数组是:${myarray[*]}"
arg1=$(echo ${myarray[*]})
result=$(addarray $arg1)
echo "结果是 $result"
7、从函数返回数组
函数先用echo语句按正确顺序输出数组的各个元素,然后脚本再将数组元素重组成一个新的数组变量。
#!/bin/bashfunction arraydb {local origarraylocal newarraylocal elementlocal iorigarray=($(echo "$@")) newarray=($(echo "$@"))# "$@"会将每个参数作为独立的字符串保留element=$[ $# - 1 ] # $#表示参数数量for (( i = 0; i <= $element; i++ )){newarray[$i]=$[ ${origarray[$i]} * 2 ]}echo ${newarray[*]}
}myarray=(1 2 3 4 5 6)
echo "原始数组是:${myarray[*]}"
arg1=$(echo ${myarray[*]})
result=($(arraydb $arg1))
echo "新数组是:${result[*]}"
8、函数递归
局部函数变量的一个特性是自成体系(self-containment),除了获取函数参数,自成体系函数不需要使用任何外部资源。这个特性使函数可以递归地调用,即函数通过调用自己得到结果。
递归函数通常有一个最终可以迭代到的基准值。
递归算法的经典例子就是计算阶乘。
#!/bin/bash
function factorial {if [ $1 -eq 1 ]thenecho 1elselocal temp=$[ $1 - 1 ]local result=$(factorial $temp)echo $[ $result * $1 ]fi
}read -p "请输入一个数值:" value
result=$(factorial $value)
echo "$value的阶乘是:$result"
exit
9、函数库
bash shell允许创建函数库文件,然后在多个shell脚本中引用此库文件。
可以通过 source 命令(也叫点号操作符)在shell脚本中运行库文件,它会在当前shell的上下文中执行命令,而不是创建新的shell并在其中执行命令。
格式:. ./绝对路径/函数库文件名
#!/bin/bash#创建一个函数库文件function addem {echo $[ $1 + $2 ]
}function multem {echo $[ $1 * $2 ]
}function divem {if [ $2 -ne 0 ]thenecho $[ $1 / $2 ]elseecho -1fi
}
在其它shell脚本中运行上面的函数库文件。
#!/bin/bash#用source命令允许库文件
. ./myfuncs.sh # 该函数库文件与此脚本在同一目录下value1=60
value2=20
result1=$(addem $value1 $value2)
echo "两数相加之和是:$result1"
result2=$(multem $value1 $value2)
echo "两数相乘之积是:$result2"
result3=$(divem $value1 $value2)
echo "两数相除之商是:$result3"
二、在命令行中使用函数
一旦在shell中定义了函数,就可以在整个系统的任意目录中使用,而不用担心该函数是否位于PATH环境变量中。
1、在命令行中创建函数
因为shell会解释用户输入的命令,所以可以在命令行中直接定义一个函数。缺点就是,在退出shell时,函数也会跟着消失
单行方式
使用单行方式定义函数时,需要在每个命令后面加分号,这样shell才能知道哪里是命令的起止。
多行方式
使用多行方式定义函数时,bash shell会使用次提示符来提示输入更多命令,此方法不需要再每条命令后面加分号,只需按下回车键即可。
当输入函数末尾的花括号 } 时,shell就能知道用户已经完成函数的定义了。
2、在.bashrc文件中定义函数
可以将函数定义在每次新shell启动时都会重新读取该函数的地方:.bashrc 文件。
不管是交互式shell还是从现有shell启动新的shell,bash shell在每次启动时都会在用户主目录中查找.bashrc文件。
直接定义函数
可以直接在用户主目录的.bashrc文件中定义函数。将函数直接放在末尾即可,如下所示。
此函数在下次启动新的bash shell时生效,然后就能在系统中的任意地方使用此函数。
源引函数文件
只要是在shell脚本中,就可以用source命令(点号操作符)将库文件中的函数添加到.bashrc脚本中,如下所示:
确保库文件的路径正确,下次重新启动bash shell时,就可以使用库中的所有函数。
也可以直接在shell脚本中使用,因为shell会将定义好的函数传给子shell进程,如此一来,这些函数就能够自动用于该shell会话中的任何shell脚本了。
#!/bin/bashvalue1=20
value2=5result1=$(addem $value1 $value2)
echo "两数之和是:$result1."
result2=$(multem $value1 $value2)
echo "两数之积是:$result2."
result3=$(divem $value1 $value2)
echo "两数之商是:$result3."
如下所示,执行脚本时报错,但能在交互式命令行执行。
相关文章:

【Linux】函数
一、函数 1、创建函数 如果定义了同名函数,则新定义的函数就会覆盖原先的定义的函数,而且在运行时不会报错。 创建函数的语法: 方法1:使用关键字function function name { commands } shell脚本中的函数名不能重复 方法2&#x…...
Maven中管理SNAPSHOT版本含义及作用
在开发过程中突然产生了一个疑问:IDEA中 maven deploy的依赖包的版本号,比如 1.0.0-SNAPSHOT是在哪配置的?在远程仓库中的版本和这个有关系吗 ? 在 Maven 中,-SNAPSHOT 后缀是用于标识项目版本为快照(Snapshot…...

win10 VS2019上libtorch库配置过程
win10 VS2019上libtorch库配置过程 0 引言1 获取libtorch2 在VS上配置使用libtorch库3 结语 0 引言 💻💻AI一下💻💻 libtorch库是一个用于深度学习的C库,是PyTorch的官方C前端。它提供了用于构建和训练深度学习模…...

【计算机网络】课程 实验二 交换机基本配置和VLAN 间路由实现
实验二 交换机基本配置和VLAN 间路由实现 一、实验目的 1.了解交换机的管理方式。 2.掌握通过Console接口对交换机进行配置的方法。 3.掌握交换机命令行各种模式的区别,能够使用各种帮助信息以及命令进行基本的配置。 4&…...
Oracle Dataguard(主库为单节点)配置详解(4):将主库复制到备库并启动同步
Oracle Dataguard(主库为单节点)配置详解(4):将主库复制到备库并启动同步 目录 Oracle Dataguard(主库为单节点)配置详解(4):将主库复制到备库并启动同步一、…...

OpenCL(贰):浅析CL内核程序接口函数
目录 1.前言 2.获取平台信息 1.cl_int类型 2.cl_platform_id类型 3.clGetPlatformIDs():查询系统OpenCL平台数量或获取具体的平台信息 4.clGetPlatformInfo():查询指定OpenCL平台的信息,例如平台名称、供应商、版本等 3.设置OpenCL上下文…...
Leetcode 3407. Substring Matching Pattern
Leetcode 3407. Substring Matching Pattern 1. 解题思路2. 代码实现 题目链接:3407. Substring Matching Pattern 1. 解题思路 这一题是一道leetcode easy的题目,照说应该没啥的,不过实际我做的时候在这里卡了一下,所以还是拿…...

学英语学压测:02jmeter组件-测试计划和线程组ramp-up参数的作用
📢📢📢:先看关键单词,再看英文,最后看中文总结,再回头看一遍英文原文,效果更佳!! 关键词 Functional Testing功能测试[ˈfʌŋkʃənəl ˈtɛstɪŋ]Sample样…...

Vue笔记-001-声明式渲染
https://cn.vuejs.org/tutorial/#step-2https://cn.vuejs.org/tutorial/#step-2 Vue 单文件组件 (Single-File Component,缩写为 SFC) 单文件组件是一种可复用的代码组织形式,它将从属于同一个组件的 HTML、CSS 和 JavaScript 封装在使用 .vue 后缀的文件…...
26考研资料分享 百度网盘
26考研资料分享考研资料合集 百度网盘(仅供参考学习) 基础班: 通过网盘分享的文件:2026【考研英语】等3个文件 链接: https://pan.baidu.com/s/1Q6rvKop3sWiL9zBHs87kAQ?pwd5qnn 提取码: 5qnn --来自百度网盘超级会员v3的分享…...
.NET 8 + Ocelot + Consul 实现代理网关、服务发现
.NET 8 Ocelot Consul 实现代理网关、服务发现 本文环境:.NET 8 Ocelot 23.4.2 Consul 1.7.14.6 1 实现网关 分别创建3个WebApi工程:OcelotGw、TestGwAService、TestGwBService;在OcelotGw工程中安装Ocelot包:Install-Packag…...
使用 Nginx 轻松处理跨域请求(CORS)
使用 Nginx 轻松处理跨域请求(CORS) 在现代 Web 开发中,跨域资源共享(CORS)是一种重要的机制,用于解决浏览器的同源策略限制。CORS 允许服务器声明哪些来源可以访问其资源,从而确保安全性与可用…...

【LeetCode Hot100 二分查找】搜索插入位置、搜索二维矩阵、搜索旋转排序数组、寻找两个正序数组的中位数
二分查找 搜索插入位置搜索二维矩阵在排序数组中查找元素的第一个和最后一个位置寻找旋转排序数组中的最小值搜索旋转排序数组寻找两个正序数组的中位数(hard) 搜索插入位置 给定一个排序数组和一个目标值,在数组中找到目标值,并…...
使用MediaPipe Face Mesh 面部动作检测
一、技术选型 OpenCV(Open Source Computer Vision Library) 用于视频流捕捉、图像预处理和基本图像处理操作。 MediaPipe 提供高效的人脸检测与关键点提取功能(Face Mesh)。 Python 作为后端开发语言,整合上述库进行…...
【Vue】<script setup>和 <script>区别是什么?在使用时的写法区别?
<script setup> 是 Vue 3 引入的一种新的脚本语法,它提供了一种更简洁和声明式的方式来编写组件逻辑。它是为了解决传统 <script> 标签在 Vue 单文件组件(SFC)中的一些局限性而设计的。 <script setup> 与 <script>…...

微服务框架,Http异步编程中,如何保证数据的最终一致性
一、背景 在微服务框架下,跨服务之间的调用,当遇到操作耗时或者量大的情况,我们一般会采用异步编程实现。 本文出现的问题是:异步回调过来时,却未查询到数据库中的任务,导致未能正常处理回调。 下面是当…...

vue3-dom-diff算法
vue3diff算法 什么是vue3diff算法 Vue3中的diff算法是一种用于比较虚拟DOM树之间差异的算法,其目的是为了高效地更新真实DOM,减少不必要的重渲染 主要过程 整个过程主要分为以下五步 前置预处理后置预处理仅处理新增仅处理后置处理包含新增、卸载、…...

年会抽奖Html
在这里插入图片描述 <!-- <video id"backgroundMusic" src"file:///D:/background.mp3" loop autoplay></video> --> <divstyle"width: 290px; height: 580px; margin-left: 20px; margin-top: 20px; background: url(D:/nianhu…...
ubuntu16 重启之后lvm信息丢失故障恢复
一、背景 1、问题背景 业务有一台物理开发服务器,文件系统有损坏;由于重启时没有检查,导致重启卡住。后面通过断电重新启动之后,无法进入系统;进入救援模式,注释数据盘挂载。重启之后进入系统,…...
【华为OD-E卷 - 热点网站统计 100分(python、java、c++、js、c)】
【华为OD-E卷 - 热点网站统计 100分(python、java、c、js、c)】 题目 企业路由器的统计页面,有一个功能需要动态统计公司访问最多的网页URL top N。请设计一个算法,可以高效动态统计Top N的页面 输入描述 每一行都是一个URL或…...
ubuntu搭建nfs服务centos挂载访问
在Ubuntu上设置NFS服务器 在Ubuntu上,你可以使用apt包管理器来安装NFS服务器。打开终端并运行: sudo apt update sudo apt install nfs-kernel-server创建共享目录 创建一个目录用于共享,例如/shared: sudo mkdir /shared sud…...

遍历 Map 类型集合的方法汇总
1 方法一 先用方法 keySet() 获取集合中的所有键。再通过 gey(key) 方法用对应键获取值 import java.util.HashMap; import java.util.Set;public class Test {public static void main(String[] args) {HashMap hashMap new HashMap();hashMap.put("语文",99);has…...
IGP(Interior Gateway Protocol,内部网关协议)
IGP(Interior Gateway Protocol,内部网关协议) 是一种用于在一个自治系统(AS)内部传递路由信息的路由协议,主要用于在一个组织或机构的内部网络中决定数据包的最佳路径。与用于自治系统之间通信的 EGP&…...

为什么需要建设工程项目管理?工程项目管理有哪些亮点功能?
在建筑行业,项目管理的重要性不言而喻。随着工程规模的扩大、技术复杂度的提升,传统的管理模式已经难以满足现代工程的需求。过去,许多企业依赖手工记录、口头沟通和分散的信息管理,导致效率低下、成本失控、风险频发。例如&#…...

汽车生产虚拟实训中的技能提升与生产优化
在制造业蓬勃发展的大背景下,虚拟教学实训宛如一颗璀璨的新星,正发挥着不可或缺且日益凸显的关键作用,源源不断地为企业的稳健前行与创新发展注入磅礴强大的动力。就以汽车制造企业这一极具代表性的行业主体为例,汽车生产线上各类…...
Unit 1 深度强化学习简介
Deep RL Course ——Unit 1 Introduction 从理论和实践层面深入学习深度强化学习。学会使用知名的深度强化学习库,例如 Stable Baselines3、RL Baselines3 Zoo、Sample Factory 和 CleanRL。在独特的环境中训练智能体,比如 SnowballFight、Huggy the Do…...
大语言模型(LLM)中的KV缓存压缩与动态稀疏注意力机制设计
随着大语言模型(LLM)参数规模的增长,推理阶段的内存占用和计算复杂度成为核心挑战。传统注意力机制的计算复杂度随序列长度呈二次方增长,而KV缓存的内存消耗可能高达数十GB(例如Llama2-7B处理100K token时需50GB内存&a…...

SAP学习笔记 - 开发26 - 前端Fiori开发 OData V2 和 V4 的差异 (Deepseek整理)
上一章用到了V2 的概念,其实 Fiori当中还有 V4,咱们这一章来总结一下 V2 和 V4。 SAP学习笔记 - 开发25 - 前端Fiori开发 Remote OData Service(使用远端Odata服务),代理中间件(ui5-middleware-simpleproxy)-CSDN博客…...

AirSim/Cosys-AirSim 游戏开发(四)外部固定位置监控相机
这个博客介绍了如何通过 settings.json 文件添加一个无人机外的 固定位置监控相机,因为在使用过程中发现 Airsim 对外部监控相机的描述模糊,而 Cosys-Airsim 在官方文档中没有提供外部监控相机设置,最后在源码示例中找到了,所以感…...

三分算法与DeepSeek辅助证明是单峰函数
前置 单峰函数有唯一的最大值,最大值左侧的数值严格单调递增,最大值右侧的数值严格单调递减。 单谷函数有唯一的最小值,最小值左侧的数值严格单调递减,最小值右侧的数值严格单调递增。 三分的本质 三分和二分一样都是通过不断缩…...