- Shell是用户与内核进行交互操作的一种接口,目前最流行的Shell称为bash Shell
Shell也是一门编程语言<解释型的编程语言>,即shell脚本<就是在用linux的shell命令编程>。一个系统可以存在多个shell,可以通过cat /etc/shells命令查看系统中安装的shell,不同的shell可能支持的命令语法是不相同的 - Linux Shell中的变量分为“系统变量”和“用户自定义变量”可以通过set命令查看变量,unset取消变量,readonly B=2 声明静态的变量 B=2,不能 unset
- 语法: 1.等号两侧不能有空格
2.单引号会将变量引用比如 {hello}s world’ 输出结果为:${hello}s world - 变量总结:
1、a.sh中直接调用b.sh,会让b.sh在a所在的bash进程的“子进程”空间中执行
2、而子进程空间只能访问父进程中用export定义的变量
3、一个shell进程无法将自己定义的变量提升到父进程空间中去
5、source执行脚本时,会让脚本在调用者所在的shell进程空间中执行。例如 环境变量的配置 /etc/profile
4、“.”号执行脚本时,会让脚本在调用者所在的shell进程空间中执行 - 反引号赋值
A=ls -la
## 反引号,运行里面的命令,并把结果返回给变量A
A=$(ls -la) ## 等价于反引号 - 特殊变量
$? 表示上一个命令退出的状态码
$$ 表示当前进程编号
$0 表示当前脚本名称
$n 表示n位置的输入参数(n代表数字,n>=1)
$# 表示参数的个数,常用于循环
@ 都表示参数列表 .`
注:
@区别
$* 和 $@ 都表示传递给函数或脚本的所有参数
不被双引号" “包含时——
$* 和 $@ 都以$1 $2 …
*” 会将所有的参数作为一个整体,以"$1 $2 …
@" 会将各个参数分开,以"$1" “
n” 的形式组成一个参数列表
7. 算术运算: 用expr,格式 expr m + n 或
[]注意expr运算符间要有空格
8. 流程控制:
#!/bin/bash
read -p “please input your name:” NAME ## read命令用于从控制台读取输入数据
printf ‘%s\n’ $NAME
if [ $NAME = root ]
then
echo “hello ${NAME}, welcome !”
elif [ $NAME = zhangsan ]
then
echo “hello ${NAME}, welcome !”
else
echo “SB, get out here !”
fi
判断条件:[ ]内部的=周边的空格,有区别
[] 与[[ ]] 的区别:[[ ]] 中逻辑组合可以使用 && || 符号, 而[] 里面逻辑组合可以用 -a -o
if [ a = b ];then echo ok;else echo notok;fi
if [ a=b ];then echo ok;else echo notok;fi
字符串比较:= !=
-z 字符串长度是为0返回true
-n 字符串长度是不为0返回true
if [ ‘aa’ = ‘bb’ ]; then echo ok; else echo notok;fi
if [ -n “aa” ]; then echo ok; else echo notok;fi
if [ -z “” ]; then echo ok; else echo notok;fi
整数比较:(待测试)
-lt 小于
-le 小于等于
-eq 等于
-gt 大于
-ge 大于等于
-ne 不等于
文件判断:
-d 是否为目录
if [ -d /bin ]; then echo ok; else echo notok;fi
-f 是否为文件
if [ -f /bin/ls ]; then echo ok; else echo notok;fi
-e 是否存在
if [ -e /bin/ls ]; then echo ok; else echo notok;fi
while语法
1.
while expression
do
command
…
done
2.
i=1
while ((i<=3))
do
echo $i
let i++
done
case语法
case $1 in
start)
echo “starting”
;;
stop)
echo “stoping”
;;
*)
echo “Usage: {start|stop}”
esac
for语法
for N in 1 2 3; do echo $N; done
for N in {1…3}; do echo $N; done
for ((i = 0; i <= 5; i++)); do echo “welcome $i times”; done
- 函数的使用
定义:
#!/bin/sh
func1.sh
hello() ## 函数定义
{
echo "Hello there today’s date is date +%Y-%m-%d
"
# return 2 ###返回值其实是状态码,只能在[0-255]范围内
}
hello
echo $? 获取函数的return值
echo “now going to the function hello”
echo “back from the function”
返回值:
#!/bin/bash
fun2.sh
funWithReturn(){
echo “这个函数会对输入的两个数字进行相加运算…”
echo "输入第一个数字: "
read aNum
echo "输入第二个数字: "
read anotherNum
echo “两个数字分别为 $aNum 和 $anotherNum !”
return
aNum+$anotherNum))
}
funWithReturn
echo “输入的两个数字之和为 $? !”
跨脚本调用函数
在调用脚本的前面使用. 或者source
- shell编程综合练习
自动化软件部署脚本
需求:
公司内有一个N个节点的集群,需要统一安装一些软件(jdk)需要开发一个脚本,实现对集群中的N台节点批量自动下载、安装jdk
思路: - 编写一个启动脚本,用来发送一个软件安装脚本到每一台机器
- 然后启动每台机器上的软件安装脚本来执行软件下载和安装
expect的安装:
链接:https://blog.csdn.net/jdbc/article/details/42001695 (提前得安装gcc)
准备一个jdk安装包放在内网web服务器上 httpd /var/www/html/software
启动脚本:
=====================================================================================================
#!/bin/bash
boot.sh
待安装的机器
SERVERS=“mini1 mini2”
登录密码
PASSWORD=hadoop
母机地址
BASE_SERVER=192.168.153.11
实现免密登陆配置的函数
auto_ssh_copy_id() {
# 注意:expect路径问题
/usr/expect/bin/expect -c “set timeout -1;
spawn ssh-copy-id $1;
expect {
(yes/no) {send – yes\r;exp_continue;}
password: {send – $2\r;exp_continue;}
eof {exit 0;}
}”;
}
ssh_copy_id_to_all() {
for SERVER in $SERVERS
do
auto_ssh_copy_id $SERVER $PASSWORD
done
}
调用免密登陆配置函数,实现母机到各子机的免密登陆配置
ssh_copy_id_to_all
完成分发install.sh到各子机的操作
并让子机启动install.sh
for SERVER in
SERVER:/root
ssh root@$SERVER /root/install.sh
done
=====================================================================================================
安装脚本:
#!/bin/bash
install.sh
BASE_SERVER=192.168.153.11
为本机安装wget命令
yum install -y wget
使用wget从母鸡的web服务器上下载jdk压缩包
wget $BASE_SERVER/software/jdk-8u201-linux-x64.tar.gz
将下载的压缩包解压
tar -zxvf jdk-8u201-linux-x64.tar.gz -C /usr/local
修改profile配置文件
cat >> /etc/profile << EOF
export JAVA_HOME=/usr/local/jdk1.8.0_201
export PATH=$PATH:$JAVA_HOME/bin
EOF