bash千万不要这样"<cmd> | while read X Y Z; do ... done"使用

bash千万不要这样"<cmd> | while read X Y Z; do ... done"使用,第1张

bash千万不要这样"<cmd> | while read X Y X; do done"使用

因为在bash里面”|"后命令是在一个新的shell里面运行的,即“|”前后不在一个进程里面,所以如果在while循环里面使用了变量之类的赋值,就会无效。

举个例子:

运行结果:

看到在while循环内尽管已经把VAR置成了BBB,但是在循环退出后,VAR的值还是回退到AAA,也就是循环内变量赋值无效。

为什么呢,因为while循环实在一个独立的进程空间里面运行的,在那个进程内把VAR改成了BBB,但是并不影响在外面主进程的空间。

我们把PID打出来看看:

运行:

貌似PID也一样啊,不是吗?

不是的,因为打印PID的用法错误了,"$$"在脚本启动的时候就解析完了,全部替换成了当前进程的PID所以都一样;我们改一下:

在运行:

这里就可以看出while循环内的PID和循环外的PID是不一样的。

那应该怎么用呢:

答:

例:

得到运行结果:

bash的while循环变量在循环外无效的问题

如下例子:

假设文件1txt内容为:

运行结果:

变量NUM在循环体内累加,但是在循环结束的时候NUM的值丢了,也就是说循环结果没有保留下来。为什么呢,因为while循环在一个新的shell进程里面运行的,循环内的NUM和外面的NUM不是一个变量。

解决办法:

运行结果:

如果是命令输出的格式:

运行结果:

可见后面一种才是我们想要的结果。

前言:在linux中,Bash脚本是很基础的知识,大家可能一听脚本感觉很高大上,像我当初刚开始学一样,感觉会写脚本的都是大神。虽然复杂的脚本是很烧脑,但是,当我们熟练的掌握了其中的用法与技巧,再多加练习,总有一天也会成为得心应手的脚本大神。脚本在生产中的作用,想必我我不说,大家也都知道,脚本写的6,可以省下很多复杂的 *** 作,减轻自己的工作压力。好了,废话不多说,接下来,就是Bash脚本的用法展示。

(1)用法格式

if 判断条件 1 ; then

条件为真的分支代码

elif 判断条件 2 ; then

条件为真的分支代码

elif 判断条件 3 ; then

条件为真的分支代码

else

以上条件都为假的分支代码

fi

逐条件进行判断,第一次遇为“真”条件时,执行其分支,而后结束整个if。

(2)经典案例:

① 判断年纪

分析:请输入年纪,先判断输入的是否含有除数字以外的字符,有,就报错;没有,继续判断是否小于150,是否大于18。

② 判断分数

分析:请输入成绩,先判断输入的是否含有除数字以外的字符,有,就报错;没有,继续判断是否大于100,是否大于85,是否大于60。

(1)用法格式

case $name in;

PART1)

cmd

;;

PART2)

cmd

;;

)

cmd

;;

esac

注意:case 支持glob 风格的通配符:

: 任意长度任意字符

: 任意单个字符

[] :指定范围内的任意单个字符

a|b: a 或b

(2)案例:

判断yes or no

分析:请输入yes or no,回答Y/y、yes各种大小写组合为yes;回答N/n、No各种大小写组合为no。

(1)用法格式

① for name in 列表 ;do

循环体

done

for (( exp1; exp2; exp3 )) ;do

cmd

done

exp1只执行一次,相当于在for里嵌了while

③ 执行机制:依次将列表中的元素赋值给“变量名”; 每次赋值后即执行一次循环体; 直到列表中的元素耗尽,循环结束

列表的表示方法,可以glob 通配符,如{110} 、sh ;也可以变量引用,如: `seq 1 $name`

(2)案例

① 求出(1+2++n)的总和

分析:sum初始值为0,请输入一个数,先判断输入的是否含有除数字以外的字符,有,就报错;没有判断是否为0,不为0进入for循环,i的范围为1~输入的数,每次的循环为sum=sum+i,循环结束,最后输出sum的值。

② 求出(1+2++100)的总和

分析:i=1,num=0;当i<=100,进入循环,若i 2取余=1,则sum=sum+i,i=i+1。

(1)用法格式

while 循环控制条件 ;do

循环

done

循环控制条件;进入循环之前,先做一次判断;每一次循环之后会再次做判断;条件为“true” ,则执行一次循环;直到条件测试状态为“false” 终止循环

(2)特殊用法(遍历文件的每一行):while read line; do控制变量初始化循环体done < /PATH/FROM/SOMEFILE或 cat /PATH/FROM/SOMEFILE | while read line; do循环体done依次读取/PATH/FROM/SOMEFILE文件中的每一行,且将行赋值给变量line

(3)案例:

① 100以内所有正奇数之和

分析:sum初始值为0,i的初始值为1;请输入一个数,先判断输入的是否含有除数字以外的字符,有,就报错;没有当i<100时,进入循环,判断 i 2取余 是否不为0,不为0时为奇数,sum=sum+i,i+1,为0,i+1;循环结束,最后输出sum的值。

