《shell十三问》是网中人前辈首发在CU论坛上对SHELL的一些整理,非常值得一读


注:笔记的标号非问题标号,而是知识点的标号。本篇笔记记录的是1-10问的知识点

(1)IFS:Shell预设使用的栏位分隔符号,可以由一个几多个“空白键(White Space)、表格键(Tab)、回车键(Enter)”组成

(2)echo命令支持的常用选项和反斜线控制字符

  • -e:启用反斜线控制字符的转换(参考下表)
  • -E:关闭反斜线控制字符的转换(预设如此)
  • -n:取消行末之换行符号(与-e选项下的\c字符同意)

[toggle title="反斜线控制字符"]


\a:ALERT / BELL (从系统喇叭送出铃声)
\b:BACKSPACE ,也就是向左退格键
\c:取消行末之换行符号
\E:ESCAPE,跳脱键
\f:FORMFEED,换页字符
\n:NEWLINE,换行字符
\r:RETURN,回车键
\t:TAB,表格跳位键
\v:VERTICAL TAB,垂直表格跳位键
\n:ASCII 八进位编码(以 x 开首为十六进位)
  \\:反斜线本身


[/toggle]

(3)‘’单引号为Hard quote、“”双引号为Soft quote

  1. $ A=B\ C
  2. $ echo '"$A"' # 最外面的是单引号
  3. "$A"
  4. $ echo "'$A'" # 最外面的是双引号
  5. 'B C'

(提示:单引号及双引号,在 quoting 中均被关闭了。)

(4)var=${str=expr}的意思


首先,var=$str 这个大家都可理解吧。
而接下来的思考方向是,究竟 $str 这个变量是如下哪一种情况呢:
1) unset
2) null
3) not null

1) 假如是 unset(未定义) ,那麽 var=${str=expr} 的结果将是:
var=expr
str=expr

2) 假如是 null(空) ,那 var=${str=expr} 的结果是:
var=
str=

3) 假如是 not null (非空,比方为 xyz ),那 var=${str=expr} 之结果是:
var=xyz
str=xyz


(5)三种执行shell方法的区别(sh、source、exec)

sh:父进程会fork一个子进程,shell script在子进程中执行

source:在原进程中执行,不会fork子进程

exec:在原进程中执行,但是同时会终止原进程

注:使用export会把父进程中的变量向子进程中继承,但是反过来却不行,在子进程中,不管环境如果改变,均不会影响父进程

根据例子来更加清晰的理解一下执行的过程:

1.sh:

  1. #!/bin/bash
  2. A=B
  3. echo "PID for 1.sh before exec/source/fork:$$"
  4. export A
  5. echo "1.sh: \$A is $A"
  6. case $ in
  7. exec)
  8. echo "using exec..."
  9. exec ./.sh ;;
  10. source)
  11. echo "using source..."
  12. . ./.sh ;;
  13. *)
  14. echo "using fork by default..."
  15. ./.sh ;;
  16. esac
  17. echo "PID for 1.sh after exec/source/fork:$$"
  18. echo "1.sh: \$A is $A"

2.sh

  1. #!/bin/bash
  2. echo "PID for 2.sh: $$"
  3. echo "2.sh get \$A=$A from 1.sh"
  4. A=C
  5. export A
  6. echo "2.sh: \$A is $A"

再依次按如下依次执行查看结果

  1. $ ./.sh fork
  2. $ ./.sh source
  3. $ ./.sh exec

(如不能运行可能是你新建文件后没有给予权限)

运行:

  1. sudo chmod .sh .sh
  1.  
  1. [sincerefly@localhost ]$ ./.sh fork
  2. PID for .sh before exec/source/fork:
  3. .sh: $A is B
  4. using fork by default...
  5. PID for .sh:
  6. .sh get $A=B from .sh
  7. .sh: $A is C
  8. PID for .sh after exec/source/fork:
  9. .sh: $A is B
  10.  
  11. [sincerefly@localhost ]$ ./.sh source
  12. PID for .sh before exec/source/fork:
  13. .sh: $A is B
  14. using source...
  15. PID for .sh:
  16. .sh get $A=B from .sh
  17. .sh: $A is C
  18. PID for .sh after exec/source/fork:
  19. .sh: $A is C
  20.  
  21. [sincerefly@localhost ]$ ./.sh exec
  22. PID for .sh before exec/source/fork:
  23. .sh: $A is B
  24. using exec...
  25. PID for .sh:
  26. .sh get $A=B from .sh
  27. .sh: $A is C
  28. [sincerefly@localhost ]$

