⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 bash编程教学实例.txt

📁 linux shell bash 编程
💻 TXT
📖 第 1 页 / 共 3 页
字号:
I am C 

1) a 

2) b 

3) c 

#? 4 

返回状态Exit 

在继续下去之前,我们必须要切入另一个话题,即返回状态值 - Exit Status。因为if/while/until都迁涉到了使用Exit Status来控制程式流程的问题。 

*************************

许多人都知道,在许多语言中(C/C++/Perl....),都有一个exit的函数,甚至连Bash自己都有个exit的内建命令。而exit后面所带的数字,便是返回状态值 - Exit Status。 

返回状态值可以使得程式与程式之间,利用Shell script来结合的可能性大增,利用小程式,透过Shell script,来完成很杂的工作。 

在shell中,返回值为零表示成功(True),非零值为失败(False)。 

*************************

举例来说,以下这个两个小程式yes/no分别会返回0/1(成功/失败): 

/* yes.c */ 

void main(void) { exit(0); } 

/* no.c */ 

void main(void) { exit(1); } 

那么以下这个"YES"的shell script便会显示"YES"。
#!/bin/sh 

# YES 

if yes ; then 

echo "YES" 

fi 

而"NO"不会显示任何东西。 

#!/bin/sh 

# NO 

if no ; then 

echo "YES" 

fi 

*************************

test express 

[ express ] 

在Shell script中,test express/[ express ]这个语法被大量地使用,它是个非常实用的指令。由于它的返回值即Exit Status,经常被运用在if/while/until的场合中。而在后面,我们也会大量运用到,在进入介绍if/while/until之前,有必要 先解一下。 

其返回值为0(True)或1(False),要看表述(express)的结果为何。 

express格式 

-b file : 当档案存在并且属性是Block special(通常是/dev/xxx)时,返回True。 

-c file : 当档案存在并且属性是character special(通常是/dev/xxx)时,返回True。 

-d file : 当档案存在并且属性是目录时,返回True。 

-e file : 当档案存在时,返回True。 

-f file : 当档案存在并且是正常档案时,返回True。 

-g file : 当档案存在并且是set-group-id时,返回True。 

-k file : 当档案存在并且有"sticky" bit被设定时,返回True。 

-L file : 当档案存在并且是symbolic link时,返回True。 

-p file : 当档案存在并且是name pipe时,返回True。 

-r file : 当档案存在并且可读取时,返回True。 

-s file : 当档案存在并且档案大小大于零时,返回True。 

-S file : 当档案存在并且是socket时,返回True。 

-t fd : 当fd被开启为terminal时,返回True。 

-u file : 当档案存在并且set-user-id bit被设定时,返回True。 

-w file : 当档案存在并且可写入时,返回True。 

-x file : 当档案存在并且可执行时,返回True。 

-O file : 当档案存在并且是被执行的user id所拥有时,返回True。 

-G file : 当档案存在并且是被执行的group id所拥有时,返回True。 

file1 -nt file2 : 当file1比file2新时(根据修改时间),返回True。 

file1 -ot file2 : 当file1比file2旧时(根据修改时间),返回True。
file1 -ef file2 : 当file1与file2有相同的device及inode number时,返回True。 

-z string : 当string的长度为零时,返回True。 

-n string : 当string的长度不为零时,返回True。 

string1 = string2 : string1与string2相等时,返回True。 

string1 != string2 : string1与string2不相等时,返回True。 

! express : express为False时,返回True。 

expr1 -a expr2 : expr1及expr2为True。 

expr1 -o expr2 : expr1或expr2其中之一为True。 

arg1 OP arg2 : OP是-eq[equal]、-ne[not-equal]、-lt[less-than]、-le[less-than-or-equal]、-gt [greater-than]、-ge[greater-than-or-equal]的其中之一。 

*************************

在Bash中,当错误发生在致命信号时,bash会返回128+signal number做为返回值。如果找不到命令,将会返回127。如果命令找到了,但该命令是不可执行的,将返回126。除此以外,Bash本身会返回最后一个 指令的返回值。若是执行中发生错误,将会返回一个非零的值。 

Fatal Signal : 128 + signo 

Can't not find command : 127 

Can't not execute : 126 

Shell script successfully executed : return the last command exit status 

Fatal during execution : return non-zero 

