Bash脚本进阶指南
Repo
  • 正文
    • 第一部分 初见shell
      • 1. 为什么使用shell编程
      • 2. 和Sha-Bang(#!)一起出发
        • 2.1 调用一个脚本
        • 2.2 牛刀小试
    • 第二部分 shell基础
      • 3. 特殊字符
      • 4. 变量与参数
        • 4.1 变量替换
        • 4.2 变量赋值
        • 4.3 Bash弱类型变量
        • 4.4 特殊变量类型
      • 5. 引用
        • 5.1 引用变量
        • 5.2 转义
      • 6. 退出与退出状态
      • 7. 测试
        • 7.1 测试结构
        • 7.2 文件测试操作
        • 7.3 其他比较操作
        • 7.4 嵌套 if/then 条件测试
        • 7.5 牛刀小试
      • 8. 运算符相关话题
        • 8.1 运算符
        • 8.2 数字常量
        • 8.3 双圆括号结构
        • 8.4 运算符优先级
    • 第三部分 shell进阶
      • 9. 换个角度看变量
        • 9.1 内部变量
        • 9.2 变量类型标注:declare 与 typeset
          • 9.2.1 declare 的另类用法
        • 9.3 $RANDOM:生成随机数
      • 10. 变量处理
        • 10.1 字符串处理
          • 10.1.1 使用 awk 处理字符串
          • 10.1.2 参考资料
        • 10.2 参数替换
      • 11. 循环与分支
        • 11.1 循环
        • 11.2 嵌套循环
        • 11.3 循环控制
        • 11.4 测试与分支
      • 12. 命令替换
      • 13. 算术扩展
      • 14. 休息时间
    • 第四部分 命令
      • 15. 内建命令
        • 15.1 任务控制命令
      • 16. 外部筛选器,任务及命令
        • 16.1 基础命令
        • 16.2 复杂命令
        • 16.3 时间/日期命令
        • 16.4 文本处理命令
        • 16.5 文件与归档命令
        • 16.6 通信命令
        • 16.7 终端控制命令
        • 16.8 数学命令
        • 16.9 杂项命令
      • 17. 系统与管理命令
        • 17.1 分析一个系统脚本
    • 第五部分 高级话题
      • 18.正则表达式
        • 18.1正则表达式简介
        • 18.2文件名替换
      • 19. 嵌入文档
      • 20. I/O 重定向
        • 20.1 使用 exec
        • 20.2 重定向代码块
        • 20.3 应用程序
      • 21. 子shell
      • 22. 限制模式的Shell
      • 23. 进程替换
      • 24. 函数
        • 24.1 复杂函数和函数复杂性
        • 24.2 局部变量
        • 24.3 不适用局部变量的递归
      • 25. 别名
      • 26. 列表结构
      • 27. 数组
      • 28. 间接引用
      • 29. /dev 和 /proc
        • 29.1 /dev
        • 29.2 /proc
      • 30. 网络编程
      • 32. 调试
      • 33. 选项
      • 34. 陷阱
      • 36. 杂项
        • 36.1 交互和非交互shell以及脚本
        • 36.2 shell wrappers
        • 36.3 测试和比较的其他方法
        • 36.4 递归:调用自己的脚本
        • 36.5 “彩色”的脚本
        • 36.6 优化
        • 36.7 其他技巧
        • 36.8 安全问题
        • 36.9 可移植性问题
        • 36.10 Windows系统下的脚本
    • 38. 后记
      • 38.1 作者后记
      • 38.2 关于作者
      • 38.3 从哪里可以获得帮助
      • 38.4 用来制作这本书的工具
      • 38.5 致谢
      • 38.6 免责声明
  • 附录及索引
    • 参考文献
    • 附录
    • 索引
由 GitBook 提供支持
在本页

这有帮助吗?

  1. 正文
  2. 第五部分 高级话题

28. 间接引用

上一页27. 数组下一页29. /dev 和 /proc

最后更新于5年前

这有帮助吗?

我们已经看到,$var,获取其值。但是值的值又如何呢?可以是 $$var 吗?

真正的表达式是 \$$var,一般前面有 eval(有时候是 echo)。这被称作间接引用。

例 28-1. 间接变量引用

#!/bin/bash
# ind-ref.sh: 间接变量引用。
# 获取一个变量内容的内容。

# 首先让我们随便玩玩。

var=23

echo "\$var   = $var"           # $var   = 23
# 到现在为止,一切都在预料中。但是……

echo "\$\$var  = $$var"         # $$var  = 4570var
#  没用
#  \$\$ 展开为脚本的 PID
#  -- 引用了 $$ 变量的条目 --
#+ 并且“var”作为文本打印了出来。
#  (感谢你,Jakob Bohm,指出了这一点。)

echo "\\\$\$var = \$$var"       # \$$var = $23
#  符合预期。第一个 $ 被转义,传给了 var 的值($var = 23)。
#  有意思了,但还不能用。

# 现在,让我们开始做正确的事情。

# ============================================== #


a=letter_of_alphabet   # 变量“a”保存了另一个变量的名字。
letter_of_alphabet=z

echo

# 直接引用。
echo "a = $a"          # a = letter_of_alphabet

# 间接引用。
  eval a=\$$a
# ^^^        强制运算(evaluation),并且……
#        ^   转义第一个 $ ……
# ------------------------------------------------------------------------
# “eval”强制更新了 $a,将其设置为更新后的 \$$a 的值。
# 所以,我们看到为什么“eval”如此频繁地出现在间接引用的表达式中了。
# ------------------------------------------------------------------------
  echo "Now a = $a"    # 现在 a = z

echo


# 现在,让我们尝试改变第二个顺序的引用。

t=table_cell_3
table_cell_3=24
echo "\"table_cell_3\" = $table_cell_3"            # "table_cell_3" = 24
echo -n "dereferenced \"t\" = "; eval echo \$$t    # dereferenced "t" = 24
# 在这个简单的情况中,下面的语句也可以(为什么?)。
#         eval t=\$$t; echo "\"t\" = $t"

echo

t=table_cell_3
NEW_VAL=387
table_cell_3=$NEW_VAL
echo "Changing value of \"table_cell_3\" to $NEW_VAL."
echo "\"table_cell_3\" now $table_cell_3"
echo -n "dereferenced \"t\" now "; eval echo \$$t
# “eval”获取两个参数“echo”和“\$$t”(设为等于 $table_cell_3)

echo

# (Thanks, Stephane Chazelas, for clearing up the above behavior.)


#   更直接的方法是 ${!t} 表示法,在 “Bash,版本 2” 一节中讨论。
#   另请参考 ex78.sh。

exit 0

Bash 中的间接引用是一个多步骤过程。首先,获取变量名称:varname。然后,引用它:$varname。然后,引用这个引用:$$varname。然后,转义第一个 $:\$$varname。最后,强制重新计算表达式并赋值:eval newvar=\$$varname。

间接引用变量有什么实际用途?它给了 Bash 一些 C 语言中指针的一点功能,例如,表格查询。此外,它也有其他的一些非常有趣的应用……

Nils Radtke 展示了如何构建“动态”变量名称并计算他们的内容。这在 source 配置文件的时候会有用处。

#!/bin/bash


# ---------------------------------------------
# 这部分可以从独立的文件中“source”。
isdnMyProviderRemoteNet=172.16.0.100
isdnYourProviderRemoteNet=10.0.0.10
isdnOnlineService="MyProvider"
# ---------------------------------------------
      

remoteNet=$(eval "echo \$$(echo isdn${isdnOnlineService}RemoteNet)")
remoteNet=$(eval "echo \$$(echo isdnMyProviderRemoteNet)")
remoteNet=$(eval "echo \$isdnMyProviderRemoteNet")
remoteNet=$(eval "echo $isdnMyProviderRemoteNet")

echo "$remoteNet"    # 172.16.0.100

# ================================================================

#  更好了。

#  考虑下面这个片段,给了变量 getSparc,但没有变量 getIa64:

chkMirrorArchs () { 
  arch="$1";
  if [ "$(eval "echo \${$(echo get$(echo -ne $arch |
       sed 's/^\(.\).*/\1/g' | tr 'a-z' 'A-Z'; echo $arch |
       sed 's/^.\(.*\)/\1/g')):-false}")" = true ]
  then
     return 0;
  else
     return 1;
  fi;
}

getSparc="true"
unset getIa64
chkMirrorArchs sparc
echo $?        # 0
               # True

chkMirrorArchs Ia64
echo $?        # 1
               # False

# 注:
# -----
# 即使是要被替换的变量名称部分也是独立构建的。
# 给 chkMirrorArchs 调用的参数都是小写。
# 变量名称由两部分组成:“get”和“Sparc”……

例 28-2. 传递间接引用给 awk

#!/bin/bash

#  “列求和”脚本的另一个版本,它将目标文件中指定列的数字累加。
#  这个版本使用了间接引用。
ARGS=2
E_WRONGARGS=85

if [ $# -ne "$ARGS" ] # 检查命令行参数的数量是否正确
then
   echo "Usage: `basename $0` filename column-number"
   exit $E_WRONGARGS
fi

filename=$1         # 操作的文件名
column_number=$2    # 对哪一列求和

#===== 到此为止与原脚本一样 =====#


# 多行 awk 脚本,由 awk 调用
#   awk "
#   ...
#   ...
#   ...
#   "


# awk 脚本开始。
# -------------------------------------------------
awk "

{ total += \$${column_number} # 间接引用
}
END {
     print total
     }

     " "$filename"
# 注意 awk 不需要在 \$$ 前置 eval。
# -------------------------------------------------
# awk 脚本结束。

#  间接变量引用避免了嵌入 awk 脚本内引用 shell 变量的麻烦。
#  感谢 Stephane Chazelas。


exit $?

Bash 不支持指针运算,这严重限制了间接引用的用处。实际上,在脚本语言中间接引用充其量只是事后的想法。

这个间接引用的方法有些小技巧。如果二级变量改变了值,那么必须正确地取消对一级变量的引用。所幸的是,版本 2 的 Bash 引入了 ${!variable} 表示法(参见 和)),让间接引用更直观。

引用变量
例 37-2
例 A-22
Caution