Linux Shell 脚本

Q:1 Shell脚本是什么、它是必需的吗?

 

答:一个Shell脚本是一个文本文件,包含一个或多个命令。作为系统管理员,我们经常需要使用多个命令来完成一项任务,我们可以添加这些所有命令在一个文本文件(Shell脚本)来完成这些日常工作任务。

 

Q:2 什么是默认登录shell,如何改变指定用户的登录shell

 

答:在Linux操作系统,“/bin/bash”是默认登录shell,是在创建用户时分配的。使用chsh命令可以改变默认的shell。示例如下所示:

 

 

# chsh <用户名> -s <新shell>
# chsh linuxtechi -s /bin/sh

 

Q:3 可以在shell脚本中使用哪些类型的变量?

 

答:在shell脚本,我们可以使用两种类型的变量:

 

  • 系统定义变量

  • 用户定义变量

 

系统变量是由系统系统自己创建的。这些变量通常由大写字母组成,可以通过“set”命令查看。

 

用户变量由系统用户来生成和定义,变量的值可以通过命令“echo $<变量名>”查看。

 

Q:4 如何将标准输出和错误输出同时重定向到同一位置?

 

答:这里有两个方法来实现:

 

方法一:

 

 

2>&1 (如# ls /usr/share/doc > out.txt 2>&1 )

 

方法二:

 

 

&> (如# ls /usr/share/doc &> out.txt )

 

Q:5 shell脚本中“if”语法如何嵌套?

 

答:基础语法如下:

 

if [ 条件 ]
then
命令1
命令2
…..
else
if [ 条件 ]
then
命令1
命令2
….
else
命令1
命令2
…..
fi
fi

 

Q:6 shell脚本中“$?”标记的用途是什么?

 

答:在写一个shell脚本时,如果你想要检查前一命令是否执行成功,在if条件中使用“$?”可以来检查前一命令的结束状态。简单的例子如下:

 

 

root@localhost:~# ls /usr/bin/shar
/usr/bin/shar
root@localhost:~# echo $?
0

 

如果结束状态是0,说明前一个命令执行成功。

 

 

root@localhost:~# ls /usr/bin/share
ls: cannot access /usr/bin/share: No such file or directory
root@localhost:~# echo $?
2

 

如果结束状态不是0,说明命令执行失败。

 

Q:7 在shell脚本中如何比较两个数字 ?

 

答:在if-then中使用测试命令( -gt 等)来比较两个数字,例子如下:

 

 

#!/bin/bash
x=10
y=20
if [ $x -gt $y ]
then
echo “x is greater than y”
else
echo “y is greater than x”
fi

 

Q:8 shell脚本中break命令的作用 ?

 

答:break命令一个简单的用途是退出执行中的循环。我们可以在while和until循环中使用break命令跳出循环。

 

Q:9 shell脚本中continue命令的作用 ?

 

答:continue命令不同于break命令,它只跳出当前循环的迭代,而不是整个循环。continue命令很多时候是很有用的,例如错误发生,但我们依然希望继续执行大循环的时候。

 

Q:10 告诉我shell脚本中Case语句的语法 ?

 

答:基础语法如下:

 

 

case 变量 in
值1)
命令1
命令2
…..
最后命令
!!
值2)
命令1
命令2
……
最后命令
;;
esac

 

Q:11 shell脚本中while循环语法 ?

 

答:如同for循环,while循环只要条件成立就重复它的命令块。不同于for循环,while循环会不断迭代,直到它的条件不为真。基础语法:

 

 

while [ 条件 ]
do
命令…
done

 

Q:12 如何使脚本可执行 ?

 

答:使用chmod命令来使脚本可执行。例子如下:

 

 

# chmod a+x myscript.sh

 

Q:13 “#!/bin/bash”的作用 ?

 

答:#!/bin/bash是shell脚本的第一行,称为释伴(shebang)行。这里#符号叫做hash,而! 叫做 bang。它的意思是命令通过 /bin/bash 来执行。

 

Q:14 shell脚本中for循环语法 ?

 

答:for循环的基础语法:

 

 

for 变量 in 循环列表
do
命令1
命令2
….
最后命令
done

 

Q:15 如何调试shell脚本 ?

 

答:使用'-x'参数(sh -x myscript.sh)可以调试shell脚本。另一个种方法是使用‘-nv’参数( sh -nv myscript.sh)。

 

Q:16 shell脚本如何比较字符串?

 

答:test命令可以用来比较字符串。测试命令会通过比较字符串中的每一个字符来比较。

 

Q:17 Bourne shell(bash) 中有哪些特殊的变量 ?

 

答:下面的表列出了Bourne shell为命令行设置的特殊变量。

 

 

内建变量    解释
$0    命令行中的脚本名字
$1    第一个命令行参数
$2    第二个命令行参数
…..    …….
$9    第九个命令行参数
$#    命令行参数的数量
$*    所有命令行参数,以空格隔开

 

Q:18 在shell脚本中,如何测试文件 ?

 

答:test命令可以用来测试文件。基础用法如下表格:

 

 

Test         用法
-d 文件名    如果文件存在并且是目录,返回true
-e 文件名    如果文件存在,返回true
-f 文件名    如果文件存在并且是普通文件,返回true
-r 文件名    如果文件存在并可读,返回true
-s 文件名    如果文件存在并且不为空,返回true
-w 文件名    如果文件存在并可写,返回true
-x 文件名    如果文件存在并可执行,返回true

 

Q:19 在shell脚本中,如何写入注释 ?

 

答:注释可以用来描述一个脚本可以做什么和它是如何工作的。每一行注释以#开头。例子如下:

 

 