流程控制if 

if list then list [ elif list then list ] ... [ else list ] fi 

几种可能的写法 

*************************

第一种 

if list then 

do something here 

fi 

当list表述返回值为True(0)时,将会执行"do something here"。 

例一 : 当我们要执行一个命令或程式之前,有时候需要检查该命令是否存在,然后才执行。 

if [ -x /sbin/quotaon ] ; then 

echo "Turning on Quota for root filesystem" 

/sbin/quotaon / 

fi 

例二 : 当我们将某个档案做为设定档时,可先检查是否存在,然后将该档案设定值载入。 

# Filename : /etc/ppp/settings
PHONE=1-800-COLLECT 

#!/bin/sh 

# Filename : phonebill 

if [ -f /etc/ppp/settings ] ; then 

source /etc/ppp/settings 

echo $PHONE 

fi 

执行 

[foxman@foxman ppp]# ./phonebill 

1-800-COLLECT 

*************************

第二种 

if list then 

do something here 

else 

do something else here 

fi 

例三 : Hostname 

#!/bin/sh 

if [ -f /etc/HOSTNAME ] ; then 

HOSTNAME=`cat /etc/HOSTNAME` 

else 

HOSTNAME=localhost 

fi 

*************************

第三种 

if list then 

do something here 

elif list then 

do another thing here 

fi 

例四 : 如果某个设定档允许有好几个位置的话,例如crontab,可利用if then elif fi来找寻。 

#!/bin/sh 

if [ -f /etc/crontab ] ; then 

CRONTAB="/etc/crontab" 

elif [ -f /var/spool/cron/crontabs/root ] ; then 

CRONTAB="/var/spool/cron/crontabs/root" 

elif [ -f /var/cron/tabs/root ] ; then 

CRONTAB="/var/cron/tabs/root" 

fi 

export CRONTAB 

*************************

第四种 

if list then 

do something here 

elif list then 

do another thing here 

else 

do something else here 

fi 

例五 : 我们可利用uname来判断目前系统,并分别做各系统状况不同的事。 

#!/bin/sh 

SYSTEM=`uname -s` 

if [ $SYSTEM = "Linux" ] ; then
echo "Linux" 

elif [ $SYSTEM = "FreeBSD" ] ; then 

echo "FreeBSD" 

elif [ $SYSTEM = "Solaris" ] ; then 

echo "Solaris" 

else 

echo "What?" 

fi 

控制圈while/until 

while list do list done 

当list为True时,该圈会不停地执行。 

例一 : 无限回圈写法 

#!/bin/sh 

while : ; do 

echo "do something forever here" 

sleep 5 

done 

例二 : 强迫把pppd杀掉。 

#!/bin/sh 

while [ -f /var/run/ppp0.pid ] ; do 

killall pppd 

done 

*************************

until list do list done 

当list为False(non-zero)时,该圈会不停地执行。 

例一 : 等待pppd上线。 

#!/bin/sh 

until [ -f /var/run/ppp0.pid ] ; do 

sleep 1 

done 

参数与变数 

在继续下去介绍function之前,我们必须停下来介绍"参数与变数"。 

*************************

参数(Parameters)是用来储存"值"的资料型态,有点像是一般语言中的变数。它可以是个名称(name)、数字(number)、或者是以下所列出来一些特殊符号(Special Parameters)。 

在shell中,变数是由name形式的参数所构成的。 

*************************

在前面的许多例中,我们事实上已经看到许多参数的运用。要设定一个Parameter实际很简单: 

name=value 

例如说: 

MYHOST="foxman" 

而要使用它时,则是加个"$"符号。 

echo $MYHOST 

*************************

位置参数(Positional Parameters) 

*************************

所谓的位置参数便是0,1,2,3,4,5,6,7,8,9...。使用时,用,,...。 

位置参数是当script被载入时,后面所附加的参数。是本身,则为第一个参数,为第二个,依此类推。而当Positional Parameters被function所使用时,它们会被暂时取代(下一节会介绍function)。
例如以下这个script: 

#!/bin/sh 

# Filename : position 

echo 

echo 

执行时: 

[foxman@foxman bash]# ./position abc 

./position 

abc 

当位置参数超过两位数时,有特别的方法来展开,称为Expansion。 

*************************

特殊参数(Speical Parameters) 

