Bash 和 ShellScript 的基础

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)
-gtn1 大于 n2 (greater than)
-ltn1 小于 n2 (less than)
-gen1 大于等于 n2 (greater than or equal)
-len1 小于等于 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 在父程序中运行;

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值