Linux扩展篇-shell编程(三)-shell运算符
基本语法:
格式一
expr +、 -、 \*、/、 %(加、减、乘、除、求余)
格式二
"$((运算式))"或者"$[运算式]"
基本运算符
Shell 和其他编程语言一样,支持多种运算符,包括:
- 算术运算符
- 关系运算符
- 布尔运算符
- 字符串运算符
- 文件测试运算符
- 赋值运算符
- 逻辑运算符
(1)算术运算符
| 运算符 | 说明 | 示例 |
|---|---|---|
| + | 单目正号 | ((+3))结果为3 |
| - | 单目负号 | ((-3))结果为-3 |
| ++ | 自增(变量前,先运算后取值;变量后,先取值后运算) | b=2时((a=++b))后a为3,b为3;((a=b++))后a为2,b为3 |
| – | 自减(变量前,先运算后取值;变量后,先取值后运算) | b=2时((a=--b))后a为1,b为1;((a=b--))后a为2,b为1 |
| + | 加 | expr 3 + 2结果为5 |
| - | 减 | expr 3 - 2结果为1 |
| * | 乘 | expr 3 \* 2结果为6 |
| / | 除 | expr 3 / 2结果为1 |
| % | 取模 | expr 3 % 2结果为1 |
| ** | 幂运算 | ((3**2))结果为9 |
注意:
expr可用于整数运算,也可以处理字符串,使用expr进行运算时,表达式和运算符之间必须加空格,乘号*和小括号()前要加\转义(小括号是左右括号前都要加\转义)。(( expression ))会对算术表达式求值,如果表达式的值不是0,则返回状态是0,否则返回状态是1,这和let "expression"等价。在
(( ))前面加上$符号可以获取(( ))命令的执行结果,即整个表达式的值。
(( ))只能进行整数运算,不能对浮点数或字符串进行运算。
(2)关系运算符
在test、[]或[[ ]]中使用的关系运算符:
| 运算符 | 说明 | 示例 |
|---|---|---|
| -eq | (equal)检测两个数是否相等,是返回0,否则返回1。 | 如果a为3,b为2,则[ ${a} -eq ${b} ]返回1 |
| -ne | (not equal)检测两个数是否不相等,是返回0,否则返回1。 | 如果a为3,b为2,则[ ${a} -ne ${b} ]返回0 |
| -gt | (greater than)检测左边的数是否大于右边的,是返回0,否则返回1。 | 如果a为3,b为2,则[ ${a} -gt ${b} ]返回0 |
| -lt | (less than)检测左边的数是否小于右边的,是返回0,否则返回1。 | 如果a为3,b为2,则[ ${a} -lt ${b} ]返回1 |
| -ge | (greater equal)检测左边的数是否大于等于右边的,是返回0,否则返回1。 | 如果a为3,b为2,则[ ${a} -ge ${b} ]返回0 |
| -le | (less equal)检测左边的数是否小于等于右边的,是返回0,否则返回1。 | 如果a为3,b为2,则[ ${a} -le ${b} ]返回1 |
在(())中使用的关系运算符:
| 运算符 | 说明 | 示例 |
|---|---|---|
| == | 检测两个数是否相等,是返回0,否则返回1。 | 如果a为3,b为2,则((a==b))返回1 |
| != | 检测两个数是否不相等,是返回0,否则返回1。 | 如果a为3,b为2,则((a!=b))返回0 |
| > | 检测左边的数是否大于右边的,是返回0,否则返回1。 | 如果a为3,b为2,则((a>b))返回0 |
| >= | 检测左边的数是否大于等于右边的,是返回0,否则返回1。 | 如果a为3,b为2,则((a>=b))返回0 |
| < | 检测左边的数是否小于右边的,是返回0,否则返回1。 | 如果a为3,b为2,则((a<b))返回1 |
| <= | 检测左边的数是否小于等于右边的,是返回0,否则返回1。 | 如果a为3,b为2,则((a<=b))返回1 |
注意:
- 关系运算符只支持整数数字,不支持字符串,除非字符串的值是整数数字。
(( ))中表达式的值不是0,返回状态才是0,否则返回状态是1,(( ))进行整数比较的时候表达式的值与(())返回状态正好相反,表达式的值为0,返回状态为1,表达式的值为1,返回状态为0。
(3)布尔运算符
| 运算符 | 说明 | 示例 |
|---|---|---|
| -a | 与运算,两个表达式都为true才返回true。 | 如果a为3,b为2,则[ ${a} -gt 1 -a ${b} -lt 1 ]返回false |
| -o | 或运算,有一个表达式为true,则返回true。 | 如果a为3,b为2,则[ ${a} -gt 1 -o ${b} -lt 1 ]返回true |
| ! | 非运算,表达式为true,则返回false,否则返回true。 | 如果b为2,则[ ! ${b} -lt 1 ]返回true |
(-a、-o)必须在[]中或配合test命令使用。
(4)字符串运算符
可以在test、[]或[[ ]]中使用:
| 运算符 | 说明 | 示例 |
|---|---|---|
| ==、= | 如果两个字符串相等,则为true。 | 如果a为"expression",b为"expansion",则[[ ${a} == ${b} ]]返回1 |
| != | 如果两个字符串不相等,则为true。 | 如果a为"expression",b为"expansion",则[[ ${a} != ${b} ]]返回0 |
| > | 如果左边字符串在字典顺序上排在右边字符串之后,则为true。 | 如果a为"expression",b为"expansion",则[[ ${a} > ${b} ]]返回0 |
| < | 如果左边字符串在字典顺序上排在右边字符串之前,则为true。 | 如果a为"expression",b为"expansion",则[[ ${a} < ${b} ]]返回1 |
| -z string | 如果字符串的长度为零,则为true。 | 如果a为"expression",则[[ -z ${a} ]]返回1 |
| -n string | 如果字符串的长度不为零,则为true。 | 如果a为"expression",则[[ -n ${a} ]]返回0 |
| $ | 检测字符串是否不为空,不为空返回 true。 | 如果a为"expression",则[[ ${a} ]]返回0,与-n用法类似 |
注意:
- 在
test、[]或[[ ]]中使用字符串运算符==、=、!=、>、<只能比较字符串(只是字符串的值为数字时可以把它当成数字来比较,这只是字符串比较的结果与数值比较正好巧合,不推荐用它来比较数字,如[[ -3 < -2 ]]返回1,[[ 03 < 2 ]]返回0,[[ 1.5 == 1.50 ]]返回1,都是错误结果。 - 如果要使用
test、[]或[[ ]]比较整数,需使用关系运算符-eq、-ne、-gt、-lt、-ge、-le,推荐在(( ))中使用关系运算符==、!=、>、<、>=、<=进行整数比较。 test或[]使用>和<需要加\转义,[[ ]]则不需要转义。- 使用
test、[]或[[ ]]则没有>=和<=,可以通过[ ${a} \> ${b} -o ${a} == ${b} ]或[[ ${a} > ${b} || ${a} == ${b} ]]这种方式替代。 test或[]会进行单词拆分,而[[ ]]不会进行单词拆分。- 当与
[[ ]]一起使用时,<和>使用当前区域设置按字典顺序排序,test命令使用ASCII排序。
(5)文件测试运算符
| 运算符 | 说明 | 示例 |
|---|---|---|
| -a file | 如果文件存在,则为true。 | [ -a ${file} ] |
| -b file | 如果文件存在并且是个块设备文件,则为true。 | [ -b ${file} ] |
| -c file | 如果文件存在并且是个字符设备文件,则为true。 | [ -c ${file} ] |
| -d file | 如果文件存在并且是个目录,则为true。 | [ -d ${file} ] |
| -e file | 如果文件存在,则为true。 | [ -e ${file} ] |
| -f file | 如果文件存在并且是个普通文件(既不是目录也不是设备文件),则为true。 | [ -f ${file} ] |
| -g file | 如果文件存在并且设置了set-group-id位,则为true。 | [ -g ${file} ] |
| -h file、-L file | 如果文件存在并且是个符号链接,则为true。 | [ -h ${file} ] |
| -k file | 如果文件存在并且设置了“sticky”位,则为true。 | [ -k ${file} ] |
| -p file | 如果文件存在并且是个命名管道(FIFO),则为true。 | [ -p ${file} ] |
| -r file | 如果文件存在并且可读,则为true。 | [ -r ${file} ] |
| -s file | 如果文件存在并且大小大于0,则为true。 | [ -s ${file} ] |
| -u file | 如果文件存在并且设置了set-user-id位,则为true。 | [ -u ${file} ] |
| -w file | 如果文件存在并且可写,则为true。 | [ -w ${file} ] |
| -x file | 如果文件存在并且可执行,则为true。 | [ -x ${file} ] |
| -G file | 如果文件存在并且被有效组id所拥有,则为true。 | [ -G ${file} ] |
| -N file | 如果文件存在并且自上次读取后被修改,则为true。 | [ -N ${file} ] |
| -O file | 如果文件存在并且被有效用户id所拥有,则为true。 | [ -O ${file} ] |
| -S file | 如果文件存在并且是个套接字,则为true。 | [ -S ${file} ] |
| file1 -ef file2 | 如果file1和file2指向相同的设备和inode号,则为true。 | [ ${file1} -ef ${file2} ] |
| file1 -nt file2 | 如果file1比file2更新(根据修改日期),或者file1存在而file2不存在,则为true。 | [ ${file1} -nt ${file2} ] |
| file1 -ot file2 | 如果file1比file2更旧,或者file2存在而file1不存在,则为true | [ ${file1} -ot ${file2} ] |
(6)赋值运算符
| 运算符 | 说明 | 示例 |
|---|---|---|
| = | 赋值 | c=3 a=${c}则a为3 |
| += | 加赋值 | 如果a为3,b为2,则((a+=b))后a为5,b为2 |
| -= | 减赋值 | 如果a为3,b为2,则((a-=b))后a为1,b为2 |
| *= | 乘赋值 | 如果a为3,b为2,则((a*=b))后a为6,b为2 |
| /= | 除赋值 | 如果a为3,b为2,则((a/=b))后a为1,b为2 |
| %= | 取模赋值 | 如果a为3,b为2,则((a%=b))后a为1,b为2 |
| <<= | 左移位赋值 | 如果a为3,b为2,则((a<<=b))后a为12,b为2 |
| >>= | 右移位赋值 | 如果a为3,b为2,则((a>>=b))后a为0,b为2 |
| &= | 按位与赋值 | 如果a为3,b为2,则((a&=b))后a为2,b为2 |
| |= | 按位或赋值 | 如果a为3,b为2,则((a|=b))后a为3,b为2 |
| ^= | 按位异或赋值 | 如果a为3,b为2,则((a^=b))后a为1,b为2 |
可以使用(( ))和let命令进行运算,let和(( ))用法类似,都是用于整数运算。
(7)逻辑运算符
| 运算符 | 说明 | 示例 |
|---|---|---|
| && | 逻辑与 | 如果a为3,b为2,则[[ ${a} > 1 && ${b} < 1 ]]返回false |
| || | 逻辑或 | 如果a为3,b为2,则[[ ${a} > 1 || ${b} < 1 ]]返回true |
| ! | 逻辑非 | 如果b为2,则[[ ! ${b} < 1 ]]返回true |
&&和||可以在[[ ]]或(( ))中使用,不能在test或[]中使用,!可以在[[ ]]中使用,不能在(( ))中使用。
(8)位运算符
| 运算符 | 说明 | 示例 |
|---|---|---|
| << | 左移位 | ((3<<2))结果为12 |
| >> | 右移位 | ((6>>2))结果为1 |
| & | 按位与 | ((6&3))结果为2 |
| | | 按位或 | ((6|3))结果为7 |
| ^ | 按位异或 | ((6^3))结果为5 |
| ~ | 按位非 | ((~6))结果为-7 |
注:加粗部分为较为常用的运算符
概念理解、区分
(1)[ 和 [[ 的区别
区别一:
在 [ 中使用逻辑运算符,需要使用 -a(and)或者 -o(or)。
在 [[ 中使用逻辑运算符,需要使用 && 或者 ||。
区别二:
[] 是 bash 内部命令,与test是等同的,所以字符串比较符 > 和 < 需要转义,否则就变成 io 重定向了。
[[ 是 bash 关键字,不会做命令扩展,所以 < 和 > 不需要进行转义。但是语法相对严格,如在 [ 中可以用引号括起操作符,[[ 则不行。如:if [ "-z" "ab" ]。
[root@abc-1 ~]# [[ 6>A ]] && echo Y ||echo N
-bash: unexpected token 284 in conditional command
-bash: syntax error near `6>'
[root@abc-1 ~]# [[ 6 > A ]] && echo Y ||echo N
N
[root@abc-1 ~]# [ 6 > A ] && echo Y ||echo N
Y
解释,数字的ASCCII应该小于大写字母,使用[[]]的结果是正确的。
区别三:
[[ 可以做算术扩展,[ 则不行。
区别四:
在[ ]中==是字符匹配,在[[ ]]中是模式匹配
区别五:
[ ]不支持正则匹配,[[ ]]支持用=~进行正则匹配
[root@abc-1 ~]# var="hello"
[root@abc-1 ~]# [ $var =~ "ll" ] && echo Y || echo N
-bash: [: =~: binary operator expected
N
[root@abc-1 ~]# [[ $var =~ "ll" ]] && echo Y || echo N
Y
解释,[[ ]]判断变量var的值是否包含字符串ll,[]报错,[[]]输出正确
区别六:
[ ]中如果变量没有定义,那么需用双引号引起来,[[ ]]中不需要
总结:
| [[]] | [] |
|---|---|
| < 排序比较 | 不支持(仅部分Shell解释器支持<) |
| > 排序比较 | 不支持(仅部分Shell解释器支持>) |
| && 逻辑与 | -a 逻辑与 |
| || 逻辑或 | -o 逻辑或 |
| == 模式匹配 | ==字符匹配 |
| =~正则匹配 | 不支持 |
| ()分组测试 | 不支持(仅部分Shell解释器支持()) |
实践
(1)暂无
参考:
https/blog.csdn.net/RtxTitanV/article/details/115007727
https://www.runoob.com/linux/linux-shell-basic-operators.html
Linux扩展篇-shell编程(三)-shell运算符的更多相关文章
- Linux入门篇(六)——Shell(二)
这一系列的Linux入门都是本人在<鸟哥的Linux私房菜>的基础上总结的基本内容,主要是记录下自己的学习过程,也方便大家简要的了解 Linux Distribution是Ubuntu而不 ...
- 【Shell编程】Shell程序设计
1.Shell简介 作为Linux灵感来源的Unix系统最初是没有图形化界面的,所有的任务都是通过命令行来实现的.因此,Unix的命令行系统得到了很大的发展,逐步成为一个功能强大的系统. Sh ...
- Linux入门篇(五)——Shell(一)
这一系列的Linux入门都是本人在<鸟哥的Linux私房菜>的基础上总结的基本内容,主要是记录下自己的学习过程,也方便大家简要的了解 Linux Distribution是Ubuntu而不 ...
- Linux shell编程02 shell程序的执行 及文件权限
第一个shell脚本 1. shell编程的方式 交互式shell编程 非交互式shell编程:执行的语句存放到一个文件 shell脚本:可以任意文件名,建议扩展名为sh 2. ...
- 书写优雅的shell脚本(三) - shell中exec解析
参考:<linux命令.编辑器与shell编程> <unix环境高级编程> exec和source都属于bash内部命令(builtins commands),在bash下输入 ...
- shell编程01—shell基础
01.学习shell编程需要的知识储备 1.vi.vim编辑器的命令,vimrc设置 2.命令基础,100多个命令 3.基础.高端的网络服务,nfs,rsync,inotify,lanmp,sersy ...
- Linux Shell编程三
case分支条件语句. case "string" in pattern_1) commands ;; pattern_2) commands ;; *) commands ;; ...
- linux shell编程(三) if 和 for
if 条件判断: 单分支的if语句if 判断条件: then statement1fi双分支的if语句if 判断条件;then statement1 statementelse statement3f ...
- shell编程(三)之条件判断(if语句)
练习:写一个脚本判断当前系统上是否有用户的默认shell为bash: 如果有,就显示有多少个这类用户:否则,就显示没有这类用户: #!/bin/bash # grep '\<bash$' /et ...
- Linux shell编程 4 ---- shell中的循环
1 for循环 1 for语句的结构 for variable in values; do statement done 2 for循环通常是用来处理一组值,这组值可以是任意的字符串的集合 3 for ...
随机推荐
- 探索Kimi智能助手:如何用超长文本解锁高效信息处理新境界
目前,Kimi备受瞩目,不仅在社交平台上引起了广泛关注,而且在解决我们的实际问题方面也显示出了巨大潜力.其支持超长文本的特性使得我们能够更加灵活地配置信息,避免了频繁与向量数据库进行交互以及编写提示词 ...
- Serverless Kubernetes:理想,现实与未来
简介: 当前 Serverless 容器的行业趋势如何?有哪些应用价值?如果 Kubernetes 天生长在云上,它的架构应该如何设计?Serverless 容器需要哪些基础设施?阿里云容器服务产品负 ...
- 贝壳基于 Flink 的实时计算演进之路
简介: 贝壳找房在实时计算之路上的平台建设以及实时数仓应用. 摘要:贝壳找房大数据平台实时计算负责人刘力云带来的分享内容是贝壳找房的实时计算演进之路,内容如下: 发展历程 平台建设 实时数仓及其应用场 ...
- dotnet 修复 Uno 中文乱码
这是一个历史问题,在使用 Uno 展示中文的时候,如果设置 Uno 的底层使用 Skia 系进行渲染,那么将会因为中文字体问题,导致渲染出现乱码.此问题已被我修复,最佳解法是更新到最新版本 在上一篇博 ...
- UWP 从文件 StorageFile 转 SoftwareBitmap 图片方法
本文告诉大家如何在 UWP 从 文件 StorageFile 转 SoftwareBitmap 图片的方法 使用以下三步即可从文件 StorageFile 转 SoftwareBitmap 图片 第一 ...
- 4.k8s-配置网络策略 NetworkPolicy
一.基本了解 官方文档:https://kubernetes.io/zh-cn/docs/concepts/services-networking/network-policies/基本了解: 1.网 ...
- 实验8 #第8章 Verilog有限状态机设计-3 #Verilog #Quartus #modelsim
3. 状态机A/D采样控制电路 3.1 目标:用状态机控制ADC0809实现数据采集. 3.2 ADC0809简介 (1)ADC0809是8位A/D转换器,片内有8路模拟开关,可控制8个 模拟量中 的 ...
- postman使用中问题汇总
当用postman来通过接口造数据时,读取参数化文件中身份证字段的值读取错误. 参数文件如下 选择参数文件后预览的数据如下 身份证号码全部变成了0000结尾的 解决方案: 需要将身份证号码用引号引起来 ...
- Solution Set - 线段树
A[洛谷P5787]无向图,每条边有一个出现时段,求每个单位时段图是否是二分图. B[洛谷P5227]无向图,询问删除一个边集后是否连通. C[洛谷P3733]连通无向图,边有权,加边.修改(增加的边 ...
- blazor优雅的方式导入组件相关的js脚本
基本的组件导入方式为: 1 await JsRuntime.InvokeVoidAsync("import", $"XXXXX.js"); 优雅的组件导入方式: ...