这些符号,非常不人性,对新手来说很困扰。但上手后,会觉得方便无比,有些如果您看不懂的话,就--算了,不用浪费太多时间在上面。 

*************************

* 星号 

将Positional Parameters合成一个参数,其间隔为IFS内定参数的第一个字元(见内建变数一节)。 

例: 

#!/bin/sh 

# starsig 

echo $* 

执行: 

[foxman@foxman bash]# starsig a b c d e f g 

a b c d e f g 

*************************

@ at符号 

与*星号类同。不同之处在于不参照IFS。 

例: 

#!/bin/sh 

# atsig 

echo $@ 

执行: 

[foxman@foxman bash]# atsig a b c d e f g 

a b c d e f g 

*************************

# 井字号 

展开Positional parameters的数量。 

例: 

#!/bin/sh 

# poundsig 

echo $# 

执行 

[foxman@foxman bash]# poundsig a b c d e f g 

7 

*************************

? 问号 

最近执行的foreground pipeline的状态。 

*************************

- 减号 

最近执行的foreground pipeline的选项参数。 

*************************

$ 钱钱钱 

本身的Process ID。 

[foxman@foxman bash]# ps ax | grep bash 

1635 p1 S 0:00 /bin/bash 

[foxman@foxman bash]# echo $$ 

1635 

*************************
! 惊号 

最近执行背景命令的Process ID。 

*************************

0 零 

在Positional Parameters一部份已经说明过了,是执行的shell script本身。但如果是用"bash -c",则被设为第一个参数。 

[foxman@foxman bash]# echo 

/bin/bash 

*************************

_ 底线符号 

显示出最后一个执行的命令。 

[foxman@foxman bash]# echo $_ 

bash 

*************************

内建变数(Shell Variables) 

Bash有许多内建变数,像PATH、HOME、ENV......等等。这些内建变数将在另一节中,专门一一说明。

函数function 

[ function ] name () { list; } 

function的参数是Positional Paraments。 


#!/bin/sh 

function func() { 

echo 

echo 

return 1 

} 

func "Hello" "function" 

局部变数可用local来宣告。 

函数可export,使用下一层的shell可以使用。 

函数可递,没有递层数的限制。 

Bash内建指令集 

以下的命令,大部份都没有使用例,您可能会看不出所以然,摸不著头脑。在我加入例说明前,建议您"man bash",然后自己实际操作一次。 

*************************

: [arguments] 

不做任何事,除了[arguments]一些参数展开及一些特定重导向的作业外。 

永远返回零。它的用法跟true一样。 

*************************

. filename [arguments] 

source filename [arguments] 

由filename中读取命令,并执行。 

您会在/etc/rc.d/*中发现很多 

. /xxxx 

的指令,而xxxx的permission都不是可执行的。事实上,在tcsh中,需要用 

source /xxxx 

来做同样的指令。 

注意到"."的后面是有空格的(比较一下". /"跟"./",不一样)。filename是内含指令的纯文字档即可,无须chmod 755 filename。 


filename : my_source
DEV=lo 

IP=127.0.0.1 

NETMASK=255.0.0.0 

BROADCAST=127.255.255.255 

ifconfig $IP netmask $NETMASK broadcast $BROADCAST dev $DEV 

接下来 

. my_source 


source my_source 

便可执行该script,而不需要"chmod 755 my_source" 

*************************

alias [name[=value] ...] 

昵称命令 

例如您如果来自DOS的世界,对UNIX的指令不习惯,可用alias来修改,以符合您的习惯。 


alias ls="ls --color" 

alias dir="ls" 

alias cd..="cd .." 

alias copy="cp -f" # dangerous, recommend, "cp -i" 

alias del="rm -f" # dangerous, recommend, "rm -i" 

alias move="mv -f" # dangerous, recommend, "mv -i" 

alias md="mkdir" 

alias rd="rmdir" 

*************************

unalias [-a] [name ...] 

unalias取消alias的设定。"unalias -a"将全部alias取消。 


unalias copy 

*************************

bg [jobspec] 

将指定任务放到背景中,如果jobspec未指定,内定为目前的。 

*************************

fg [jobspec] 

将指定任务放到前景中,如果jobsepc没有指定,那么内定为目前的。 

*************************

jobs [-lnp] [ jobspec ... ] 

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -