shell function

😀
这里写文章的前言: 一个简单的开头,简述这篇文章讨论的问题、目标、人物、背景是什么?并简述你给出的答案。
可以说说你的故事:阻碍、努力、结果成果,意外与转折。
 

📝 主旨内容

linux基础—-shell函数

函数function是由若干条shell命令组成的语句块,实现代码重用和模块化编程 它与shell程序形式上是相似的,不同的是它不是一个单独的进程,不能独立运行,而是shell程序的一部分函数和shell程序比较相似,区别在于
  • Shell程序在子Shell中运行
  • 而Shell函数在当前Shell中运行。因此在当前Shell中,函数可以对shell中变量进行修改

定义函数

函数由两部分组成:函数名和函数体help function语法一: f_name (){ ...函数体... } 语法二: function f_name { ...函数体... } 语法三: function f_name () { ...函数体... }

函数使用

函数的定义和使用:
  • 可在交互式环境下定义函数
  • 可将函数放在脚本文件中作为它的一部分
  • 可放在只包含函数的单独文件中 调用:函数只有被调用才会执行
  • 调用:给定函数名
  • 函数名出现的地方,会被自动替换为函数代码 函的生命周期:被调用时创建,返回时终止

函数返回值

函数有两种返回值: 函数的执行结果返回值: (1) 使用echo等命令进行输出 (2) 函数体中调用命令的输出结果 函数的退出状态码: (1) 默认取决于函数中执行的最后一条命令的退出状态码 (2) 自定义退出状态码,其格式为: return 从函数中返回,用最后状态命令决定返回值(退出函数本身) return 0 无错误返回 return 1-255 有错误返回
交互式环境下定义和使用函数 示例:
定义该函数后,若在$后面键入dir,其显示结果同ls -l的作用相同dir 该dir函数将一直保留到用户从系统退出,或执行了如下所示的unset命令unset dir
在脚本中定义及使用函数 函数在使用前必须定义,因此应将函数定义放在脚本开始部分,直至shell首次发现它后才能使用 调用函数仅使用其函数名即可 示例:
使用函数文件 可以将经常使用的函数存入函数文件,然后将函数文件载入shell 文件名可任意选取,但最好与相关任务有某种联系。例如:functions.main 一旦函数文件载入shell,就可以在命令行或脚本中调用函数。可以使用set命令查看所有定义的函数,其输出列表包括已经载入shell的所有函数 若要改动函数,首先用unset命令从shell中删除函数。改动完毕后,再重新载入此文件
创建函数文件 函数文件示例:
载入函数 函数文件已创建好后,要将它载入shell 定位函数文件并载入shell的格式
. filename 或 source filename
注意此即<点> <空格> <文件名> 这里的文件名要带正确路径
示例:
上例中的函数,可使用如下命令. functions.main
检查载入函数 使用set命令检查函数是否已载入。set命令将在shell中显示所有的载入函数
示例:

执行shell函数

要执行函数,简单地键入函数名即可 示例:findit groups/usr/bin/groups/usr/local/backups/groups.bak

删除shell函数

现在对函数做一些改动后,需要先删除函数,使其对shell不可用。使用unset命令完成删除函数 命令格式为:unset function_name 示例:unset findit 再键入set命令,函数将不再显示 环境函数 使子进程也可使用 声明:export -f function_name 查看:export -f 或 declare -xf

函数参数

函数可以接受参数: 传递参数给函数:调用函数时,在函数名后面以空白分隔给定参数列表即可;例如“testfunc arg1 arg2 …” 在函数体中当中,可使用$1, $2, …调用这些参数;还可以使用$@, $*, $#等特殊变量

函数变量

变量作用域: 环境变量:当前shell和子shell有效 本地变量:只在当前shell进程有效,为执行脚本会启动专用子shell进程;因此,本地变量的作用范围是当前shell脚本程序文件,包括脚本中的函数 局部变量:函数的生命周期;函数结束时变量被自动销毁 注意:如果函数中有局部变量,如果其名称同本地变量,使用局部变量 在函数中定义局部变量的方法 local NAME=VALUE

函数递归示例

