关于 “2015-07-17shell脚本练习题” 的疑问与测试

回复 收藏
题目内容:

脚本可以带参数也可以不带,参数可以有多个,每个参数必须是一个目录,脚本检查参数个数,若等于0,则列出当前目录本身;否则,显示每个参数包含的子目录。

铭哥的脚本:
  1. #!/bin/bash
  2. if [ $# == 0 ];then
  3.      ls -ld `pwd`
  4. else
  5.      for i in `seq 1 $#`
  6.      do
  7.          a=$i
  8.          echo "ls ${!a}"
  9.          ls -l ${!a} |grep '^d'
  10.      done
  11. fi
我的测试:
   从“ ls -l ${!a} |grep '^d'”看,${!a} 应该是传入的参数目录。但是测试结果:
因为$#会获取目录的个数,a=$i ,则使a 的值为一个数值,假使a=1;
  1. [root@master shell]# a=1
  2. [root@master shell]# echo ${!a}
  3. [root@master shell]# i=1;a=$i;echo ${!a}
输出的${!a}为空。

使用ls命令测试;可以列出目录下的内容:
  1. [root@master shell]# i=1;a=$i;ll ${!a}
  2. 总用量 4
  3. -rw-r--r-- 1 root root  584 11月 23 16:37 1.sh
  4. -rw-r--r-- 1 root root   33 11月 23 21:16 1.txt
  5. -rw-r--r-- 1 root root 1546 11月 23 21:58 adduser.sh
  6. -rw-r--r-- 1 root root   12 11月  3 19:37 arg.txt
  7. [root@master shell]# i=2;a=$i;ll ${!a}
  8. 总用量 4
  9. -rw-r--r-- 1 root root  584 11月 23 16:37 1.sh
  10. -rw-r--r-- 1 root root   33 11月 23 21:16 1.txt
  11. -rw-r--r-- 1 root root 1546 11月 23 21:58 adduser.sh
  12. -rw-r--r-- 1 root root   12 11月  3 19:37 arg.txt
但问题是无论 i循环时获取值为何,都是列出本目录下的目录与“ll”命令单独使用没有区别。。

但现实是,脚本可以正常运行,列出各个参数下的目录:
  1. [root@master shell]# sh -x dir.sh /root /tmp
  2. + '[' 2 == 0 ']'
  3. ++ seq 1 2
  4. + for i in '`seq 1 $#`'
  5. + a=1
  6. + echo 'ls /root'   
  7. ls /root                ## 列出的是/root
  8. + ls -l /root
  9. + grep '^d'
  10. drwxr-xr-x  3 root root    4096 11月  3 21:14 jpg
  11. drwxr-xr-x  4 root root    4096 11月 18 19:33 patch
  12. drwxr-xr-x  2 root root    4096 11月 24 01:02 shell
  13. drwxr-xr-x  2 root root    4096 11月 20 11:12 test1
  14. drwxr-xr-x  2 root root    4096 11月 20 11:13 test2
  15. + for i in '`seq 1 $#`'
  16. + a=2
  17. + echo 'ls /tmp'
  18. ls /tmp                ## 列出的是/tmp
  19. + ls -l /tmp
  20. + grep '^d'
  21. drwx------ 2 root root   4096 11月 19 04:06 ssh-eeIRD33261
修改脚本如下:
  1. #!/bin/bash
  2. if [ $# == 0 ];then
  3.      ls -ld `pwd`
  4. else
  5.         pwd             ## 进入循环前加入pwd 打印目录
  6.      for i in `seq 1 $#`
  7.      do
  8.         pwd                ## 循环内加入pwd 打印目录
  9.          a=$i
  10.          echo "ls ${!a}"
  11.          ls -l ${!a} |grep '^d'
  12.      done
  13. fi
运行结果:目录没有改变
  1. [root@master shell]# sh dir.sh /root /tmp
  2. /root/shell
  3. /root/shell        ## /root/shell目录
  4. ls /root
  5. drwxr-xr-x  3 root root    4096 11月  3 21:14 jpg
  6. drwxr-xr-x  4 root root    4096 11月 18 19:33 patch
  7. drwxr-xr-x  2 root root    4096 11月 24 01:46 shell
  8. drwxr-xr-x  2 root root    4096 11月 20 11:12 test1
  9. drwxr-xr-x  2 root root    4096 11月 20 11:13 test2
  10. /root/shell        ## /root/shell目录
  11. ls /tmp
  12. drwx------ 2 root root   4096 11月 19 04:06 ssh-eeIRD33261
问题:
这个${!a} 是数组的用法,还是变量的用法,还是其他特性??








2015-12-10 20:22 举报
已邀请:
0

xueyongbo

赞同来自:

测试过程中的其他问题:
  1. [root@master shell]# a=(1 2 3)
  2. [root@master shell]# echo ${!a}

  3. [root@master shell]# echo ${a[*]}
  4. 1 2 3
  5. [root@master shell]# echo ${!a[*]}       ## 为何这里数组的所有元素都减1
  6. 0 1 2

赋一个变量,可以变量,可以数组。。

  1. [root@master shell]# b=/root/shell
  2. [root@master shell]# echo ${b}
  3. /root/shell
  4. [root@master shell]# echo ${b[*]}
  5. /root/shell
  6. [root@master shell]# echo ${b[0]}
  7. /root/shell
  8. [root@master shell]# echo ${b[3]}

  9. [root@master shell]# echo ${#b}
  10. 11
  11. [root@master shell]# echo ${b:2}
  12. oot/shell
  13. [root@master shell]# echo ${b[*]:2}
  14. oot/shell
  15. [root@master shell]# echo ${b[*]:2}
  16. oot/shell
  17. [root@master shell]# echo ${b[*]:2:2}
  18. oo

  19. 这两个看不懂。。
  20. [root@master shell]# echo ${!b}

  21. [root@master shell]# echo ${!b[*]}
  22. 0



0

xueyongbo

赞同来自:

本帖最后由 xueyongbo 于 2015-12-10 20:38 编辑
  1. shell中${}的妙用

  2. 1. 截断功能

  3. ${file#*/}:       拿掉第一条/及其左边的字符串:dir1/dir2/dir3/my.file.txt
  4. ${file##*/}:    拿掉最后一条/及其左边的字符串:my.file.txt
  5. ${file#*.}:       拿掉第一个.及其左边的字符串:file.txt
  6. ${file##*.}:    拿掉最后一个.及其左边的字符串:txt
  7. ${file%/*}:     拿掉最后条/及其右边的字符串:/dir1/dir2/dir3
  8. ${file%%/*}: 拿掉第一条/及其右边的字符串:(空值)
  9. ${file%.*}:    拿掉最后一个.及其右边的字符串:/dir1/dir2/dir3/my.file
  10. ${file%%.*}: 拿掉第一个.及其右边的字符串:/dir1/dir2/dir3/my

  11. 记忆的方法为:
  12. [list]#是去掉左边, ##最后一个
  13.       %是去掉右边, %%第一个
  14.       单一符号是最小匹配﹔两个符号是最大匹配。

  15. 2. 字符串提取

  16. ${file:0:5}:提取最左边的 5 个字节:/dir1
  17. ${file:5:5}:提取第 5 个字节右边的连续 5 个字节:/dir2
  18. 3. 字符串替换
  19. ${file/dir/path}:将第一个 dir 提换为 path:/path1/dir2/dir3/my.file.txt
  20. ${file//dir/path}:将全部 dir 提换为 path:/path1/path2/path3/my.file.txt

  21. 4. 针对不同的变量状态赋值(没设定、空值、非空值):

  22. ${file-my.file.txt}: 若$file没有设定,则使用my.file.txt作返回值。(空值及非空值时不作处理)
  23. ${file:-my.file.txt}:若$file没有设定或为空值,则使用my.file.txt作返回值。(非空值时不作处理)
  24. ${file+my.file.txt}: 若$file设为空值或非空值,均使用my.file.txt作返回值。(没设定时不作处理)
  25. ${file:+my.file.txt}:若$file为非空值,则使用my.file.txt作返回值。(没设定及空值时不作处理)
  26. ${file=my.file.txt}: 若$file没设定,则使用my.file.txt作返回值,同时将$file 赋值为 my.file.txt。(空值及非空值时不作处理)
  27. ${file:=my.file.txt}:若$file没设定或为空值,则使用my.file.txt作返回值,同时将 $file 赋值为 my.file.txt。(非空值时不作处理)
  28. ${file?my.file.txt}: 若$file没设定,则将my.file.txt输出至 STDERR。(空值及非空值时不作处理)
  29. ${file:?my.file.txt}:若$file没设定或为空值,则将my.file.txt输出至STDERR。(非空值时不作处理)

  30. 注意:
  31. ":+"的情况是不包含空值的.
  32. ":-", ":="等只要有号就是包含空值(null).

  33. 5. 变量的长度
  34. ${#file}

  35. 6. 数组运算
  36. A=(a b c def)
  37. ${A[@]} 或 ${A[*]} 可得到 a b c def (全部组数)
  38. ${A[0]} 可得到 a (第一个组数),${A[1]} 则为第二个组数...
  39. ${#A[@]} 或 ${#A[*]} 可得到 4 (全部组数数量)
  40. ${#A[0]} 可得到 1 (即第一个组数(a)的长度),${#A[3]} 可得到 3 (第四个组数(def)的长度)
来源:http://www.2cto.com/os/201310/248691.html

回复帖子,请先登录注册

退出全屏模式 全屏模式 回复
评分
可选评分理由: