管道下while循环中定义的变量在退出while循环时引用为空的问题

最近在编写一个shell脚本的时候,在while循环中定义了一些变量,但是当while退出之后调用这些变量的时候发现,变量值都为空了,折腾了一整天,才找到其原因之所在:

我在这里只是大致做一个实验,说明下这里问题出在说明地方,开始的时候代码大意就是:

#!/bin/sh
cat config.ini | while read var
do
a=$var
echo $a
done
echo "$a mark"
[[email protected]_102 ~]# cat config.ini
test
[[email protected]_102 ~]# ./a.sh 
test
 mark

这个时候我们发现,在while循环中定义好的变量,为什么在while循环退出后变量值却为空了呢,会不会是局部变量的问题导致的呢,这里使用export试下:

[[email protected]_102 ~]# cat a.sh 
#!/bin/sh
cat config.ini | while read var
do
export a=$var
echo $a
done
echo "$a mark"
[[email protected]_102 ~]# ./a.sh
test
 mark

依然不行,于是开始了漫长的Google之旅,后来发现,这里是因为管道符重定向导致的,先把代码改下试试看:

[[email protected]_102 ~]# cat a.sh 
#!/bin/sh
while read var
do
export a=$var
echo $a
done < config.ini
echo "$a mark"
[[email protected]_102 ~]# ./a.sh 
test
test mark

成功!!!!!!!!

这里为什么会出现这种问题呢,其实原因很简单,当bash在使用管道符的时候,会产生subshell,从而使在这个subshell中对变量的赋值只在自身的subshell中生效,不会影响到shell脚本本身的进程。

但是ksh现在有个feature,能使管道中对变量的赋值反映到父shell中来,这就是网上有人问说这样的代码在k shell中就是OK的原因。

所以这里如果我们使用ksh的话就可以打印出变量的变量值:

[[email protected]_102 ~]# cat a.sh 
#!/bin/ksh
cat config.ini | while read var
do
export a=$var
echo $a
done
echo "$a mark"
[[email protected]_102 ~]# ./a.sh 
test
test mark
时间: 2025-01-01 08:43:11

管道下while循环中定义的变量在退出while循环时引用为空的问题的相关文章

【Javascript】: for循环中定义的变量在for循环体外也有效

for循环中定义的变量在for循环体外也有效 <script> (function(){ var a = 111; for(var i=0;i<5;i++){ var carl = 111; console.log(i); } alert("carl is:"+carl); })(); </script> 结果: carl is:111

bash的循环中无法保存变量

在bash中,如果循环在一个子shell中运行,那么在循环中对循环外面的变量的更改将在循环退出后不可见.像下面的例子: #!/bin/sh var=1 python run.py | while read line do echo $line var=var+1 let "var=var+1" done echo $var 最后面的echo输出的var仍然是1.原因是while循环位于管道中,这意味着在运行过程中,while循环实际是位于一个新的SHELL中的,while循环中的var

Objective-c中定义成员变量

ios定义成员变量的方法: 如果只是内部访问,完全可以直接用_xxx(下划线),无需自己定义,编译器会自动生成 如果需要外部访问这个变量,可以用 @property xxx ; @synthesize xxx = _xxx ,这样,编译器会自动生成 get,set函数 :可以很明确的表明这是一个向外部提供set.get方法的属性. http://www.tairan.com/archives/194 Objective-c中定义成员变量

ES6 class类中定义私有变量

ES6 class类中定义私有变量 class类的不足 看起来, es6 中 class 的出现拉近了 JS 和传统 OOP 语言的距离.但是,它仅仅是一个语法糖罢了,不能实现传统 OOP 语言一样的功能.在其中,比较大的一个痛点就是私有变量问题. 何为私有变量?私有变量就是只能在类内部访问的变量,外部无法访问的变量.在开发中,很多变量或方法你不想其他人访问,可以定义为私有变量,防止被其他人使用.在 Java 中,可以使用 private 实现私有变量,但是可惜的是, JS 中并没有该功能. 来

javascript在私有作用域中定义私有变量和私有函数 (1)

javascript没有私有成员,但是有私有变量,所有对戏那个属性都是公有的 任何在函数中定义的变量,都可以认为是私有变量,因为函数内部不能访问. 私有变量包括:函数的参数.局部变量.函数内部定义的其他函数 - 在私有作用域中定义私有变量和私有函数 function MyObject(){ //私有变量和私有函数 var privateVariable=10; function privateFunction(params) { return false } //特权方法----有权访问私有变量

能否向编译后得到的类中增加实例变量?能否向运行时创建的类中添加实例变量?为什么

不能向编译后得到的类中增加实例变量!能向运行时创建的类中添加实例变量! 因为编译后的类已经注册在runtime中,类结构体中的objc_ivar_list 实例变量的链表和instance_size实例变量的内存大小已经确定,同时runtime 会调用class_setIvarLayout 或 class_setWeakIvarLayout来处理strong weak引用,所以不能向存在的类中添加实例变量. 运行时创建的类是可以添加实例变量,调用 class_addIvar 函数,但是得在调用 

shell脚本中定义路径变量出现的BUG

=========================================================================== if 语句中的定义路径变量 引发命令的PATH路径改变出现的bug 问题 原脚本文件:inotify +rsync 实时共享 #!/bin/bash PATH=/data/ IP=172.16.1.41 cd $PATH &&\ 切换到一个新的变量中了 /usr/bin/inotifywait -mrq --format '%w%f' -e

java接口中定义成员变量

//抽象类中可以定义如下成员变量:public abstract class People { public String name; public int age; public abstract void work(); } 而如下截图中的People接口中,定义成员变量报错: 接口的含义理解:接口可以理解成统一的"协议",而接口中的属性也属于协议中的内容;但是接口的属性都是公共的,静态的,最终的 接口的成员特点: A:成员变量 只能是常量.默认修饰符 public static

不能用private修饰方法中定义的变量

在敲代码时发现了一个自己之前没有注意到点:(确切来说是没有认真理解权限修饰符)在方法中定义变量时,不能加权限修饰符原因如下: 原文地址:https://blog.51cto.com/14234228/2468546