shell 关于 while 循环中赋值的问题

本文探讨了三种Shell脚本程序处理文件数据的不同方式,并详细分析了为何在使用管道时,变量更新无法在父进程生效的原因。通过具体的代码示例和运行结果对比,帮助读者理解Shell脚本中变量的作用域及正确使用方法。

程序1

#!/bin/sh
        count=0
        cat  temp.txt|while read fre
        do

                count=$(($count+$fre))
        done
        echo "$count"
exit 1

程序2

#!/bin/sh
        count=0
        exec 3< temp.txt
        while read fre <&3
        do

                count=$(($count+$fre))
        done
        echo "$count"
exit 1

程序3

#!/bin/sh
        count=0
        while read fre
        do

                count=$(($count+$fre))
        done<temp.txt
        echo "$count"
exit 1

 

temp.txt文件

2

2

 

 

在运行以上3个程序以后我们会发现

1程序输出结果是0

2,3的输出结果为正确的4

 

原因是因为程序1使用了管道,管道中的while是在子shell中运行的,并不能返回到父shell中

所以count的值成为了初始化时的值

 

Shell 脚本中,循环变量的作用域行为可能会根据使用的 Shell 类型(如 Bash、Zsh 或其他)以及脚本的结构而有所不同。通常情况下,在 `for`、`while` 或 `until` 循环中声明的变量具有全局作用域,这意味着它们可以在循环外部访问。 ### Bash 中的 `for` 循环 在 Bash 中,`for` 循环的变量是全局的,并且在循环结束后仍然存在: ```bash for i in {1..3}; do echo "Inside loop: $i" done echo "Outside loop: $i" # 变量 i 依然可访问 ``` 输出结果: ``` Inside loop: 1 Inside loop: 2 Inside loop: 3 Outside loop: 3 ``` 在这个例子中,变量 `i` 在循环结束后仍然保留其值 `3`,这表明它不是局部变量 [^1]。 ### Bash 中的 `while` 循环 对于 `while` 循环,如果使用管道或子进程来提供输入,则循环内部的变量可能运行在子 shell 中,因此不会影响外部的变量: ```bash var=0 echo "Start value: $var" echo "a b c" | while read -r word; do var=1 echo "Inside loop: $var" done echo "End value: $var" # 输出仍然是 0 ``` 输出结果: ``` Start value: 0 Inside loop: 1 End value: 0 ``` 由于 `while` 循环运行在子 shell 中,变量 `var` 的修改不会传播到父 shell 。 ### 使用 `local` 关键字 如果你希望在函数中限制变量的作用域,可以使用 `local` 关键字来定义局部变量: ```bash my_function() { for local_var in {1..3}; do local inner_var="local_value" echo "Inside loop, inner_var: $inner_var" done echo "After loop, inner_var: $inner_var" # inner_var 仍然存在 } my_function echo "Outside function, local_var: $local_var" # local_var 不可见 ``` 在这个例子中,`inner_var` 是一个局部变量,但它仍然可以在循环之外访问,因为 `local` 关键字仅对函数作用域有效 [^1]。 ### 子 shell 和作用域 某些构造(如命令替换 `$(...)` 或管道 `|`)会创建子 shell,其中的变量更改不会影响父 shell: ```bash var=0 var=$(echo 1) echo "Value after command substitution: $var" # 输出 1 echo "a b c" | while read -r word; do var=2 done echo "Value after pipe: $var" # 输出仍然是 1 ``` 在这个例子中,`var` 的值在命令替换后变为 `1`,但在 `while` 循环中修改为 `2` 后,这个变化没有传递到父 shell 。 ### 总结 Shell 脚本中的变量作用域主要取决于上下文和 Shell 的实现方式。为了确保变量不会意外覆盖或泄露,建议使用 `local` 关键字来定义函数内的局部变量,并注意避免在子 shell 中修改全局状态。
评论 1
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值