函数递归: 函数直接或间接调用自身 注意递归层数 递归实例: 阶乘是基斯顿·卡曼于 1808 年发明的运算符号,是数学术语,一个正整数的阶 乘(factorial)是所有小于及等于该数的正整数的积,并且有0的阶乘为1,自然 数n的阶乘写作n!n!=1×2×3×...×n 阶乘亦可以递归方式定义:0!=1,n!=(n-1)!×n n!=n(n-1)(n-2)...1 n(n-1)! = n(n-1)(n-2)!
函数递归示例 示例:fact.sh

ork炸弹

fork炸弹是一种恶意程序,它的内部是一个不断在fork进程的无限循环,实质是 一个简单的递归程序。由于程序是递归的,如果没有任何限制,这会导致这个 简单的程序迅速耗尽系统里面的所有资源 函数实现:(){ :|:& };:bomb() { bomb | bomb & }; bomb 脚本实现

练习

编写函数,实现OS的版本判断
编写函数,实现取出当前系统eth0的IP地址
编写函数,实现打印绿色OK和红色FAILED
编写函数,实现判断是否无位置参数,如无参数,提示错误
编写服务脚本/root/bin/testsrv.sh,完成如下要求 (1) 脚本可接受参数:start, stop, restart, status (2) 如果参数非此四者之一,提示使用格式后报错退出 (3) 如是start:则创建/var/lock/subsys/SCRIPT\_NAME, 并显示“启动成功” 考虑:如果事先已经启动过一次,该如何处理? (4) 如是stop:则删除/var/lock/subsys/SCRIPT\_NAME, 并显示“停止完成” 考虑:如果事先已然停止过了,该如何处理? (5) 如是restart,则先stop, 再start 考虑:如果本来没有start,如何处理? (6) 如是status, 则如果/var/lock/subsys/SCRIPT_NAME文件存在,则显示“SCRIPT_NAME isrunning...”,如果/var/lock/subsys/SCRIPT_NAME文件不存在,则显示“SCRIPT_NAME isstopped...” (7)在所有模式下禁止启动该服务,可用chkconfig 和 service命令管理 说明:SCRIPT\_NAME为当前脚本名
编写脚本/root/bin/copycmd.sh (1) 提示用户输入一个可执行命令名称 (2) 获取此命令所依赖到的所有库文件列表 (3) 复制命令至某目标目录(例如/mnt/sysroot)下的对应路径下如:/bin/bash ==> /mnt/sysroot/bin/bash /usr/bin/passwd ==> /mnt/sysroot/usr/bin/passwd (4) 复制此命令依赖到的所有库文件至目标目录下的对应路径下: 如:/lib64/ld-linux-x86-64.so.2 ==> /mnt/sysroot/lib64/ld-linux-x86-64.so.2 (5)每次复制完成一个命令后,不要退出,而是提示用户键入新的要复制的命令,并重复完成上述功能;直到用户输入quit退出
编写函数实现两个数字做为参数,返回最大值 斐波那契数列又称黄金分割数列,因数学家列昂纳多·斐波那契以兔子繁殖为例子而引入,故又称为“兔子数列”,指的是这样一个数列:0、1、1、2、3、5、8、13、21、34、……,斐波纳契数列以如下被以递归的方法定义:F(0)=0,F(1)=1,F(n)=F(n-1)+F(n-2)(n≥2)利用函数,求n阶斐波那契数列
汉诺塔(又称河内塔)问题是源于印度一个古老传说。大梵天创造世界的时候做了三根金刚石柱子,在一根柱子上从下往上按照大小顺序摞着64片黄金圆盘。 大梵天命令婆罗门把圆盘从下面开始按大小顺序重新摆放在另一根柱子上。并且规定,在小圆盘上不能放大圆盘,在三根柱子之间一次只能移动一个圆盘,利用函数,实现N片盘的汉诺塔的移动步骤

信号捕捉trap

trap ‘触发指令’ 信号 进程收到系统发出的指定信号后,将执行自定义指令,而不会执行原操作trap '' 信号 忽略信号的操作trap '-' 信号 恢复原信号的操作trap -p 列出自定义信号操作trap finish EXIT 当脚本退出时,执行finish函数
trap示例 #kill -l查看信号列表

转载请注明来源,欢迎对文章中的引用来源进行考证,欢迎指出任何有错误或不够清晰的表达。可以在下面评论区评论,也可以邮件至 438803792@qq.com

🤗 总结归纳

总结文章的内容

📎 参考文章

  • 一些引用
  • 引用文章
 
💡
有关Notion安装或者使用上的问题,欢迎您在底部评论区留言,一起交流~
Loading...