【shell脚本实战学习笔记】#1

shell脚本实战学习笔记#1

脚本编写场景需求:
编写一个比较数据大小的shell脚本,要求判断用户只能输入两位数字,不能是字符或其他特殊字符;并且在shell脚本中需要用到函数来控制执行顺序。

知识点:shell函数|正则匹配|全局变量

完整脚本:

#!/bin/bash#获取用户输入信息
number1=$1
number2=$2#判断用户输入的变量个数
check_input(){if [ $# -lt 2 ]; thenecho "当前输入的数值小于2,请输入需要比较的两个数字,两个数字之间用空格分开,谢谢合作!"exit 1elif [ $# -gt 2 ]; then echo "当前脚本仅支持比较两个数据,暂不支持两个以上数据的比较,请重新输入,谢谢合作!"exit 1fi
}#定义一个正在表达式来比较用户输入的是否为数字
re='^[+-]?[0-9]+([.][0-9]+)?$'#执行判断条件,分别判断两个输入是否为数字
check_Number(){if ! [[ $number1 =~ $re ]] || ! [[ $number2 =~ $re ]]; thenecho "您输入的信息有误,请输入两个数字,并用空格分开,谢谢!"exit 1fi
}#执行比较
judge_number(){echo "正在执行比较运算,请稍后..."sleep 5if [ $number1 -gt $number2 ]; thenecho "$number1大于判断值$number2"elif [ $number1 -eq $number2 ];then echo "$number1等于判断值$number2"elseecho "$number1小于判断值$number2"fi
}#主函数运行部分,定义执行哪些函数,以及执行的先后顺序
main(){check_input "$@"check_Numberjudge_number
}
#程序运行
main "$@" 

需求分析:

  1. 首先要让用户执行脚本时传入两个参数,因此给定$1 $2两个变量;
  2. 需求表明只能输入两个数字,因此判断逻辑有两层:判断变量个数为2个;判断输入内容是否为数字。
  3. 若满足两个判断条件,需要做数字比较的动作,这是脚本的核心需求。

实现过程:

#!/bin/bash# 定义变量
number=10# 条件判断
if [ $number -gt 10 ]; thenecho "数字大于10"
elif [ $number -eq 10 ]; thenecho "数字等于10"
elseecho "数字小于10"
fi

在这个脚本中,我们首先定义了一个变量number,然后通过if、elif和else语句检查这个变量的值,并打印出相应的信息。这里的-gt表示"大于"(greater than),-eq表示"等于"(equal)。
通过这个脚本我们初步实现了脚本最基本的比较功能,但这是跟一个固定数值去做比较,我们需要修改逻辑,让脚本实现两个数据的对比:

#!/bin/bash# 检查是否传入了两个参数
if [ $# -ne 2 ]; thenecho "请输入需要比较的两个数字,两个数字之间用空格分开,谢谢合作!"exit 1
fi# 获取参数
number1=$1
number2=$2# 进行比较
if [ $number1 -gt $number2 ]; thenecho "$number1 大于 $number2"
elif [ $number1 -eq $number2 ]; thenecho "$number1 等于 $number2"
elseecho "$number1 小于 $number2"
fi

在这个脚本中,我使用$#来检查传入的参数个数,如果不等于2(-ne 表示不等于),则打印提示信息并退出(exit 1)。之后,我用$1和$2来获取用户输入的两个数字,并进行比较。

但是我还想在这个脚本的基础上去做优化,现在能够判断的仅仅是输入的数据个数,我想添加判断条件判别用户输入的是否是字符或者其他东西,如果是字符,则抛出一个提示说目前不支持字符比较,请输入两位数字;如果输入的是特殊字符,则抛出提示说输入字符有误,请重新输入,因此继续调整:

#!/bin/bash# 判断用户输入的变量个数
if [ $# -lt 2 ]; thenecho "当前输入的数值小于2,请输入需要比较的两个数字,两个数字之间用空格分开,谢谢合作!"exit 1
elif [ $# -gt 2 ]; thenecho "当前脚本仅支持比较两个数据,暂不支持两个以上数据的比较,请重新输入,谢谢合作!"exit 1
fi# 定义变量
number1=$1
number2=$2# 判断输入是否为数字
re='^[+-]?[0-9]+([.][0-9]+)?$'
if ! [[ $number1 =~ $re ]] || ! [[ $number2 =~ $re ]]; thenecho "输入有误,请确保输入的是两个数字。"exit 1
fiecho "正在执行比较运算,请稍后..."
sleep 5# 进行比较
if [ $number1 -gt $number2 ]; thenecho "$number1大于判断值$number2"
elif [ $number1 -eq $number2 ]; thenecho "$number1等于判断值$number2"
elseecho "$number1小于判断值$number2"
fi
正则表达式(Regular Expression):这是一个强大的工具,用于匹配字符串模式。在这个脚本中,我使用了正则表达式'^[+-]?[0-9]+([.][0-9]+)?$'来检测输入是否为数字。这个表达式支持整数和浮点数,包括可选的正负号。条件判断:通过[[ $variable =~ $re ]]来判断变量$variable是否符合正则表达式$re定义的模式。如果不符合,我们打印错误消息并退出脚本。增强的用户体验和错误处理:通过确保输入为数字,脚本的鲁棒性和用户体验都得到了提升。如果用户输入了非数字,脚本会提供明确的反馈,而不是产生不可预测的行为。

正则表达式中的 在正则表达式中通常用来表示匹配的开始。它指定了模式必须出现在字符串的开头。

[±] 后的 ?:这个 ? 表示前面的字符组 [±](即正号或负号)是可选的。它表示正号或负号可以出现一次,或者不出现。

[0-9] 是匹配一个数字:[0-9] 表示匹配单个数字,即0到9之间的任意一个数字。

括号前面的 +:这个 + 表示前面的模式(括号中的模式)必须至少出现一次。在这个正则表达式中,[0-9]+ 表示至少有一个数字。

[0-9]+ 意思是匹配多个0-9之间的数字:[0-9]+ 表示匹配一个或多个数字。

正则表达式最后的 ?$ :这里的 ? 表示前面的分组(即括号内的模式)是可选的。$ 表示匹配的结束,确实和 ^ 是对应的。^ 表示开头,$ 表示结尾。

判断数据是否符合正则表达式的 if 语句中为什么是两个中括号:在 Bash 中,双中括号 [[ ]] 用于条件表达式。与单中括号 [ ] 相比,双中括号提供了更多的功能,比如模式匹配和正则表达式支持。在这种情况下,使用双中括号可以让我们利用正则表达式来检查变量值。

在 Bash 中,=~ 是一个特定的操作符,用于在条件表达式中执行正则表达式匹配。这个操作符仅在双中括号的条件测试([[ ]])中有效。

if [[ $variable =~ $re ]]; thenecho "匹配成功"
elseecho "匹配失败"
fi

这里, $variable是你要检查的字符串,$re 是一个正则表达式。如果 $variable符合这个正则表达式,那么表达式的结果为真(true),否则为假(false)。

注意点:

只在双中括号中有效:=~ 操作符只能在 [[ ]] 条件测试中使用。如果在单中括号 [ ] 中使用,它将不会按照预期工作。引用和不引用的区别:在使用正则表达式时,通常最好不要引用正则表达式变量(即使用 $re 而不是 "$re"),因为引用会使得表达式被当作普通字符串处理,而非正则表达式。Bash版本:请注意,这个特性在较老的 Bash 版本中可能不可用。它在 Bash 3.0 及以后的版本中是有效的。正则表达式的兼容性:Bash 使用的正则表达式是基于 POSIX ERE(扩展正则表达式),它与某些其他编程语言中使用的正则表达式(如Perl兼容正则表达式)略有不同。

所以,=~ 是一个强大的工具,用于在 Bash 脚本中进行模式匹配和数据验证。希望这些信息能帮助你更好地理解和使用这个特性!如果还有其他问题或疑惑,随时欢迎提问!

继续执行,将脚本内容用函数进行控制:

#!/bin/bash#获取用户输入信息
number1=$1
number2=$2#判断用户输入的变量个数
check_input(){if [ $# -lt 2 ]; thenecho "当前输入的数值小于2,请输入需要比较的两个数字,两个数字之间用空格分开,谢谢合作!"exit 1
elif [ $# -gt 2 ]; then echo "当前脚本仅支持比较两个数据,暂不支持两个以上数据的比较,请重新输入,谢谢合作!"exit 1
elseecho "正在执行比较运算,请稍后..."sleep 5
fi
}#定义一个正在表达式来比较用户输入的是否为数字
re='^[+-]?[0-9]+([.][0-9]+)?$'#执行判断条件,分别判断两个输入是否为数字
check_Number(){
if ! [[ $number1 =~ $re ]] || ! [[ $number2 =~ $re ]]; thenecho "您输入的信息有误,请输入两个数字,并用空格分开,谢谢!"exit 1
fi
}#执行比较
judge_number(){
if [ $number1 -gt $number2 ]; thenecho "$number1大于判断值$number2"
elif [ $number1 -eq $number2 ];then echo "$number1等于判断值$number2"
elseecho "$number1小于判断值$number2"
fi
}#主函数运行部分,定义执行哪些函数,以及执行的先后顺序
main(){
check_input
check_Number
judge_number
}
#程序运行
main

此时,我将脚本修改为了函数的方式,但是当我运行的时候我发现最开始number两个变量获取到了用户的输入,在执行check_input函数的时候我发现此时函数判断的$#是0,而不是用户实际输入的信息

问题分析:
特殊变量 $# 在函数中的作用域:在 Bash 脚本中,当你在函数内部使用 $# 时,它代表的是传递给该函数的参数个数,而不是传递给脚本的参数个数。由于你在调用 check_input 函数时没有传递任何参数,所以 $# 在 check_input 函数内部的值是 0。

解决方案:
在 main 函数中传递所有脚本参数:当调用 check_input 时,我们使用 $@ 来传递所有脚本参数。$@是一个特殊的变量,它代表了所有传递给脚本的参数,每个参数作为独立的引用字符串。

main() {check_input "$@"check_Numberjudge_number
}

继续执行发现仍然输出不对,因为单独在脚本的最末尾调用了main函数,此时main函数并未传递全局变量$@,因此在上述main函数中,实际并未传入用户输入的数据,继续修改如下:

#!/bin/bash# 其他函数定义...main() {# 在这里,"$@" 会包含传递给 main 函数的所有参数check_input "$@"check_Numberjudge_number
}# 脚本的最后调用 main 函数,并传递所有脚本参数
main "$@"

完整修改后的代码:

#!/bin/bash#获取用户输入信息
number1=$1
number2=$2#判断用户输入的变量个数
check_input(){if [ $# -lt 2 ]; thenecho "当前输入的数值小于2,请输入需要比较的两个数字,两个数字之间用空格分开,谢谢合作!"exit 1
elif [ $# -gt 2 ]; then echo "当前脚本仅支持比较两个数据,暂不支持两个以上数据的比较,请重新输入,谢谢合作!"exit 1
elseecho "正在执行比较运算,请稍后..."sleep 5judge_number "$@"
fi
}#定义一个正在表达式来比较用户输入的是否为数字
re='^[+-]?[0-9]+([.][0-9]+)?$'#执行判断条件,分别判断两个输入是否为数字
check_Number(){
if ! [[ $number1 =~ $re ]] || ! [[ $number2 =~ $re ]]; thenecho "您输入的信息有误,请输入两个数字,并用空格分开,谢谢!"exit 1
fi
}#执行比较
judge_number(){
if [ $number1 -gt $number2 ]; thenecho "$number1大于判断值$number2"
elif [ $number1 -eq $number2 ];then echo "$number1等于判断值$number2"
elseecho "$number1小于判断值$number2"
fi
}#主函数运行部分,定义执行哪些函数,以及执行的先后顺序
main(){
check_input "$@"
check_Number
#judge_number
}
#程序运行
main "$@" 

我的修改想法是:只有当用户输入的数据是两个的时候才去执行比较的动作,但是我这样修改,然后输入命令bash shell1.sh 20 aaa去执行这个脚本之后是如下的执行结果:

正在执行比较运算,请稍后...
shell1.sh: line 36: [: aaa: integer expression expected
shell1.sh: line 38: [: aaa: integer expression expected
20小于判断值aaa
您输入的信息有误,请输入两个数字,并用空格分开,谢谢!

问题分析:

逻辑顺序问题:从脚本和执行输出来看,judge_number 函数在 check_Number 函数之前被调用。这导致了即使输入的不是数字,judge_number 也会被执行。函数调用顺序:在 check_input 函数中直接调用了 judge_number 函数。这意味着只要输入的参数个数正确,无论参数是否为数字,judge_number 都会被执行。

解决方案:

脚本需要确保所有的检查都按正确的顺序进行:

首先检查参数个数:确保用户输入了两个参数。
然后检查参数类型:确保这两个参数都是数字。
最后执行比较操作。

代码修改思路:

根据上述逻辑,对脚本做如下修改:

在 main 函数中控制整体逻辑流程:先调用 check_input 检查参数个数,然后调用 check_Number 检查参数类型,最后调用 judge_number 进行比较。移除 check_input 中对 judge_number 的直接调用。确保 check_Number 在 judge_number 之前被调用。

再次修改后的脚本:

#!/bin/bash# 获取用户输入信息
number1=$1
number2=$2# 判断用户输入的变量个数
check_input() {if [ $# -lt 2 ]; thenecho "当前输入的数值小于2,请输入需要比较的两个数字,两个数字之间用空格分开,谢谢合作!"exit 1elif [ $# -gt 2 ]; then echo "当前脚本仅支持比较两个数据,暂不支持两个以上数据的比较,请重新输入,谢谢合作!"exit 1elseecho "正在执行比较运算,请稍后..."sleep 5fi
}# 定义正则表达式来比较用户输入的是否为数字
re='^[+-]?[0-9]+([.][0-9]+)?$'# 执行判断条件,分别判断两个输入是否为数字
check_Number() {if ! [[ $number1 =~ $re ]] || ! [[ $number2 =~ $re ]]; thenecho "您输入的信息有误,请输入两个数字,并用空格分开,谢谢!"exit 1fi
}# 执行比较
judge_number() {if [ $number1 -gt $number2 ]; thenecho "$number1大于判断值$number2"elif [ $number1 -eq $number2 ]; then echo "$number1等于判断值$number2"elseecho "$number1小于判断值$number2"fi
}# 主函数运行部分,定义执行哪些函数,以及执行的先后顺序
main() {check_input "$@"check_Numberjudge_number
}# 程序运行
main "$@"

如果我执行的数据是:

bash shell1.sh 20 aaa

脚本有如下输出:

正在执行比较运算,请稍后...
您输入的信息有误,请输入两个数字,并用空格分开,谢谢!

从输出内容来看,只要满足输入的信息是两个,不论是否为数字,都会正常执行比较的输出,
那么是不是可以将check_input函数中的else部分的输出放到judge_number函数里去?

修改建议:

简化 check_input 函数:只保留参数个数的检查,去掉 else 部分的输出。在 judge_number 函数中添加开始执行的提示:这样,只有在所有检查都通过后,才会显示这个提示并执行比较。

最后调整逻辑版本:

#!/bin/bash# 获取用户输入信息
number1=$1
number2=$2# 判断用户输入的变量个数
check_input() {if [ $# -lt 2 ]; thenecho "当前输入的数值小于2,请输入需要比较的两个数字,两个数字之间用空格分开,谢谢合作!"exit 1elif [ $# -gt 2 ]; then echo "当前脚本仅支持比较两个数据,暂不支持两个以上数据的比较,请重新输入,谢谢合作!"exit 1fi
}# 定义正则表达式来比较用户输入的是否为数字
re='^[+-]?[0-9]+([.][0-9]+)?$'# 执行判断条件,分别判断两个输入是否为数字
check_Number() {if ! [[ $number1 =~ $re ]] || ! [[ $number2 =~ $re ]]; thenecho "您输入的信息有误,请输入两个数字,并用空格分开,谢谢!"exit 1fi
}# 执行比较
judge_number() {echo "正在执行比较运算,请稍后..."sleep 5if [ $number1 -gt $number2 ]; thenecho "$number1大于判断值$number2"elif [ $number1 -eq $number2 ]; then echo "$number1等于判断值$number2"elseecho "$number1小于判断值$number2"fi
}# 主函数运行部分,定义执行哪些函数,以及执行的先后顺序
main() {check_input "$@"check_Numberjudge_number
}# 程序运行
main "$@"

以上内容就是基本实现一个数据比较的shell脚本需求,同时也确保了脚本的逻辑更加合理且健壮。

以上便是本次实战操作的所有过程了。

感谢您在百忙之中花费时间阅读本篇文章,期望对您的体能提升有所收获!

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/295566.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

ros2humble使用gazebo加载urdf文件的基本流程

1:编写xacro文件 2:把xacro文件转换为urdf文件 3:使用gazebo建立sdf文件fine,也就是ros1里面的world文件,现在后缀是sdf文件了 4:把urdf文件转换为sdf文件good 5:把修改后的good.sdf文件稍微…

swing快速入门(二十五)

注释很详细,直接上代码 新增内容 1.ImageIO.write读取并显示图片 2.ImageIO.writeImageIO.write读取并保存图片 package swing21_30;import javax.imageio.ImageIO; import java.awt.*; import java.awt.event.WindowAdapter; import java.awt.event.WindowEvent…

内存函数的学习

额外知识点 第一个 假设c为int类型,(char)c之后,之后如果还用变量c的话,c依然为int类型。()强制转换操作符并不会永久改变原本的变量类型。 第二个 \0在打印时不会显示出来 第三个 void …

浏览器原理篇—渲染阻塞

渲染阻塞 1.DOM 的解析 html 文档 边加载边解析 的;网络进程和渲染进程之间会建立一个共享数据的管道,网络进程接收到数据实时传递给渲染进程,渲染进程的 HTML 解析器,它会动态接收字节流,并将其解析为 DOM 2.字节流…

Linux Centos-7.5_64bit 等保测评

一、新增用户 新增test用户 useradd test 设置密码 passwd 修改test的密码 passwd test 修改/etc/sudoers文件,找到下面一行, /etc/sudoers test ALL(ALL) ALL 保存是出现 E45: readonly option is set (add ! to override) 解决办法&#xff…

Unity协程的定义、使用及原理,与线程的区别、缺点全方面解析

目录 协程的定义及简介 协程的用途 定时器 将复杂程序分帧执行 等待某些条件完成后执行后续 异步加载资源 协程的原理 MonoBehaviour中每一帧的游戏循环 迭代器 IEnumerator 接口 具体执行过程 协程和线程的区别 协程的缺点 无法返回值 依赖于MonoBehaviour 维护…

Uniapp + Vue3 + Pinia + Vant3 框架搭建

现在越来越多项目都偏向于Vue3开发&#xff0c;想着uniapp搭配Vue3试试效果怎么样&#xff0c;接下来就是详细操作步骤。 初始化Uniapp Vue3项目 App.vue setup语法 <script setup>import {onLaunch,onShow,onHide} from dcloudio/uni-apponLaunch(() > {console.l…

在k8s中使用Helm安装harbor并将Chart推送到私有仓库harbor

使用Helm安装harbor并将Chart推送到私有仓库harbor 注意&#xff1a;如果你的harbor是之前docker-compose安装的&#xff0c;还需要额外做一个动作&#xff0c;让它支持chart docker-compose stop ./install.sh --with-chartmuseum1&#xff09;下载harbor的chart包 Harbor的…

攻防世界——Hello, CTF

运行可以发现这是输入型的flag &#xff08;re题目分为两类&#xff0c;一种你直接输入flag&#xff0c;还有一种就是你完成某个操作后&#xff0c;给你flag&#xff09; 可以发现关键字符串就是wrong 和 input 32位 IDA打开 进入直接进入字符串界面&#xff0c;发现关键字符…

CSS:浮动

CSS&#xff1a;浮动 浮动效果浮动方式 float浮动特性标准流脱标脱标的影响脱标的影响范围 清除浮动清除浮动原理 clear基于clear的清除浮动方式额外标签法:afert伪元素法双伪元素法 清除浮动原理 BFCBFC定义BFC布局规则创建一个BFC基于BFC的清除浮动方式父级添加overflow法 浮…

UEC++ 捡电池初级案例 day16

捡电池案例 创建第三人称模版 创建Actor基类 创建一个Actor类用来作为可拾取物品基类需求&#xff1a;我们让这个基类有静态网格可识别&#xff0c;然后得有一个状态就是是否被拾取的状态&#xff0c;那就得拥有改变状态的函数与返回状态的函数&#xff0c;然后和返回这个实…

Yestar成都艺星舒适热玛吉星品沙龙会在蓉成功发布

12月21日&#xff0c;由Yestar成都艺星联合索塔医疗联合举办的“舒适无痛热玛吉星品沙龙会”在院内圆满举行&#xff0c;索塔医疗西区大客户经理肖峰、中国临床事业部刘颖&#xff0c;成都艺星运营部长程燕佳&#xff0c;皮肤科院长朱紫婷、技术院长杨海皎、主任王小琴&#xf…