由输出可知,source和exec都是在同一进程中执行,但是需要注意,使用exec终止了原来的父进程,因此,可以看到后两条命令没有执行

(6)()与{}的区别

()将command group置于sub-shell去执行,也称为nested sub-shell
{}则是在同一个shell内完成,也称为non-named command group

通常而言,若所作的修改是临时的,且不想影响原有或以后的设定,那我们就 nested sub-shell ,
反之,则用 non-named command group

(7)bash中function的两种定义方式

  1. function function_name {
  2. command1
  3. command2
  4. command3
  5. ....
  6. }
  7.  
  8. fuction_name () {
  9. command1
  10. command2
  11. command3
  12. ....
  13. }

(8)$()的意思
在 bash shell 中,$( ) 与 ` ` (反引号) 都是用来做命令替换用的

  1. command1 `command2 \`command3\` `
  1. 等效于
  1. command1 $(command2 $(command3))

上面的写法更有通用性,下面的方式更清晰明了
一个$()的例子

  1. $ echo the last sunday is $(date -d "last sunday" +%Y-%m-%d)
  2. the last sunday is --

(9)${}的意思

${}常用来界定变量名的范围

还有更多的其它实用功能

为了完整起见,我这裡再用一些例子加以说明 ${ } 的一些特异功能:
假设我们定义了一个变量为:
file=/dir1/dir2/dir3/my.file.txt
我们可以用 ${ } 分别替换获得不同的值:
${file#*/}:拿掉第一条 / 及其左边的字串:dir1/dir2/dir3/my.file.txt
${file##*/}:拿掉最后一条 / 及其左边的字串:my.file.txt
${file#*.}:拿掉第一个 . 及其左边的字串:file.txt
${file##*.}:拿掉最后一个 . 及其左边的字串:txt
${file%/*}:拿掉最后条 / 及其右边的字串:/dir1/dir2/dir3
${file%%/*}:拿掉第一条 / 及其右边的字串:(空值)
${file%.*}:拿掉最后一个 . 及其右边的字串:/dir1/dir2/dir3/my.file
${file%%.*}:拿掉第一个 . 及其右边的字串:/dir1/dir2/dir3/my
记忆的方法为:

# 是去掉左边(在鑑盘上 # 在 $ 之左边)
% 是去掉右边(在鑑盘上 % 在 $ 之右边)
单一符号是最小匹配﹔两个符号是最大匹配。

${file:0:5}:提取最左边的 5 个字节:/dir1
${file:5:5}:提取第 5 个字节右边的连续 5 个字节:/dir2

我们也可以对变量值裡的字串作替换:
${file/dir/path}:将第一个 dir 提换为 path:/path1/dir2/dir3/my.file.txt
${file//dir/path}:将全部 dir 提换为 path:/path1/path2/path3/my.file.txt

利用 ${ } 还可针对不同的变数状态赋值(没设定、空值、非空值):
${file-my.file.txt} :假如 $file 没有设定,则使用 my.file.txt 作传回值。(空值及非空值时不作处理)
${file:-my.file.txt} :假如 $file 没有设定或为空值,则使用 my.file.txt 作传回值。 (非空值时不作处理)
${file+my.file.txt} :假如 $file 设为空值或非空值,均使用 my.file.txt 作传回值。(没设定时不作处理)
${file:+my.file.txt} :若 $file 为非空值,则使用 my.file.txt 作传回值。 (没设定及空值时不作处理)
${file=my.file.txt} :若 $file 没设定,则使用 my.file.txt 作传回值,同时将 $file 赋值为 my.file.txt 。 (空值及非空值时不作处理)
${file:=my.file.txt} :若 $file 没设定或为空值,则使用 my.file.txt 作传回值,同时将 $file 赋值为 my.file.txt 。 (非空值时不作处理)
${file?my.file.txt} :若 $file 没设定,则将 my.file.txt 输出至 STDERR。 (空值及非空值时不作处理)
${file:?my.file.txt} :若 $file 没设定或为空值,则将 my.file.txt 输出至 STDERR。 (非空值时不作处理)

tips:
以上的理解在于, 你一定要分清楚 unset 与 null 及 non-null 这三种赋值状态.
一般而言, : 与 null 有关, 若不带 : 的话, null 不受影响, 若带 : 则连 null 也受影响.

还有哦,${#var} 可计算出变量值的长度:
${#file} 可得到 27 ,因为 /dir1/dir2/dir3/my.file.txt 刚好是 27 个字节...

(10)${}处理数组
一般而言,A="a b c def" 这样的变量只是将 $A 替换为一个单一的字串,但是改为 A=(a b c def) ,则是将 $A 定义为组数...
bash 的组数替换方法可参考如下方法:

  1. ${A[@]} ${A[*]}

可得到 a b c def (全部组数)

  1. ${A[0]}

可得到 a (第一个组数),${A[1]} 则为第二个组数...

  1. ${#A[@]} ${#A[*]}

可得到 4 (全部组数数量)

  1. ${#A[0]}

可得到 1 (即第一个组数(a)的长度),${#A[3]} 可得到 3 (第四个组数(def)的长度)

  1. A[3]=xyz

则是将第四个组数重新定义为 xyz ...

(11)$(())的意思

它是用来作整数运算的。
在 bash 中,$(( )) 的整数运算符号大致有这些:
+ - * / :分别为 "加、减、乘、除"。
% :余数运算
& | ^ !:分别为 "AND、OR、XOR、NOT" 运算。

在$(())支持C语言中的自增自减操作,和比较判断,还支持2,8,16进制转换成十进制

  1. a=12
  2.  
  3. echo $((a+=3)) #自加3
  4.  
  5. echo $((a++)) #自增1
  6.  
  7. echo $((a>10)) #是则输出1,否则输出0
  8.  
  9. echo $((16#2a))

(12)$@与$*

先说下$#的作用,就是计算命令行有多少个参数

以前面的 my.sh p1 "p2 p3" 为例:
由于 p2 与 p3 之间的 IFS 是在 soft quote 中,因此 $# 可得到 2 的值。
但如果 p2 与 p3 没有置于 quoting 中话,那 $# 就可得到 3 的值了。

然后解释一下$@与$*

若在 command line 上跑 my.sh p1 "p2 p3" p4 的话,
不管是 $@ 还是 $* ,都可得到 p1 p2 p3 p4 就是了。
但是,如果置于 soft quote 中的话:
"$@" 则可得到 "p1" "p2 p3" "p4" 这三个不同的词段(word)﹔
"$*" 则可得到 "p1 p2 p3 p4" 这一整串单一的词段。

看一个演示程序

  1. #!/bin/bash
  2.  
  3. my_fun() {
  4. echo "$#"
  5. }
  6.  
  7. echo 'the number of parameter in "$@" is '$(my_fun "$@")
  8. echo 'the number of parameter in "$*" is '$(my_fun "$*")

然后执行

  1. ./my.sh p1 "p2 p3" p4
  2.  
  3. the number of parameter in "$@" is
  4.  
  5. the number of parameter in "$*" is

(13)&&与||

看下面一个例子:

  1. $ A=
  2. $ [ -n "$A" ] && echo "yes! it's ture." || echo "no, it's NOT ture."
  3. yes! it's ture.
  4. $ unset A
  5. $ [ -n "$A" ] && echo "yes! it's ture." || echo "no, it's NOT ture."
  6. no, it's NOT ture.

下面的判断是:当 $A 被赋与值时,再看是否小于 100 ,否则送出 too big! :

  1. $ A=
  2. $ [ -n "$A" ] && [ "$A" -lt ] || echo 'too big!'
  3. too big!

若我将 A 取消,照理说,应该不会送文字才对啊(因为第一个条件就不成立了)...

  1. $ unset A
  2. $ [ -n "$A" ] && [ "$A" -lt ] || echo 'too big!'
  3. too big!

为何上面的结果也可得到呢?
又,如何解决之呢?
(提示:修改方法很多,其中一种方法可利用第七章介绍过的 command group ...)

  1. [ -n "$A" ] && { [ "$A" -lt ] || echo 'too big!'; }
  1.  

到此,1-10问中的大部分知识点都记录下来了,春节这两天没事儿就看点,记录点,方便以后复习查阅,本想都整理到一篇文章中的,只是记录到10问后发现还有不少内容需要记录但是本文已经2500多字了,不可能舍弃一些内容。所以拆个上下。

网中人前辈整理的《shell十三问》确实使人收获颇多,每天看些,吸收些,真心不错,多看些真有点头痛呢、、引用白岩松同学的书名,真是“痛并快乐着”,哈哈、

《Shell十三问》笔记(上)的更多相关文章

  1. 菜鸟教程之学习Shell script笔记(上)

    菜鸟教程之学习Shell script笔记 以下内容是,学习菜鸟shell教程整理的笔记 菜鸟教程之shell教程:http://www.runoob.com/linux/linux-shell.ht ...

  2. 《Shell十三问》笔记(下)

    继续开始shell十三问中11-13问和后续补充的笔记,加油! (14)输入重定向与输出重定向 “>”是标准输出重定向,可以把输出结果送入文件 “<”是标准输入重定向,可以重新指定文件的内 ...

  3. Shell十三问[转]

    Shell十三问 转载于网络,稍加整理. (一) 为何叫做Shell? 我们知道计算机的运作不能离开硬件,但使用者却无法直接对硬件作驱动,硬件的驱动只能透过一个称为"操作系统(Operati ...

  4. shell十三问

    1) 为何叫做 shell ?在介绍 shell 是甚幺东西之前,不妨让我们重新检视使用者与计算机系统的关系:图(FIXME)我们知道计算机的运作不能离开硬件,但使用者却无法直接对硬件作驱动,硬件的驱 ...

  5. shell十三问:关于${0##*/} 和${0%/*}

    转自shell十三问:  http://bbs.chinaunix.net/thread-218853-1-1.html …… 假設我們定義了一個變量為:file=/dir1/dir2/dir3/my ...

  6. Shell编程笔记

    Shell编程笔记与Windows下熟悉的批处理类似,也可以将一些重复性的命令操作写成一个脚本方便处理.   修改别人的脚本,运行后遇到个问题 setenv: command not found 查证 ...

  7. 转:shell 经典, shell 十三问

      原文链接:http://blog.csdn.net/freexploit/article/details/626660 我在 CU 的日子并不长,有幸在 shell 版上与大家结缘.除了跟众前辈学 ...

  8. shell学习笔记

    shell学习笔记 .查看/etc/shells,看看有几个可用的Shell . 曾经用过的命令存在.bash_history中,但是~/.bash_history记录的是前一次登录前记录的所有指令, ...

  9. <转>shell经典,shell十三问

    (注:关于变量概念,我们留到下两章才跟大家说明.) 好了,更多的关于 command line 的格式,以及 echo 命令的选项,就请您自行多加练习.运用了... ----------------- ...

随机推荐

  1. Cocos3d-x 发布第一版

    从去年开始11一月,我开始一个又一个人cocos3d的C++改写版本号.现在见效.所有cocos3d的OC代码改写成了C++. 在正常Android和Windows在执行.上周,正式发布了第一个版本. ...

  2. QR代码简单

    QR代码(Quick Response Code, 高速响应码)属于二维矩阵码在一个.由DENSO(日本电装)公司开发,由JIS和ISO将其标准化. QR码分为两种模式:模式1.模式2.当中.模式1相 ...

  3. uva 10635 Prince and Princess(LCS成问题LIS问题O(nlogn))

    标题效果:有两个长度p+1和q+1该序列.的各种元素的每个序列不是相互同.并1~n^2之间的整数.个序列的第一个元素均为1. 求出A和B的最长公共子序列长度. 分析:本题是LCS问题,可是p*q< ...

  4. 【Linux命令】--(9)其他常用命令

    其他常用命令+++++++++++++++++++++++++++++++lndiffdatecal grep wcpswatchatcrontab++++++++++++++++++++++++++ ...

  5. BZOJ 1823 JSOI 2010 盛宴 2-SAT

    标题效果:有着n材料的种类,m陪审团. 每种材料具有两种不同的方法.每个法官都有两个标准.做出来的每一个法官的菜必须至少满足一个需求. 问:是否有这样一个程序. 思考:2-SAT经典的内置图形问题.因 ...

  6. Android 异步消息处理机制 让你在深入了解 Looper、Handler、Message之间的关系

    转载请注明出处:http://blog.csdn.net/lmj623565791/article/details/38377229 ,本文出自[张鸿洋的博客] 非常多人面试肯定都被问到过,请问And ...

  7. linux 文件系统解析及相关命令(转)

    简介 文件系统就是分区或磁盘上的所有文件的逻辑集合. 文件系统不仅包含着文件中的数据而且还有文件系统的结构,所有Linux 用户和程序看到的文件.目录.软连接及文件保护信息等都存储在其中. 不同Lin ...

  8. SVD在推荐系统中的应用

    一.奇异值分解SVD 1.SVD原理 SVD将矩阵分为三个矩阵的乘积,公式: 中间矩阵∑为对角阵,对角元素值为Data矩阵特征值λi,且已经从大到小排序,即使去掉特征值小的那些特征,依然可以很好地重构 ...

  9. 从源码看Android中sqlite是怎么读DB的(转)

    执行query 执行SQLiteDatabase类中query系列函数时,只会构造查询信息,不会执行查询. (query的源码追踪路径) 执行move(里面的fillwindow是真正打开文件句柄并分 ...

  10. oracle之sql语句优化

    oracle之sql语句优化 sql语句的优化 1.在where子句中使用 is null 或 is not null 时,oracle优化器就不能使用索引了. 2.对于有连接的列,即使最有一个是静态 ...