常见问题
- 解释下$*和$@ 有什么区别,分别使用在什么场景?
- 如何查找文件夹下所有包含某个字符串的文件?
- shell 中$! 有什么作用?
- var=value和export赋值变量有什么区别?
- 如何调试shell脚本?
- 如何获取当前脚本的绝对路径?
背景
运维过程经常会使用shell获取数据或者做些自动化任务,当然也是面试运维工程师常常被问到的问题,为了方便记忆,我也整理了部份shell常问的知识点。
常见问题
1 解释下$*和$@ 有什么区别,分别使用在什么场景?
1.1 区别
- $*和$@ 在不加双引号的时候作用类似,都可以通过循环遍历出每个参数。
- $*和$@ 在加双引号的时候作用不一样,$*会将传递的参数看成一个整体,循环遍历的时候也是一个整体字符串;$@则会将传递参数通过空格分隔,可以通过循环遍历每个参数。
1.2应用场景
- 当需要传递参数作为一个整体的时候可以使用"$*",如果需要分别获取每个参数的时候使用“$@”
$ cat > test.sh << EOF
#!/bin/bash
echo '没有引用双引号 $*'
for var in $*
do
echo "$var"
done
echo '没有引用双引号 $@'
for var in $@
do
echo "$var"
done
echo '--------------------------'
echo "引用了双引号 \"\$*\""
for var in "$*"
do
echo "$var"
done
echo "引用了双引号 \"\$@\""
for var in "$@"
do
echo "$var"
done
EOF
# 执行一下
$ bash test.sh 1 2 3
没有引用双引号 $*
1
2
3
没有引用双引号 $@
1
2
3
--------------------------
引用了双引号 "$*"
1 2 3
引用了双引号 "$@"
1
2
3
2. 如何查找文件夹下所有包含某个字符串的文件?
我的第一印象是使用grep命令,因为最终查找的是文件上的字符串,所以这里要使用到递归,grep 有个参数选项-r 就是递归查找某个字符串所在的文件。下面介绍我常用的grep参数:
- -i : 不区分查找字符串的大小写
- -r:递归查找文件夹中包含某字符串的文件信息及所在行
- -n: 匹配到字符串的所在行数
2.2 应用场景:
- 常常用于查找相关日志或者帮助信息。
[root@node-1 test-shell]# grep -irn "echo" /root/test-shell
/root/test-shell/test.sh:3:echo '没有引用了双引号 $*'
/root/test-shell/test.sh:6:echo "$var"
/root/test-shell/test.sh:9:echo '没有引用了双引号 $@'
/root/test-shell/test.sh:12:echo "$var"
/root/test-shell/test.sh:15:echo '--------------------------'
/root/test-shell/test.sh:17:echo "引用了双引号 \"\$*\""
/root/test-shell/test.sh:20:echo "$var"
/root/test-shell/test.sh:23:echo "引用了双引号 \"\$@\""
/root/test-shell/test.sh:26:echo "$var"
/root/test-shell/test1.sh:3:echo '没有引用了双引号 '
/root/test-shell/test1.sh:6:echo ""
/root/test-shell/test1.sh:9:echo '没有引用了双引号 '
/root/test-shell/test1.sh:12:echo ""
/root/test-shell/test1.sh:15:echo '--------------------------'
/root/test-shell/test1.sh:17:echo "引用了双引号 \"$*\""
/root/test-shell/test1.sh:20:echo ""
/root/test-shell/test1.sh:23:echo "引用了双引号 \"$@\""
/root/test-shell/test1.sh:26:echo ""
3. shell 中$! 有什么作用?
当你在shell中启动一个后台进程时,shell会立即返回到命令提示符,而不会等待该进程完成。此时,你可以使用$!来获取你刚刚放入后台的那个进程的PID。
3.1 应用场景
- 一般通过获取pid用于监控后台进程的运行状态
$ cat > test1.sh << EOF
#!/bin/bash
# 启动一个后台进程
sleep 10 &
# 使用$!获取刚刚启动的后台进程的PID
PID=$!
echo "后台进程的PID是: $PID"
# 现在你可以使用PID来做一些事情,比如等待该进程完成
wait $PID
echo "后台进程已完成"
EOF
$ bash test1.sh
后台进程的PID是: 2889
后台进程已完成
4. var=value和export赋值变量有什么区别?
4.1 区别
- var=value和export var=value 变量赋值主要区别是作用域不同。
- 使用export var=value 进行变量赋值,该变量在该shell以及子进程中都可以访问到,还将其导出到了当前 shell 的环境变量中。
- var=vaule变量赋值,该变量不能够在子进程中共享。
4.2 应用场景
- 理解了变量的作用范围才能更好的声明变量,才能编写能够正常交互的脚本和命令行程序。
[root@node-1 test-shell]# var1='var1'
[root@node-1 test-shell]# export var2='var2'
$ cat > test3.sh << EOF
#!/bin/bash
echo 'print $var1'
echo $var1
echo 'print $var2'
echo $var2
EOF
[root@node-1 test-shell]# bash test3.sh
print $var1
print $var2
var2
5. 如何调试shell脚本?
我日常写shell脚本时大都使用的vim直接编写,没有像python、java等强大的代码编写工具等友好的提示,所以每写完一段代码都要运行调试。
常用的调试方式有以下几种:
使用bash -x 查看脚本的执行流程和变量的变化,也可以结合set在脚本内进行使用set -x是开始调试 set +x是结束调试
$ cat > test4.sh << EOF
#!/bin/bash
echo 123 > /tmp/123.txt
cat /tmp/123.txt
EOF
# 执行脚本时-x 打印的调试信息
[root@node-1 test-shell]# bash -x test4.sh
+ echo 123
+ cat /tmp/123.txt
123
# 在脚本内对指定代码进行调试, set -x是开始调试 set +x是结束调试
$ cat > test5.sh << EOF
#!/bin/bash
echo "调试前----------"
echo "开始调试----------"
set -x
# $$ 获取当前shell的进程id
echo $$
ps -aux | grep $$
set +x
echo "调试结束-------------"
EOF
6.如何获取当前脚本的绝对路径?
[root@node-1 test-shell]# pwd
/root/test-shell
$ cat > test6.sh << EOF
#!/bin/bash
echo "执行pwd命令输出的脚本目录: $(pwd)"
echo "执行readlink命令输出的脚步路径: $(readlink -f "$0")"
echo "执行dirname切换到脚本所在目录,在执行pwd命令输出的脚步所在目录: $(cd $(dirname "$0"); pwd)"
EOF
[root@node-1 ~]# pwd
/root
[root@node-1 ~]# bash /root/test-shell/test6.sh
执行pwd命令输出的脚本目录: /root
执行readlink命令输出的脚步路径: /root/test-shell/test6.sh
执行dirname切换到脚本所在目录,在执行pwd命令输出的脚步所在目录: /root/test-shell