1 Shell脚本是什么、它是必需的吗?
答:一个Shell脚本是一个文本文件,包含一个或多个命令。作为系统管理员,我们经常需要使用多个命令来完成一项任务,我们可以添加这些所有命令在一个文本文件(Shell脚本)来完成这些日常工作任务。
2 什么是默认登录shell,如何改变指定用户的登录shell
答:在Linux操作系统,“/bin/bash”是默认登录shell,是在创建用户时分配的。使用chsh命令可以改变默认的shell。示例如下所示:
3 可以在shell脚本中使用哪些类型的变量?
答:在shell脚本,我们可以使用两种类型的变量:
-
系统定义变量
-
用户定义变量
系统变量是由系统系统自己创建的。这些变量通常由大写字母组成,可以通过“set”命令查看。
用户变量由系统用户来生成和定义,变量的值可以通过命令“echo $<变量名>
”查看。
4 如何将标准输出和错误输出同时重定向到同一位置?
答:这里有两个方法来实现:
方法一:
2>&1 (# ls /usr/share/doc > out.txt 2>&1 )
方法二:
&> (# ls /usr/share/doc &> out.txt )
5 shell脚本中“if”语法如何嵌套?
答:基础语法如下:
6 shell脚本中“$?”标记的用途是什么?
答:在写一个shell脚本时,如果你想要检查前一命令是否执行成功,在if条件中使用“$?”可以来检查前一命令的结束状态。简单的例子如下:
如果结束状态是0,说明前一个命令执行成功。
7 在shell脚本中如何比较两个数字 ?
答:在if-then中使用测试命令( -gt 等)来比较两个数字
8 shell脚本中break命令的作用 ?
答:break命令一个简单的用途是退出执行中的循环。我们可以在while和until循环中使用break命令跳出循环。
9 shell脚本中continue命令的作用 ?
答:continue命令不同于break命令,它只跳出当前循环的迭代,而不是整个循环。continue命令很多时候是很有用的,例如错误发生,但我们依然希望继续执行大循环的时候。
10 告诉我shell脚本中Case语句的语法 ?
答:基础语法如下:
11 shell脚本中while循环语法 ?
答:如同for循环,while循环只要条件成立就重复它的命令块。不同于for循环,while循环会不断迭代,直到它的条件不为真。基础语法:
12 如何使脚本可执行 ?
答:使用chmod命令来使脚本可执行。例子如下:
# chmod a+x myscript.sh
13 “#!/bin/bash”的作用 ?
答:#!/bin/bash是shell脚本的第一行,称为释伴(shebang)行。这里#符号叫做hash,而! 叫做 bang。它的意思是命令通过 /bin/bash 来执行。
14 shell脚本中for循环语法 ?
答:for循环的基础语法:
15 如何调试shell脚本 ?
答:使用’-x’参数(sh -x myscript.sh)可以调试shell脚本。另一个种方法是使用‘-nv’参数( sh -nv myscript.sh)。
16 shell脚本如何比较字符串?
答:test命令可以用来比较字符串。测试命令会通过比较字符串中的每一个字符来比较。
17 Bourne shell(bash) 中有哪些特殊的变量 ?
答:下面的表列出了Bourne shell为命令行设置的特殊变量。
18 在shell脚本中,如何测试文件 ?
答:test命令可以用来测试文件。基础用法如下表格:
19 在shell脚本中,如何写入注释 ?
答:注释可以用来描述一个脚本可以做什么和它是如何工作的。每一行注释以#开头。例子如下:
20 如何让 shell 就脚本得到来自终端的输入?
答:read命令可以读取来自终端(使用键盘)的数据。read命令得到用户的输入并置于你给出的变量中。例子如下:
21 如何取消变量或取消变量赋值 ?
答:“unset”命令用于取消变量或取消变量赋值。语法如下所示:
# unset <Name_of_Variable>
22 如何执行算术运算 ?
答:有两种方法来执行算术运算:
使用expr
命令(# expr 5 + 2) 2.用一个美元符号和方括号($[ 表达式 ]
)例如:test=$[16 + 4] ; test=$[16 + 4]
23 do-while语句的基本格式 ?
答:do-while语句类似于while语句,但检查条件语句之前先执行命令(LCTT 译注:意即至少执行一次。)。下面是用do-while语句的语法
do { statements } while (condition)
24 在shell脚本如何定义函数呢 ?
答:函数是拥有名字的代码块。当我们定义代码块,我们就可以在我们的脚本调用函数名字,该块就会被执行。示例如下所示:
$ diskusage () { df -h ; }
25 如何在shell脚本中使用BC(bash计算器) ?
答:使用下列格式,在shell脚本中使用bc:
variable=`echo “options; expression” | bc`
26 在shell脚本成功执行前,如何中断脚本执行?
我们需要使用‘exit’命令来实现以上描述的情境。‘exit’命令被强制输出非0值时,脚本会报错并退出。在Unix环境下的shell脚本中,0值表示成功执行。因此,在脚本终止前执行一个不带引号的‘exit -1’命令将使脚本中止。
#!/bin/bash echo "Hello" exit -1 echo "bye"
27 如何使用Linux命令来移除文件头?
当我们需要删除文件中的指定行时,‘sed’命令可以用来解决该问题。下面是用来删除文件头(文件的首行)的正确命令。
# 内建的‘-i’开关就不需要重定向符sed -i '1 d' file.txt
28 你怎么检查一个文本文件中某一行的长度?
‘sed -n ‘n p’ file.txt’可以解决,这里‘n’表示行号,‘p’打印出匹配内容(到标准输出),该命令通常与-n命令行选项连用。那么,怎样来获取长度计数呢?很明显,我们需要通过管道输出给‘wc’命令来计算。
# sed -n '5 p' tecmint.txt | wc -c
29 可以在Linux系统上查看到所有非打印字符吗?你是怎么做到的?
可以。可以在Linux中查看所有的非打印字符。要实现上面所讲的方案,我们需要‘vi’编辑器的帮助。 怎样在‘vi’编辑器中显示非打印字符?
- 打开vi编辑器。
- 先按[esc]键,然后按‘:’进入到vi编辑器的命令模式。
- 最后,从‘vi’编辑器的命令界面输入set list命令并执行。
30 假如你是一个员工组的团队领导,为xyz公司工作。公司要求你创建一个‘dir_xyz’目录,让该组成员都能在该目录下创建或访问文件,但是除了文件创建者之外的其他人不能删除文件,你会怎么做?
# mkdir dir_xyz # chmod g+wx dir_xyz # chmod +t dir_xyz
第一行命令创建了一个目录(dir_xyz),上面的第二行命令让组(g)具有‘写’和‘执行’的权限,而上面的最后一行命令——权限位最后的‘+t’是‘粘滞位’,它用来替换‘x’,表明在这个目录中,文件只能被它们的拥有者、目录的拥有者或者是超级用户root删除。
31 你能告诉我一个Linux进程经历的各个阶段吗?
一个Linux进程在它的一生中,通常经历了四个主要阶段。
- 等待:Linux进程等待资源。
- 运行:Linux进程当前正在执行中。
- 停止:Linux进程在成功执行后或收到杀死进程信号后停止。
- 僵尸:如果该进程已经结束,但仍然留在进程表中,被称为‘僵尸’。
32 Linux中cut命令怎么用?
'cut’是一个很有用的Linux命令,当我们要截取文件的指定部分并打印到标准输出,当文本区域以及文件本身很大时,这个命令很有用。
-b:仅显示行中指定直接范围的内容; -c:仅显示行中指定范围的字符; -d:指定字段的分隔符,默认的字段分隔符为“TAB”; -f:显示指定字段的内容; -n:与“-b”选项连用,不分割多字节字符;
# cut -c1-10 txt_tecmint# cut -d;-f2 -f5 -f7 txt_tecmint
33 cmp’和‘diff’命令的区别是什么?
‘cmp’和‘diff’命令用来获取相同的东西,但各有侧重。diff’命令输出为了使两个文件一样而应该做的修改。而‘cmp’命令则将两个文件逐字节对比,并报告第一个不匹配的项。
34 可以用‘echo’命令来替换‘ls’命令吗?
可以的。‘ls’命令可以用‘echo’命令来替代。‘ls’命令列出目录内容,从替换上述命令的角度讲,我们可以使用‘echo *’,两个命令的输出完全一样。
35 你可能听过inode吧。你能简要描述一下inode吗?
inode’是一个‘数据结构’,在Linux上用于文件标识。每个文件在Unix系统上有一个独立的‘inode’和一个‘唯一的’inode号。
1) 如何向脚本传递参数 ?
./script argument
例子: 显示文件名称脚本
1
2
3
4
|
. /show .sh file1.txt cat show.sh #!/bin/bash echo $1 |
(LCTT 译注:谢谢某匿名访客的提醒,原题有误,修改之。)
2) 如何在脚本中使用参数 ?
第一个参数 : $1,第二个参数 : $2
例子 : 脚本会复制文件(arg1) 到目标地址(arg2)
1
2
3
4
|
. /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) 举例如何写一个函数 ?
1
2
3
|
function example { echo "Hello world!" } |
12) 如何向连接两个字符串 ?
1
2
3
4
|
V1= "Hello" V2= "World" V3=${V1}${V2} echo $V3 |
输出
HelloWorld
13) 如何进行两个整数相加 ?
1
2
3
4
|
V1=1 V2=2 let V3=$V1+$V2 echo $V3 |
输出
3
据 @kashu 的意见,本题的更佳回答为:
两个整数相加,还有若干种方法实现:
1
2
3
4
5
6
7
|
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) 如何检查文件系统中是否存在某个文件 ?
1
2
3
4
|
if [ -f /var/log/messages ] then echo "File exists" fi |
15) 写出 shell 脚本中所有循环语法 ?
for 循环 :
1
2
3
|
foriin$( ls ); do echo item:$i done |
while 循环 :
1
2
3
4
5
6
|
#!/bin/bash COUNTER=0 while [ $COUNTER -lt 10 ]; do echo The counter is $COUNTER let COUNTER=COUNTER+1 done |
until 循环 :
1
2
3
4
5
6
|
#!/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) "chmod 500 script"
做什么 ?
使脚本所有者拥有可执行权限。
23) ">"
做什么 ?
重定向输出流到文件或另一个流。
24) &
和 &&
有什么区别
&
- 希望脚本在后台运行的时候使用它&&
- 当前一个脚本成功完成才执行后面的命令/脚本的时候使用它
25) 什么时候要在 [ condition ]
之前使用 “if”
?
当条件满足时需要运行多条命令的时候。
26) 命令: name=John && echo 'My name is $name'
的输出是什么
variable
27) bash shell
脚本中哪个符号用于注释 ?
#
28) 命令: echo ${new:-variable}
的输出是什么
variable
29) '
和 "
引号有什么区别 ?
'
- 当我们不希望把变量转换为值的时候使用它。"
- 会计算所有变量的值并用值代替。
30) 如何在脚本文件中重定向标准输出和标准错误流到 log.txt
文件 ?
在脚本文件中添加 "exec >log.txt 2>&1"
命令。
31) 如何只用 echo
命令获取字符串变量的一部分 ?
1
2
3
|
echo ${variable:x:y} x - 起始位置 y - 长度 |
例子:
1
2
|
variable= "My name 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) 写程序为用户计算主组数目并显示次数和组名
1
2
3
4
|
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 命令如何计算字符串中的单词数目 ?
1
2
|
set ${string} echo $ # |
44) "export $variable"
或 "export variable"
哪个正确 ?
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 param1 param2
b) 通过 read
命令
read -p "Destination backup Server : " desthost
70) 在脚本中如何使用 "expect"
?
1
2
3
4
5
6
|
/usr/bin/expect << EOD spawn rsync -ar ${line} ${desthost}:${destpath} expect "*?assword:*" send "${password}\r" expect eof EOD |