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

shell脚本实战学习笔记#1

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

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

完整脚本:

#!/bin/bash

#获取用户输入信息
number1=$1
number2=$2

#判断用户输入的变量个数
check_input(){
    if [ $# -lt 2 ]; then
        echo "当前输入的数值小于2,请输入需要比较的两个数字,两个数字之间用空格分开,谢谢合作!"
        exit 1
    elif [ $# -gt 2 ]; then 
        echo "当前脚本仅支持比较两个数据,暂不支持两个以上数据的比较,请重新输入,谢谢合作!"
        exit 1
    fi
}

#定义一个正在表达式来比较用户输入的是否为数字
re='^[+-]?[0-9]+([.][0-9]+)?$'

#执行判断条件,分别判断两个输入是否为数字
check_Number(){
    if ! [[ $number1 =~ $re ]] || ! [[ $number2 =~ $re ]]; then
        echo "您输入的信息有误,请输入两个数字,并用空格分开,谢谢!"
        exit 1
    fi
}

#执行比较
judge_number(){
    echo "正在执行比较运算,请稍后..."
    sleep 5

    if [ $number1 -gt $number2 ]; then
        echo "$number1大于判断值$number2"
    elif [ $number1 -eq $number2 ];then 
        echo "$number1等于判断值$number2"
    else
        echo "$number1小于判断值$number2"
    fi
}

#主函数运行部分,定义执行哪些函数,以及执行的先后顺序
main(){
    check_input "$@"
    check_Number
    judge_number
}
#程序运行
main "$@" 

需求分析:

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

实现过程:

#!/bin/bash

# 定义变量
number=10

# 条件判断
if [ $number -gt 10 ]; then
    echo "数字大于10"
elif [ $number -eq 10 ]; then
    echo "数字等于10"
else
    echo "数字小于10"
fi

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

#!/bin/bash

# 检查是否传入了两个参数
if [ $# -ne 2 ]; then
    echo "请输入需要比较的两个数字,两个数字之间用空格分开,谢谢合作!"
    exit 1
fi

# 获取参数
number1=$1
number2=$2

# 进行比较
if [ $number1 -gt $number2 ]; then
    echo "$number1 大于 $number2"
elif [ $number1 -eq $number2 ]; then
    echo "$number1 等于 $number2"
else
    echo "$number1 小于 $number2"
fi

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

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

#!/bin/bash

# 判断用户输入的变量个数
if [ $# -lt 2 ]; then
    echo "当前输入的数值小于2,请输入需要比较的两个数字,两个数字之间用空格分开,谢谢合作!"
    exit 1
elif [ $# -gt 2 ]; then
    echo "当前脚本仅支持比较两个数据,暂不支持两个以上数据的比较,请重新输入,谢谢合作!"
    exit 1
fi

# 定义变量
number1=$1
number2=$2

# 判断输入是否为数字
re='^[+-]?[0-9]+([.][0-9]+)?$'
if ! [[ $number1 =~ $re ]] || ! [[ $number2 =~ $re ]]; then
    echo "输入有误,请确保输入的是两个数字。"
    exit 1
fi

echo "正在执行比较运算,请稍后..."
sleep 5

# 进行比较
if [ $number1 -gt $number2 ]; then
    echo "$number1大于判断值$number2"
elif [ $number1 -eq $number2 ]; then
    echo "$number1等于判断值$number2"
else
    echo "$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 ]]; then
    echo "匹配成功"
else
    echo "匹配失败"
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 ]; then
    echo "当前输入的数值小于2,请输入需要比较的两个数字,两个数字之间用空格分开,谢谢合作!"
    exit 1
elif [ $# -gt 2 ]; then 
    echo "当前脚本仅支持比较两个数据,暂不支持两个以上数据的比较,请重新输入,谢谢合作!"
    exit 1
else
    echo "正在执行比较运算,请稍后..."
    sleep 5
fi
}

#定义一个正在表达式来比较用户输入的是否为数字
re='^[+-]?[0-9]+([.][0-9]+)?$'

#执行判断条件,分别判断两个输入是否为数字
check_Number(){
if ! [[ $number1 =~ $re ]] || ! [[ $number2 =~ $re ]]; then
    echo "您输入的信息有误,请输入两个数字,并用空格分开,谢谢!"
    exit 1
fi
}

#执行比较
judge_number(){
if [ $number1 -gt $number2 ]; then
    echo "$number1大于判断值$number2"
elif [ $number1 -eq $number2 ];then 
    echo "$number1等于判断值$number2"
else
    echo "$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_Number
    judge_number
}

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

#!/bin/bash

# 其他函数定义...

main() {
    # 在这里,"$@" 会包含传递给 main 函数的所有参数
    check_input "$@"
    check_Number
    judge_number
}

# 脚本的最后调用 main 函数,并传递所有脚本参数
main "$@"

完整修改后的代码:

#!/bin/bash

#获取用户输入信息
number1=$1
number2=$2

#判断用户输入的变量个数
check_input(){

if [ $# -lt 2 ]; then
    echo "当前输入的数值小于2,请输入需要比较的两个数字,两个数字之间用空格分开,谢谢合作!"
    exit 1
elif [ $# -gt 2 ]; then 
    echo "当前脚本仅支持比较两个数据,暂不支持两个以上数据的比较,请重新输入,谢谢合作!"
    exit 1
else
    echo "正在执行比较运算,请稍后..."
    sleep 5
    judge_number "$@"
fi
}

#定义一个正在表达式来比较用户输入的是否为数字
re='^[+-]?[0-9]+([.][0-9]+)?$'

#执行判断条件,分别判断两个输入是否为数字
check_Number(){
if ! [[ $number1 =~ $re ]] || ! [[ $number2 =~ $re ]]; then
    echo "您输入的信息有误,请输入两个数字,并用空格分开,谢谢!"
    exit 1
fi
}

#执行比较
judge_number(){
if [ $number1 -gt $number2 ]; then
    echo "$number1大于判断值$number2"
elif [ $number1 -eq $number2 ];then 
    echo "$number1等于判断值$number2"
else
    echo "$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 ]; then
        echo "当前输入的数值小于2,请输入需要比较的两个数字,两个数字之间用空格分开,谢谢合作!"
        exit 1
    elif [ $# -gt 2 ]; then 
        echo "当前脚本仅支持比较两个数据,暂不支持两个以上数据的比较,请重新输入,谢谢合作!"
        exit 1
    else
        echo "正在执行比较运算,请稍后..."
        sleep 5
    fi
}

# 定义正则表达式来比较用户输入的是否为数字
re='^[+-]?[0-9]+([.][0-9]+)?$'

# 执行判断条件,分别判断两个输入是否为数字
check_Number() {
    if ! [[ $number1 =~ $re ]] || ! [[ $number2 =~ $re ]]; then
        echo "您输入的信息有误,请输入两个数字,并用空格分开,谢谢!"
        exit 1
    fi
}

# 执行比较
judge_number() {
    if [ $number1 -gt $number2 ]; then
        echo "$number1大于判断值$number2"
    elif [ $number1 -eq $number2 ]; then 
        echo "$number1等于判断值$number2"
    else
        echo "$number1小于判断值$number2"
    fi
}

# 主函数运行部分,定义执行哪些函数,以及执行的先后顺序
main() {
    check_input "$@"
    check_Number
    judge_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 ]; then
        echo "当前输入的数值小于2,请输入需要比较的两个数字,两个数字之间用空格分开,谢谢合作!"
        exit 1
    elif [ $# -gt 2 ]; then 
        echo "当前脚本仅支持比较两个数据,暂不支持两个以上数据的比较,请重新输入,谢谢合作!"
        exit 1
    fi
}

# 定义正则表达式来比较用户输入的是否为数字
re='^[+-]?[0-9]+([.][0-9]+)?$'

# 执行判断条件,分别判断两个输入是否为数字
check_Number() {
    if ! [[ $number1 =~ $re ]] || ! [[ $number2 =~ $re ]]; then
        echo "您输入的信息有误,请输入两个数字,并用空格分开,谢谢!"
        exit 1
    fi
}

# 执行比较
judge_number() {
    echo "正在执行比较运算,请稍后..."
    sleep 5

    if [ $number1 -gt $number2 ]; then
        echo "$number1大于判断值$number2"
    elif [ $number1 -eq $number2 ]; then 
        echo "$number1等于判断值$number2"
    else
        echo "$number1小于判断值$number2"
    fi
}

# 主函数运行部分,定义执行哪些函数,以及执行的先后顺序
main() {
    check_input "$@"
    check_Number
    judge_number
}

# 程序运行
main "$@"

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

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

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

在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

数字乡村智慧农业云平台建设方案:PPT全文30页,附下载

关键词:数字乡村解决方那,智慧农业解决方案,智慧农业建设,数字乡村平台,智慧农业大数据平台,智慧农业项目建设规划 一、对“互联网农业”的理解 “互联网农业”是指将互联网技术与农业生产、加工、销售等…

Elasticsearch可视化平台Kibana [ES系列] - 第498篇

历史文章(文章累计490) 《国内最全的Spring Boot系列之一》 《国内最全的Spring Boot系列之二》 《国内最全的Spring Boot系列之三》 《国内最全的Spring Boot系列之四》 《国内最全的Spring Boot系列之五》 《国内最全的Spring Boot系列之六》 全…

串口服务器助力环境监测系统高效管理、远程监控

物联网的发展使得环境监测系统具备了更强大的数据采集和分析能力。传统的环境监测系统通常需要人工到现场采集数据,费时费力且容易受到外界干扰。而通过使用串口服务器,可以实现环境监测系统的远程数据采集和监控,从而提高监测效率和数据准确…

雷盛红酒分享葡萄酒在冬天如何运输和保存

这两天,强大的寒潮从西向东横扫我国大部地区。我国北方地区未来几天气温持续偏低,冷上加冷;南方的降温大幕今天也将开启,江南、华南部分地区将出现6~12℃的降温。如此寒冷的冬天,对葡萄酒来说,并不是什么好…

Vue实现响应式布局

前提准备:响应式布局有两种方法,看自己想要哪种。 方法一:百分比 用百分比去写元素的宽度,然后让子元素撑起父元素的高度 .parent {width: 50%; }.child {width:100%;height:100px; } 方法二:vh、vw vw、vh是基于视…

【excel密码】Excel工作表不能复制或移动

为什么excel文件打开之后,工作表里是可以编辑的,但是想要移动工作表或者复制、重命名等操作,这是什么原因?其实这是因为设置了工作簿保护,设置了保护的工作簿无法对整张工作表进行操作。 想要取消这种保护,…

MongoDB查询文档

3.5 MongoDB 查询文档 MongoDB 查询文档使用 find() 方法。 find() 方法以非结构化的方式来显示所有文档。find()查询数据的语法格式如下: db.collection.find(query, projection)[.pretty()] query :可选,使用查询操作符指定查询条件 pr…

AI Earth平台简介

AI Earth地球科学云平台由达摩院-视觉技术实验室打造,基于地球科学智能计算分析方面的创新研究,致力于解决地球科学领域基础性、前沿性、业务性问题,目标成为国内一流的地球科学云计算平台。(摘自官网) 下面&#xff…

C++ 比 C语言的新增的特性 1

1. C是C的增强 1.1 C是静态类型的语言,具有严格的数据类型检查 1.1.1 c 因为const修饰的变量不允许修改,但是只给了警告,不严谨 const int a10;a20; //报错int *p&a;*p20;//a的值? test1.c:6:9: warning: initialization dis…

教你如何开发并运营小程序商城或APP商城!

随着线下租金、仓储等成本的攀升,商家们面临着越来越大的压力。为了降低成本、提高效率,越来越多的商家开始转型做电商,甚至直接开发自己的电商商城小程序或APP。那么,商城小程序或APP该如何开发呢?又该如何运营呢&…

别再错过,100张BI报表,动动手指就能用

别再错过了!覆盖财务、销售、库存、采购、应收、生产六大主题,百张BI报表,不需要开发,下了就能用。 简单注册、下载方案并执行,即可完成六大主题数据分析! 在以前,即使有BI方案,要…

【MYSQL】MYSQL 的学习教程(八)之 12 种慢 SQL 查询原因

日常开发中,我们经常会遇到数据库慢查询。那么导致数据慢查询都有哪些常见的原因呢?今天就跟大家聊聊导致 MySQL 慢查询的 12 个常见原因,以及对应的解决方法: SQL 没加索引SQL 索引失效limit 深分页问题单表数据量太大join 或者…

《网络是怎样连接的》1.2、1.3节图表(自用)

图2.1:浏览器调用socket库中的解析器,向DNS服务器询问域名的ip地址 (图中的gethostbyname是解析器的名称;协议栈是操作系统的网络控制软件,也称协议驱动、TCP/IP驱动) 图2.2 DNS服务器根据客户端查询信息查…

云渲染ai加速怎么加速?云渲染ai加速平台推荐

在当下的视觉设计行业中,渲染速度成为设计师们亟需解决的难题。以往,高质量的效果图渲染需要耗费大量时间,特别是在处理复杂的视觉效果时,传统的渲染流程可能需要长时间等待,随着人工智能技术的快速进步,利…

基于AT89C51单片机的8位密码锁仿真与实物制作

点击链接获取Keil源码与Project Backups仿真图: https://download.csdn.net/download/qq_64505944/88657969?spm1001.2014.3001.5503 源码获取 C 源码仿真图毕业设计实物制作步骤01 摘要 在日常的生活和工作中, 住宅与部门的安全防范、单位的文件档案、财务报表…

阶段七-GitEE

Git:版本控制软件 Git的优点 1.1 协同修改 多人并行不悖的修改服务器端的同一个文件。 1.2 数据备份 不仅保存目录和文件的当前状态,还能够保存每一个提交过的历史状态。 1.3 版本管理 在保存每一个版本的文件信息的时候要做到不保存重复数据&…

搭建本地的pip镜像源

1. 创建文件夹:./pypi_mirror_test 2. 创建并进入conda虚拟环境,安装pip2pi包 pip install pip2pi 3. 下载pypi的packages 可以参考其他博客,或者我之前的博客. 偷懒,仅仅测试用的话: 1)在文件夹下创…

Django之按钮(actions)

开篇就是道歉,哈哈哈哈,托更了好久好久,最近太忙了没啥时间更新,各位看官有催更的阔以给我私信哇,希望各位看官给个三连!!!😍😍😍😍 …

操作系统期末复习知识点二计算与应用

1.理解银行家算法判断死锁的定理并能计算相关的参数。 2.能利用LRU、FIFO算法求缺页率。 3.纯页式管理中,求逻辑地址对应的物理地址,页号、页内地址长度,画出逻辑地址的格式,在引入块表时,求出有效访问时间。 4.可变分…

高德地图_公共交通路径规划API,获取两地点之间的驾车里程和时间

import pandas as pd import requests import jsondef get_dis_tm(origin, destination,city,cityd):url https://restapi.amap.com/v3/direction/transit/integrated?key xxx #这里就是需要去高德开放平台去申请key,请在xxxx位置填写,web服务APIlink {}origin{}&desti…