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 脚本进阶,经典用法及其案例等相关内容解答,如果想了解更多相关内容,可以关注我们,你们的支持是我们更新的动力!
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)