(1)用法

unitl 循环条件 ;do

循环

done

进入条件:循环条件为true ;退出条件:循环条件为false;刚好 和while相反 ,所以不常用,用while就行。

(2)案例

监控xiaoming用户,登录就杀死

分析:每隔05秒扫描,直到发现xiaoming用户登录,杀死这个进程,退出脚本,用于监控用户登录。

(1)用法

select variable in list

do

循环体命令

done

① select 循环主要用于创建菜单,按数字顺序排列的示菜单项将显示在标准错误上,并显示PS3 提示符,等待用户输入

② 用户输入菜单列表中的某个数字,执行相应的命令

③ 用户输入被保存在内置变量 REPLY 中

④ select 是个无限循环,因此要记住用 break 命令退出循环,或用 exit 按 命令终止脚本。也可以按 ctrl+c退出循环

⑤ select 和 经常和 case 联合使用

⑥ 与for循环类似,可以省略 in list, 此时使用位置参量

(2)案例:

生成菜单,并显示选中的价钱

分析:PS3是select的提示符,自动生成菜单,选择5break退出循环。

(1)语法

continue [N]:提前结束 第N层的本轮 循环,而直接进入下一轮判断;最内层为第1层

break [N]:提前结束 第N层 循环,最内侧为第1层

例:while CONDTITON1; doCMD1if CONDITION2; thencontinue / breakfiCMD2done (2)案例:

① 求(1+3++49+53++100)的和

分析:做1+2++100的循环,当i=51时,跳过这次循环,但是继续整个循环,结果为:sum=2449

② 求(1+3++49)的和

分析:做1+2++100的循环,当i=51时,跳出整个循环,结果为:sum=625

(1)作用

用于将参数列表list左移指定次数,最左端的那个参数就从列表中删除,其后边的参数继续进入循环

(2)案例:

① 创建指定的多个用户

分析:如果没有输入参数(参数的总数为0),提示错误并退出;反之,进入循环;若第一个参数不为空字符,则创建以第一个参数为名的用户,并移除第一个参数,将紧跟的参数左移作为第一个参数,直到没有第一个参数,退出。

② 打印直角三角形的字符

true 永远返回成功结果

: null command ,什么也不干,返回成功结果

false 永远返回错误结果

创建无限循环

while true ;do

循环体

done

(1)用法

for name in 列表 ;do

{

循环体

}&

done

wait

(2)实例:

搜寻自己指定ip(子网掩码为24的)的网段中,UP的ip地址

分析:请输入一个IP地址例19216837234,如果格式不是0000 则报错退出;正确则进入循环,IP变量的值为19216837 i的范围为1-254,并行ping 192168371-154,ping通就输出此IP为UP。直到循环结束。

trap ' 触发指令' 信号,自定义进程收到系统发出的指定信号后,将执行触发指令,而不会执行原 *** 作

trap '' 信号,忽略信号的 *** 作

trap '-' 信号,恢复原信号的 *** 作

trap -p ,列出自定义信号 *** 作

信号可以3种表达方法:信号的数字2、全名SIGINT、缩写INT

1) SIGHUP: 无须关闭进程而让其重读配置文件

2) SIGINT: 中止正在运行的进程;相当于Ctrl+c

3) SIGQUIT: 相当于ctrl+

9) SIGKILL: 强制杀死正在运行的进程

15) SIGTERM :终止正在运行的进程(默认为15)

18) SIGCONT :继续运行

19) SIGSTOP :后台休眠

9 信号,强制杀死,捕获不住

① 打印0-9,ctrl+c不能终止

分析:i=0,当i<10,每休眠1秒,i+1,捕获2信号,并执行echo press ctrl+c

② 打印0-3,ctrl+c不能终止,3之后恢复,能终止

分析:i=0,当i3时,解除捕获2信号。

1、生成随机字符 cat /dev/urandom

生成8个随机大小写字母或数字 cat /dev/urandom |tr -dc [:alnum:] |head -c 8

2、生成随机数 echo $RANDOM

确定范围 echo $[RANDOM%7] 随机7个数(0-6)

echo $[$[RANDOM%7]+31] 随机7个数(31-37)

3、echo打印颜色字

echo -e "33[31malong33[0m" 显示红色along

echo -e "33[1;31malong33[0m" 高亮显示红色along

echo -e "33[41malong33[0m" 显示背景色为红色的along

echo -e "33[31;5malong33[0m" 显示闪烁的红色along

color=$[$[RANDOM%7]+31]

echo -ne "33[1;${color};5m33[0m" 显示闪烁的随机色along

以上就是关于bash千万不要这样"<cmd> | while read X Y Z; do ... done"使用全部的内容,包括:bash千万不要这样"<cmd> | while read X Y Z; do ... done"使用、bash的while循环变量在循环外无效的问题、Bash 脚本进阶,经典用法及其案例等相关内容解答,如果想了解更多相关内容,可以关注我们,你们的支持是我们更新的动力!

欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/zz/9849072.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2023-05-02
下一篇 2023-05-02

发表评论

登录后才能评论

评论列表(0条)

保存