#!/bin/bash
# This is a command
echo “I am logged in as $USER

 

Q:20 如何让 shell 就脚本得到来自终端的输入?

 

答:read命令可以读取来自终端(使用键盘)的数据。read命令得到用户的输入并置于你给出的变量中。例子如下:

 

 

# vi /tmp/test.sh
#!/bin/bash
echo ‘Please enter your name’
read name
echo “My Name is $name
# ./test.sh
Please enter your name
LinuxTechi
My Name is LinuxTechi

 

Q:21 如何取消变量或取消变量赋值 ?

 

答:“unset”命令用于取消变量或取消变量赋值。语法如下所示:

 

 

# unset <变量名>

 

Q:22 如何执行算术运算 ?

 

答:有两种方法来执行算术运算:

 

1.使用expr命令

 

 

# expr 5 + 2

 

2.用一个美元符号和方括号($[ 表达式 ])例如:

 

 

test=$[16 + 4] ; test=$[16 + 4]

 

Q:23 do-while语句的基本格式 ?

 

答:do-while语句类似于while语句,但检查条件语句之前先执行命令(LCTT 译注:意即至少执行一次。)。下面是用do-while语句的语法

 

 

do
{
命令
while (条件)

 

Q:24 在shell脚本如何定义函数呢 ?

 

答:函数是拥有名字的代码块。当我们定义代码块,我们就可以在我们的脚本调用函数名字,该块就会被执行。示例如下所示:

 

 

$ diskusage () { df -h ; }
译注:下面是我给的shell函数语法,原文没有
[ function ] 函数名 [()]
{
命令;
[return int;]

}

1在脚本中多写注释

这条建议不但适合于shell脚本,对于其它编程语言同样实用。这样做的好处是可以帮助他人或者你自己后面阅读脚本时能够迅速了解脚本的功能,便于维护。试想下,如果有一天你的主管给了你一堆没有注释的代码,你是不是很抓狂呢。

shell脚本中,注释一般以#开头,后面跟着是注释的内容。注释不可少,但是注释也不是越多越好,对于shell脚本,注释一般包括下面几个部分:

 

1 #脚本的参数
2 #脚本的用途
3 #脚本的注意事项
4 #脚本的写作时间,作者,版权等
5 #各个函数前的说明注释6 #一些较复杂的单行命令注释

2脚本中有命令运行失败时退出脚本

如果脚本中某条命令运行失败,我们不应该让其继续运行,因为这样可能会影响脚本的其余部分,导致逻辑错误。逻辑错误一般又是很难定位的,与其这样,不如让其提前结束更早的找出脚本中的错误。可以用下面的命令设置在遇到命令失败时来退出脚本执行:

# 如果命令运行失败让脚本退出执行
set -o errexit 

# 或
set -e

3脚本中使用到未声明变量时退出脚本

脚本中使用到未声明的变量同样可能导致逻辑错误,可以用下面的命令设置脚本在使用到未声明的变量时退出执行:

# 若有用未设置的变量即让脚本退出执行
set -o nounset

# 或
set-u

4使用双引号引用变量

使用一个变量的值时,用双引号有助于防止由于空格导致单词分割开和由于识别和扩展了通配符而导致的不必要匹配,下面的例子说明了一切。

#!/bin/bash# 若命令失败让脚本退出
set -o errexit 

# 若未设置的变量被使用让脚本退出
set -o nounset

echo "Names without double quotes" 
echo
names="Tecmint FOSSMint Linusay"
for name in $names; do  echo "$name
done
echo
echo "Names with double quotes" 
echo
for name in "$names"; do  echo "$name"
done
exit 0

5在脚本中多使用函数

同样,这条建议也适用其它编程语言,函数的使用使得代码更模块化,更可读和可重用;shell脚本中定义函数的语法如下所示:

function check_root()
{  
    command1;     
    command2;
}

# 或
check_root()
{    
    command1;     
    command2;
}

 

 

6字符串比较时用=而不是==

为什么会有这个建议,原文并没有给出详细的说明,只是简单的提了句: == 是 = 的同义词,因此仅用个单个 = 来做字符串比较。不过在原文的后面有网友回答了,是因为==只适用于bash;POSIX形式是“=”,使用 = 更方便移植。

value1="tecmint.com"
value2="fossmint.com"
if [ "$value1" = "$value2" ]

 

 

7用$(command) 代替传统的`command`

这两种形式都可以用作命令替换,所谓命令替换是用这个命令的输出结果取代命令本身。这里建议用 $(command) 而不是反引号 `command `来做命令代换。

 

//不建议做法
user=`echo "$UID"`

建议做法user=
$(echo "$UID")

 

8用readonly声明静态变量

静态变量不会改变;它的值一旦在脚本中定义后就不能被修改,对于这类变量,在声明的时候应该用readonly去声明。

readonly passwd_file="/etc/passwd"
readonly group_file="/etc/group"

 

9命名规范

由于在bash环境变量名字都是大写,因此建议自己定义的变量用小写字母命名,以防止命名冲突,同时命名时要做到顾名思义,你取的名字要有意义。

#而环境变量用大写命名,自定义变量用小写,
nikto_file="$HOME/Downloads/nikto-master"

 

10对比较长的脚本进行调试

当你的脚本有成千上万行时,一旦脚本运行出错,排错是一件异常困难的事情;为了在脚本执行前更轻松地将问题定位出,需要进行一些调试,下面是一些指南来帮助你掌握调试的技巧。

How To Enable Shell Script Debugging Mode in Linux

How to Perform Syntax Checking Debugging Mode in Shell Scripts

How to Trace Execution of Commands in Shell Script with Shell Tracing

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值