博客园首页新随笔联系管理订阅
随笔- 216 文章- 1 评论- 6
阿里Linux Shell脚本面试25个经典问答
转载: 阿里Linux Shell脚本面试25个经典问答
Q:1 Shell脚本是什么、它是必需的吗?
答:一个Shell脚本是一个文本文件,包含一个或多个命令。作为系统管理员,我们经常需要使用多个命令来完成一项任务,我们可以添加这些所有命令在一个文本文件(Shell脚本)来完成这些日常工作任务。
Q:2 什么是默认登录shell,如何改变指定用户的登录shell
答:在Linux操作系统,“/bin/bash”是默认登录shell,是在创建用户时分配的。使用chsh命令可以改变默认的shell。示例如下所示:
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”语法如何嵌套?
答:基础语法如下:
Q:6 shell脚本中“?标记的用途是什么?答:在写一个脚本时,如果你想要检查前一命令是否执行成功,在条件中使用?”标记的用途是什么?答:在写一个shell脚本时,如果你想要检查前一命令是否执行成功,在if条件中使用“?”可以来检查前一命令的结束状态。简单的例子如下:
如果结束状态是0,说明前一个命令执行成功。
如果结束状态不是0,说明命令执行失败。
Q:7 在shell脚本中如何比较两个数字 ?
答:在if-then中使用测试命令( -gt 等)来比较两个数字,例子如下:
Q:8 shell脚本中break命令的作用 ?
答:break命令一个简单的用途是退出执行中的循环。我们可以在while和until循环中使用break命令跳出循环。
Q:9 shell脚本中continue命令的作用 ?
答:continue命令不同于break命令,它只跳出当前循环的迭代,而不是整个循环。continue命令很多时候是很有用的,例如错误发生,但我们依然希望继续执行大循环的时候。
Q:10 告诉我shell脚本中Case语句的语法 ?
答:基础语法如下:
Q:11 shell脚本中while循环语法 ?
答:如同for循环,while循环只要条件成立就重复它的命令块。不同于for循环,while循环会不断迭代,直到它的条件不为真。基础语法:
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循环的基础语法:
Q:15 如何调试shell脚本 ?
答:使用’-x’参数(sh -x myscript.sh)可以调试shell脚本。另一个种方法是使用‘-nv’参数( sh -nv myscript.sh)。
Q:16 shell脚本如何比较字符串?
答:test命令可以用来比较字符串。测试命令会通过比较字符串中的每一个字符来比较。
Q:17 Bourneshell(bash) 中有哪些特殊的变量 ?
答:下面的表列出了Bourne shell为命令行设置的特殊变量。
Q:18 在shell脚本中,如何测试文件 ?
答:test命令可以用来测试文件。基础用法如下表格:
Q:19 在shell脚本中,如何写入注释 ?
答:注释可以用来描述一个脚本可以做什么和它是如何工作的。每一行注释以#开头。例子如下:
Q:20 如何让 shell 就脚本得到来自终端的输入?
答:read命令可以读取来自终端(使用键盘)的数据。read命令得到用户的输入并置于你给出的变量中。例子如下:
Q:21 如何取消变量或取消变量赋值 ?
答:“unset”命令用于取消变量或取消变量赋值。语法如下所示:
# unset
Q:22 如何执行算术运算 ?
Q:23 do-while语句的基本格式 ?
答:do-while语句类似于while语句,但检查条件语句之前先执行命令(LCTT 译注:意即至少执行一次。)。下面是用do-while语句的语法
do
{
statements
} while (condition)
Q:24 在shell脚本如何定义函数呢 ?
答:函数是拥有名字的代码块。当我们定义代码块,我们就可以在我们的脚本调用函数名字,该块就会被执行。示例如下所示:
$ diskusage() { df -h ; }
Q:25 如何在shell脚本中使用BC(bash计算器) ?
答:使用下列格式,在shell脚本中使用bc:
variable=`echo“options; expression” | bc`
Q:26取出/etc/passwd文件中shell出现的次数
解析:
使用awk根据冒号分割内容,打印输出分割后的第7列,也就是shell所在列。然后调用sort命令排序并使用uniq -c统计每个shell出现的次数。
问题:
employee文件中记录了工号和姓名
bonus文件中记录工号和工资
要求把两个文件合并并输出如下
处理结果:
参考答案:
解析:
这里用到好几个命令,包括paste,awk,tr以及sort。paste命令用于合并多个文件的同行数据,如上面两个文件employee和bonus调用paste后合并成
paste命令可以使用-d指定合并时加入的符号。比如paste -d : employee bonus则结果变成类似100 Jason Smith :100$5,000 等。默认合并符号为tab符号。
awk用于提取除了tab符号的其余4列。
tr命令用于将字符串中所有大写字符转换为小写字符。
sort命令对字符排序。sort -k 2表示按文件第2个域排序,这里第二个域为姓名,所以是按姓名升序排序。如果要降序排列,则要用sort -k 2r。
问题:打印本机交换分区大小,输出如下
参考答案:
解析:
top 命令显示系统资源占用情况,-n 1表示只调用1次。
grepSwap选取Swap所在行。grep命令执行后结果可能如下:
sed命令用于字符串的一些正则匹配,这里使用了替换参数,将第1个k以及后面的字符替换成了空白。这样,sed执行后,结果为:
awk命令输出内容,对第二个参数除以1000.
问题:清除本机除了当前登陆用户以外的所有用户。
参考答案:
解析:
who-u显示所有当前用户。grep -v选取当前登录用户以外的所有用户。awk打印用户进程ID。sort -u会删除相同的行。最后用kill命令终止。
Q:30. 写一个shell脚本来得到当前的日期,时间,用户名和当前工作目录。
答案 : 输出用户名,当前日期和时间,以及当前工作目录的命令就是logname,date,who i am和pwd。
现在,创建一个名为userstats.sh文件,将下面的代码添加到它。
复制代码代码如下:
#!/bin/bash
echo "Hello, $LOGNAME"
echo "Current date is `date`"
echo "User is `who i am`"
echo "Current directory `pwd`"
给它添加执行权限,并且执行他。
复制代码代码如下:
# chmod 755 userstats.sh
# ./userstats.sh
Q:31 写一个shell脚本,进行两个数字的相加,如果没有输入参数就输出错误信息和一行使用说明
答案 : 下面是简单的shell脚本以及描述,如果没有命令行参数,它会抛出错误与如何使用脚本的说明。
再创建一个名为twonumbers.sh文件和下面的内容添加到文件里。
#!/bin/bash
if [ $# -ne 2 ]
then
echo "Usage - $0 x y"
echo" Where x and y are two nos forwhich I will print sum"
exit 1
fi
echo "Sum of $1 and $2 is `expr $1 + $2`"
给他添加可执行权限,并且执行。
复制代码代码如下:
# chmod 755two-numbers.sh
情形一: 未输入两个数字作为命令行参数运行脚本,你将得到下面的输出。
样例输出
复制代码代码如下:
# ./two-numbers.sh
Usage - ./two-numbers.sh x y
Where x and y are two nos for whichI will print sum
情形二: 当数字存在时,你会得到如图所示的结果。
复制代码代码如下:
$ ./two-numbers.sh 4 5
Sum of 4 and 5 is 9
Q:32 .你需要打印一个给定的数字的反序,如输入10572,输出27501,如果没有输入数据,应该抛出错误和使用脚本说明。在此之前,告诉我你需要在这里使用的算法。
算法
1.输入的数字为n
2.赋值 rev=0,sd=0 (反向和单个数字设置为0)
3.n % 10, 将得到最左边的数字
4.反向数字可以用这个方法生成rev * 10 + sd
5.对输入数字进行右位移操作(除以10)
6.如果n > 0, 进入第三步,否则进行第七步
7.输出rev
现在,创建一个名为`numbers.sh`文件,并添加以下代码。
复制代码代码如下:
#!/bin/bash
if [ $# -ne 1 ]
then
echo "Usage: $0 number"
echo " I will findreverse of given number"
echo " For eg. $00123, I will print 3210"
exit 1
fi
n=$1
rev=0
sd=0
while [ $n -gt 0 ]
do
sd=`expr $n % 10`
rev=`expr $rev \* 10 + $sd`
n=`expr $n / 10`
done
echo "Reverse number is $rev"
授予对文件的执行权限,并运行如下所示的脚本。
复制代码代码如下:
# chmod 755 numbers.h
情形一: 当输入不包含命令行参数,你将得到下面的输出。
样例输出
复制代码代码如下:
./numbers.sh
Usage: ./numbers.sh number
I will find reverse of given number
For eg. ./2.sh 123, I will print 321
情形二: 正常输入
复制代码代码如下:
$ ./numbers.sh 10572
Reverse number is 27501
解答:我们需要使用‘exit’命令来实现以上描述的情境。‘exit’命令被强制输出非0值时,脚本会报错并退出。在Unix环境下的shell脚本中,0值表示成功执行。因此,在脚本终止前执行一个不带引号的‘exit -1’命令将使脚本中止。
创建以下一个名为“anything.sh”的脚本。
1. #!/bin/bash
2. echo"Hello"
3. exit -1
4. echo"bye"
解答:当我们需要删除文件中的指定行时,‘sed’命令可以用来解决该问题。
1. #sed -i '1 d'file.txt
Q:35 1) 如何向脚本传递参数 ?
./script argument
例子: 显示文件名称脚本
./show.sh file1.txt
cat show.sh
#!/bin/bash
echo $1
(LCTT 译注:谢谢某匿名访客的提醒,原题有误,修改之。)
2) 如何在脚本中使用参数 ?
第一个参数 : $1,第二个参数 : $2
例子 : 脚本会复制文件(arg1) 到目标地址(arg2)
./copy.sh file1.txt/tmp/
cat copy.sh
#!/bin/bash
cp $1 $2
3) 如何计算传递进来的参数 ?
$#
4) 如何在脚本中获取脚本名称 ?
$0
5) 如何检查之前的命令是否运行成功 ?
$?
6) 如何获取文件的最后一行 ?
tail-1
7) 如何获取文件的第一行 ?
head-1
8) 如何获取一个文件每一行的第三个元素 ?
awk'{print $3}'
9) 假如文件中每行第一个元素是 FIND,如何获取第二个元素
awk'{ if ($1 =="FIND") print $2}'
10) 如何调试 bash 脚本
将 -xv 参数加到 #!/bin/bash 后
例子:
#!/bin/bash –xv
11) 举例如何写一个函数 ?
function example {
echo "Helloworld!"
}
12) 如何向连接两个字符串 ?
V1="Hello"
V2="World"
V3=${V1}${V2}
echo $V3
输出
HelloWorld
13) 如何进行两个整数相加 ?
V1=1
V2=2
let V3=$V1+$V2
echo $V3
输出
3
据 @kashu 的意见,本题的更佳回答为:
两个整数相加,还有若干种方法实现:
A=5
B=6
echo$(($A+$B)) # 方法 2
echo $[$A+$B] # 方法 3
expr $A + $B # 方法 4
echo $A+$B |bc # 方法 5
awk 'BEGIN{print'"$A"'+'"$B"'}' # 方法 6
14) 如何检查文件系统中是否存在某个文件 ?
if [ -f/var/log/messages ]
then
echo "Fileexists"
fi
15) 写出 shell 脚本中所有循环语法 ?
for 循环 :
foriin$(ls);do
echo item:$i
done
while 循环 :
#!/bin/bash
COUNTER=0
while [ $COUNTER-lt 10 ]; do
echo The counter is$COUNTER
let COUNTER=COUNTER+1
done
until 循环 :
#!/bin/bash
COUNTER=20
until [ $COUNTER-lt 10 ]; do
echo COUNTER$COUNTER
let COUNTER-=1
done
16) 每个脚本开始的 #!/bin/sh 或 #!/bin/bash 表示什么意思 ?
这一行说明要使用的 shell。#!/bin/bash 表示脚本使用 /bin/bash。对于 python 脚本,就是 #!/usr/bin/python。
17) 如何获取文本文件的第 10 行 ?
head -10 file|tail-1
18) bash 脚本文件的第一个符号是什么
#
19) 命令:[ -z"" ] && echo 0 || echo 1 的输出是什么
0
20) 命令 “export” 有什么用 ?
使变量在子 shell 中可用。
21) 如何在后台运行脚本 ?
在脚本后面添加 “&”。
据 @kashu 的意见,更好的答案是:
nohup command&
大部分时间我们可能是远程使用Linux,我碰到过由于网络断线使得在后台运行的command &没了...
22) "chmod500 script" 做什么 ?
使脚本所有者拥有可执行权限。
23) ">" 做什么 ?
重定向输出流到文件或另一个流。
24) & 和 && 有什么区别
· & - 希望脚本在后台运行的时候使用它
· && - 当前一个脚本成功完成才执行后面的命令/脚本的时候使用它
25) 什么时候要在 [ condition ] 之前使用 “if” ?
当条件满足时需要运行多条命令的时候。
26) 命令: name=John && echo 'My name is $name' 的输出是什么
variable
27) bashshell 脚本中哪个符号用于注释 ?
#
28) 命令: echo ${new:-variable} 的输出是什么
variable
29) ' 和 " 引号有什么区别 ?
· ' - 当我们不希望把变量转换为值的时候使用它。
· " - 会计算所有变量的值并用值代替。
30) 如何在脚本文件中重定向标准输出和标准错误流到 log.txt 文件 ?
在脚本文件中添加 "exec >log.txt 2>&1" 命令。
31) 如何只用 echo 命令获取字符串变量的一部分 ?
echo${variable:x:y}
x - 起始位置
y - 长度
例子:
variable="Myname is Petras, and I am developer."
echo${variable:11:6} # 会显示 Petras
32) 如果给定字符串 variable="User:123:321:/home/dir",如何只用 echo 命令获取 home_dir ?
echo${variable#*:*:*:}
或
echo${variable##*:}
33) 如何从上面的字符串中获取 “User” ?
echo${variable%:*:*:*}
或
echo${variable%%:*}
34) 如何使用 awk 列出 UID 小于 100 的用户 ?
awk -F: '$3<100'/etc/passwd
35) 写程序为用户计算主组数目并显示次数和组名
cat /etc/passwd|cut-d: -f4|sort|uniq -c|while read c g
do
{ echo $c; grep :$g:/etc/group|cut -d: -f1;}|xargs -n 2
done
36) 如何在 bash shell 中更改标准的域分隔符为 ":" ?
IFS=":"
37) 如何获取变量长度 ?
${#variable}
38) 如何打印变量的最后 5 个字符 ?
echo ${variable:-5}
39) ${variable:-10} 和 ${variable: -10} 有什么区别?
· ${variable:-10} - 如果之前没有给 variable 赋值则输出 10;如果有赋值则输出该变量
· ${variable:-10} - 输出 variable 的最后 10 个字符
40) 如何只用 echo 命令替换字符串的一部分 ?
echo${variable//pattern/replacement}
41) 哪个命令将命令替换为大写 ?
tr '[:lower:]''[:upper:]'
42) 如何计算本地用户数目 ?
wc -l/etc/passwd|cut -d" " -f1 或者 cat /etc/passwd|wc -l
43) 不用 wc 命令如何计算字符串中的单词数目 ?
set ${string}
echo $#
44) "export$variable" 或 "exportvariable" 哪个正确 ?
export variable
45) 如何列出第二个字母是 a 或 b 的文件 ?
ls -d ?[ab]*
46) 如何将整数 a 加到 b 并赋值给 c ?
c=$((a+b))
或
c=`expr $a + $b`
或
c=`echo"$a+$b"|bc`
47) 如何去除字符串中的所有空格 ?
echo $string|tr -d" "
48) 重写这个命令,将输出变量转换为复数: item="car"; echo "I like$item" ?
item="car";echo "I like ${item}s"
49) 写出输出数字 0 到 100 中 3 的倍数(0 3 6 9 …)的命令 ?
for i in{0..100..3}; do echo $i; done
或
for (( i=0;i<=100; i=i+3 )); do echo "Welcome $i times"; done
50) 如何打印传递给脚本的所有参数 ?
echo $*
或
echo $@
51) [ $a== $b ] 和 [ $a -eq $b ] 有什么区别
· [ $a== $b ] - 用于字符串比较
· [ $a-eq $b ] - 用于数字比较
52) = 和 == 有什么区别
· = - 用于为变量赋值
· == - 用于字符串比较
53) 写出测试 $a 是否大于 12 的命令 ?
[ $a -gt 12 ]
54) 写出测试 $b 是否小于等于 12 的命令 ?
[ $b -le 12 ]
55) 如何检查字符串是否以字母 "abc" 开头 ?
[[ $string == abc*]]
56) [[$string == abc* ]] 和 [[ $string =="abc*" ]] 有什么区别
· [[$string == abc* ]] - 检查字符串是否以字母 abc 开头
· [[$string == "abc" ]] - 检查字符串是否完全等于 abc
57) 如何列出以 ab 或 xy 开头的用户名 ?
egrep"^ab|^xy" /etc/passwd|cut -d: -f1
58) bash 中 $! 表示什么意思 ?
后台最近执行命令的 PID.
59) $? 表示什么意思 ?
前台最近命令的结束状态。
60) 如何输出当前 shell 的 PID ?
echo $$
61) 如何获取传递给脚本的参数数目 ?
echo $#
(LCTT 译注:和第3题重复了。)
*62) $* 和 $@ 有什么区别*
· $* - 以一个字符串形式输出所有传递到脚本的参数
· $@ - 以 $IFS 为分隔符列出所有传递到脚本中的参数
63) 如何在 bash 中定义数组 ?
array=("Hi""my" "name" "is")
64) 如何打印数组的第一个元素 ?
echo ${array[0]}
65) 如何打印数组的所有元素 ?
echo ${array[@]}
66) 如何输出所有数组索引 ?
echo ${!array[@]}
67) 如何移除数组中索引为 2 的元素 ?
unset array[2]
68) 如何在数组中添加 id 为 333 的元素 ?
array[333]="New_element"
69) shell 脚本如何获取输入的值 ?
a) 通过参数
./script param1param2
b) 通过 read 命令
read -p"Destination backup Server : " desthost
70) 在脚本中如何使用 "expect" ?
/usr/bin/expect<< EOD
spawn rsync -ar${line} ${desthost}:${destpath}
expect"*?assword:*"
send"${password}\r"
expect eof
EOD
Q:36 linux 系统开机启动过程
http://www.cnblogs.com/unicode/archive/2010/06/12/1756755.html
1.载入BOIS硬件信息,获取第一启动设备代号;
2.读取第一启动设备MBR中的引导加载程序(Bootloader),如grub lilo spfdisk等等;
3.引导加载程序从磁盘文件中载入操作系统核心;
4.启动管理器upstart启动预定义runlevel的系统服务;
5.启动管理器启动用户定义程序。
6.启动终端和(/或)x-window,等待用户登录。