Bash基础:
jobs控制:
1) 前台运行、后台运行:
通过在命令后接上&符,可以让命令在后台执行;当命令需要较长时间,可以通过在命令后加&符使其不占用前台;
例:
前台执行: find /etc -exec grep "tty" {} /; -exec ls -l {} /; >~/FindResult 2>/dev/null
后台执行: find / etc -exec grep "tty" {} /; -exec ls -l {} /; >~/FindResult 2>/dev/null &
2) 查看Bash后台工作状态:
通过jobs命令查看目前后台的工作状态;
例:
find / etc -exec grep "tty" {} /; -exec ls -l {} /; >~/FindResult 2>/dev/null &
jobs -l ("-l"是为了带出进程的PID)
3) 将当前正在前台运行的进程暂停至后台:
通过[ctrl + z]将当前正在前台运行的进程暂停至后台;
例:
find / etc -exec grep "tty" {} /; -exec ls -l {} /; >~/FindResult 2>/dev/null
[ctrl + z]
jobs
4) 将后台运行的进程恢复到前台运行:
通过 fg %n 将后台运行的进程恢复到前台运行(n为jobs命令中查看到的编号;如果不写%n,则恢复最迟置到后台运行的进程);
例:
find / etc -exec grep "tty1" {} /; -exec ls -l {} /; >~/FindResult 2>/dev/null &
find / etc -exec grep "tty2" {} /; -exec ls -l {} /; >~/FindResult 2>/dev/null &
jobs
fg %2
5) 将后台停止的进程置为后台运行:
通过 bg %n 将后台停止的进程置为后台运行;
例:
find / etc -exec grep "tty" {} /; -exec ls -l {} /; >~/FindResult 2>/dev/null
[ctrl + z]
bg %1
6) 终止进程:
通过 kill %n 或 kill pid 将进程终止;
例:
kill %1 (将后台编号为1的进程终止)
kill 18778 (将pid为18778的进程终止)
shell变量,环境变量:
1) 变量的取用: $变量名
例:
显示变量的内容: echo $PATH (PATH为变量名)
2) 变量的设置:
2.1) 设置变量
[1]直接设置变量:myname=xsz0606(变量名=内容)
[2]使用read命令以对谈方式设置变量:read -p "input something for var" var (read -p "对谈内容" 变量名)
2.2) 取消变量:unset myname(unset 变量名)
另, 设置变量时,需要注意以下几点:
[1] 变量与变量内容之间以=(等号)连接; =两边不能连接空白字符;
[2] 变量内容含有空格的,可以使用双引号或单引号将内容包含起来; 其中:双引号内的特殊字符(如$符)可以保留原来的特性; 单引号内的则按一般字符处理(纯文字);
[3] 可以使用转义字符/(斜线)将特殊字符转换为一般字符; 例如将双引号按一般字符处理, 则: text="/"double qoute/""
[4] 为了方便区分, 系统的环境变量均使用大写字母;自订变量多使用小写字母;但这并不是硬性规定;
[5] 可以通过 $(指令) 或`指令`(反引号)的方式调用命令, 并赋予变量: version=$(uname -r)
[6] 使用unset命令,取消已存在的变量;
3) 普通变量 与 环境变量
环境变量与普通变量的区别在于:可以被当前shell的子程序引用的为环境变量; 不可以被子程序引用的为普通变量;
3.1) 环境变量与普通变量的转换:
将XXX置为环境变量:declare -x XXX 或 export XXX;
将环境变量XXX置为普通变量:declare +x XXX;
3.2) 查看当前Shell中的变量
[1] 使用 env 或 export 查看当前shell环境下的所有环境变量
[2] 使用 set 查看所有变量(包括普通变量及环境变量)
特殊变量,变量声明:
1) 特殊变量(持续更新):
[1]$$: 当前shell 的PID (需要确认, 是当前shell的PID, 还是当前子程序/sh脚本的PID)
[2]$?: 上次执行命令的返回值
[3]$PS1: 命令行提示字符的设定; 可以使用[man bash], 搜索 PS1 查看具体说明
[4]$OSTYPE, $HOSTTYPE, $MACHTYPE:主机硬件与内核的等级(详细说明参见鸟哥的Linux私房菜基础篇第11章)
2) 变量的声明(declare / typeset):
declare 与 typeset 是同样的功能
[1] declare -a: 将变量声明为一个数组(需要确认数组是否基于0),例:
declare -a array
array[1]=222
array[2]=alpha
[2] declare -i: 将变量声明为一个整型,例:
declare -i intVal
intVal=100+200
echo $intVal
[3] declare -x: 将变量声明为环境变量
[4] declare -r: 将变量声明为只读(注意,变量设置了只读类型之后,就不能修改,也无法unset; 通常只有退出再登录shell才能重置变量)
流重导向,管道命令:
1) 三种流:
标准输入(stdin):0
标准输出(stdout):1
标准异常(stderr):2
2) 流重导向:
输出流:> (注意,单独的 > 等于 1>, 即操作的流是stdout)
追加流:>>
输入流:<
输入流并指定结束符:<<
具体操作参见(鸟哥的Linux私房菜基础篇第11章);
另,0、1、2这些代号称为file descriptor
变量加载顺序:
1) 基本认识(看了鸟哥相关章节后,自己的总结)
变量的生命周期应该和shell是一致的(这句话是我自己的总结,不一定正确);即退出shell之后,之前在shell中设定的变量就会失效;
矛盾的是,每次登录shell之后,为什么我们总可以看到$PATH,$HOME这些变量?
其实这是因为系统中存在着一些环境设定文件,当bash启动时会读取这些文件,以预设一些环境变量,供后续环境设定工作使用;
这些设定文件又分为"系统全局设定文件",以及"个人风格设定文件";
由于退出shell后,之前的设定就会消失;所以希望保留某些设定的话,就应该将这些设定写入相关的设定文件;
2) login shell 与 non-login shell
login shell:需要完整登录流程的才能取得的shell;
non-login shell:不需要登录流程的就能取得的shell;例如, 在当前shell中键入[bash]进入的子程序,就属于non-login shell;
[1]login shell会读取的设定文件:
/etc/profile: 整体环境设定,用于设定一些基础的环境变量,例如PATH,USER等;另外它还会呼叫其他的设定文件;
~/.bash_profile,~/.bash_login,~/.profile: 用户个人设定文件, 三选其一的执行;如果存在 ~/.bashrc,也会执行这个文件;
[2]non-login shell会读取的设定文件: ~/.bashrc
---------------------------------------
大致的环境流程是:
[/etc/profile] -> [~/.bash_profile](~/.bashrc) -> 开始操作bash
详细说明参见(鸟哥的Linux私房菜基础篇第11章)
Shell Script基础:
1) 注释:
第一行: #!/bin/bash 说明脚本使用的shell的名称;
#开头的行, 均为注释
2) 判断式:
[1] test 命令(摘自鸟哥):
测试的标志 | 代表意义 |
1. 关于某个档名的‘档案类型’判断,如 test -e filename 表示存在否 | |
-e | 该‘档名’是否存在?(常用) |
-f | 该‘档名’是否存在且为档案(file)?(常用) |
-d | 该‘档名’是否存在且为目录(directory)?(常用) |
-b | 该‘档名’是否存在且为一个 block device 装置? |
-c | 该‘档名’是否存在且为一个 character device 装置? |
-S | 该‘档名’是否存在且为一个 Socket 档案? |
-p | 该‘档名’是否存在且为一个 FIFO (pipe) 档案? |
-L | 该‘档名’是否存在且为一个连结档? |
2. 关于档案的权限侦测,如 test -r filename 表示可读否 (但 root 权限常有例外) | |
-r | 侦测该档名是否存在且具有‘可读’的权限? |
-w | 侦测该档名是否存在且具有‘可写’的权限? |
-x | 侦测该档名是否存在且具有‘可执行’的权限? |
-u | 侦测该档名是否存在且具有‘SUID’的属性? |
-g | 侦测该档名是否存在且具有‘SGID’的属性? |
-k | 侦测该档名是否存在且具有‘Sticky bit’的属性? |
-s | 侦测该档名是否存在且为‘非空白档案’? |
3. 两个档案之间的比较,如: test file1 -nt file2 | |
-nt | (newer than)判断 file1 是否比 file2 新 |
-ot | (older than)判断 file1 是否比 file2 旧 |
-ef | 判断 file1 与 file2 是否为同一档案,可用在判断 hard link 的判定上。 主要意义在判定,两个档案是否均指向同一个 inode 哩! |
4. 关于两个整数之间的判定,例如 test n1 -eq n2 | |
-eq | 两数值相等 (equal) |
-ne | 两数值不等 (not equal) |
-gt | n1 大于 n2 (greater than) |
-lt | n1 小于 n2 (less than) |
-ge | n1 大于等于 n2 (greater than or equal) |
-le | n1 小于等于 n2 (less than or equal) |
5. 判定字串的资料 | |
test -z string | 判定字串是否为 0 ?若 string 为空字串,则为 true |
test -n string | 判定字串是否非为 0 ?若 string 为空字串,则为 false。 注: -n 亦可省略 |
test str1 = str2 | 判定 str1 是否等于 str2 ,若相等,则回传 true |
test str1 != str2 | 判定 str1 是否不等于 str2 ,若相等,则回传 false |
6. 多重条件判定,例如: test -r filename -a -x filename | |
-a | (and)两状况同时成立!例如 test -r file -a -x file,则 file 同时具有 r 与 x 权限时,才回传 true。 |
-o | (or)两状况任何一个成立!例如 test -r file -o -x file,则 file 具有 r 或 x 权限时,就可回传 true。 |
! | 反相状态,如 test ! -x file ,当 file 不具有 x 时,回传 true |
[2] 使用中括号 [ 表达式 ]:
使用中括号也可以实现test命令的效果,例如:
test -e file 等于 [ -e file ]
另外需要注意的是,中括号内的每个元素都需要以空格隔开;以下划线_代替空格的话,例如:
[-z_file] //错误
[_-z_file_] //正确
3) 预设变量:
$0,$1,$2...$n: 指第几个参数, $0为脚本名称, $1为第一个参数,如此类推
$#: 参数的个数
$@: 查看所有参数("$1" "$2" "$3"),$@比较常用
$*: 查看所有参数("$1 $2 $3")
4) 分支控制语句:
[1] if...else if...else...
if [ condition ]; then
......
elif [ condition ]; then
......
else
......
fi
[2]case....
case $XXX in
"condition1" )
......
;;
"condition2" )
......
;;
"condition3" )
......
;;
* )
......
;;
esac
5) 函数
[1] 函数定义:
function fn()
{
........;
// 通过$1, $2获取参数
}
[2] 函数调用
fn para1 para2;
6) 循环
[1] while...
while [ condition ]
do
......
done
[2] for...in...
for var in $(seq 1 100)
do
......
done
[3] for...;...;...
for (( i=1; i<$n; i=i+1 ))
do
......
done
一些技巧及其他知识(杂,持续更新):
[1] 查看系统可用的Shell:cat /etc/shells
[2] $符的嵌套使用: eval echo /$$i
[3] 使用[Ctrl + s]"锁住"当前屏幕(即键盘输入任何东西都"没反应");使用[Ctrl + q]解锁;
[4] shell script执行方式的差异([./XXX.sh], [sh XXX.sh], [. XXX.sh], [source XXX.sh])
./ XXX.sh 与 sh XXX.sh 在子程序中运行;
. XXX.sh 与 source XXX.sh 在父程序中运行;