title | date | tags | categories | ||
---|---|---|---|---|---|
Shell学习 |
2018-12-15 04:10:46 -0800 |
|
|
目录 start
目录 end|2019-06-02 11:58|
切换shell
chsh -s /bin/bash
- sh
- 大多Linux都支持的shell类别
- bash
- zsh
- 十分现代化 配置oh my zsh
- dash
- 它主要是为了执行脚本而出现,而不是交互,它速度更快,但功能相比bash要少很多,语法严格遵守POSIX标准
- 速度确实要快,输入上的交互确实交互不了
- fish
- 交互式的, 补全功能比较好
- 学习内建命令的使用
-
文件头部
#!/bin/sh
表示要使用sh解释器来执行, 可以替换成bash dash- 只要该文件具有执行权限就可以直接运行了
./a.sh
或者绝对路径
- 只要该文件具有执行权限就可以直接运行了
read answer
并且在处理管道的输入也是一样的使用 read
while read line; do
echo $line
done
echo printf
red='\033[0;31m'
green='\033[0;32m'
yellow='\033[0;33m'
blue='\033[0;34m'
purple='\033[0;35m'
cyan='\033[0;36m'
white='\033[0;37m'
default='\033[0m'
比Python的作用域更加恶心
# 实现了读取 A_host变量的值
perfix='A_'
name=${perfix}host
host=${!name}
${command}
echo ${command}|awk '{run=$0;system(run)}'
最好
- 自增:
- i=$(( $i + 1 )) dash sh 都有效
- ((a++))
i=`expr $i + 1`;
- let i+=1;
- i=$[$i+1];
- 取余
- i=$(( $i % 3))
- 四则运算 参考博客
- ((i=$j+$k)) 等价于 i=
expr $j + $k
- ((i=$j-$k)) 等价于 i=
expr $j -$k
- ((i=$j*$k)) 等价于 i=
expr $j \*$k
- ((i=$j/$k)) 等价于 i=
expr $j /$k
- ((i=$j+$k)) 等价于 i=
判断变量是否为数值
if [ "$1" -gt 0 ] 2>/dev/null ;then
echo "$1 is number."
else
echo 'no.'
fi
- 字符串截取 | Blog:变量字符串截取 | Shell正则
Pattern | 描述 | |
---|---|---|
${varible#*str} |
截取 首个 |
str 右 的字符串 |
${varible##*str} |
截取 最末 |
str 右 的字符串 |
${varible%%str*} |
截取 首个 |
str 左 的字符串 |
${varible%str*} |
截取 最末 |
str 左 的字符串 |
获取命令的输出
-
使用 保存结果的变量名=
需要执行的linux命令
这种方式来获取命令的输出时,注意的情况总结如下: -
1)保证反单引号内的命令执行时成功的,也就是所命令执行后$?的输出必须是0,否则获取不到命令的输出
-
2)即便是命令的返回值是0,也需要保证结果是通过标准输出来输出的,而不是标准错误输出,否则需要重定向
-
因此我们推荐使用 保存结果的变量名=
需要执行的linux命令 2>&1
的方式来获取命令的执行结果。 -
输出变量时:
$var
会丢失换行和空格"$var"
不会
字符串的包含问题
isGithub=`expr match "$line" ".*"$2`
# 简单的就是使用grep
isGithub=`echo $line | grep "github" `
# return 0 is $1 is substring of $2, otherwise 1
strIsSubstring(){
local x=1
case "$2" in
*$1*) x=0;;
esac
echo $x
}
求长 ${#var}
字符串拆分成数组
- 如果是空格分割的字符串
- 直接
for element in $target
- 直接
参考博客
命令行选项 参数处理
参数 | 说明 |
---|---|
$# |
传递到脚本的参数个数 |
$* |
以一个单字符串显示所有向脚本传递的参数。以"$1 $2 … $n"的形式输出所有参数。 |
$$ |
脚本运行的当前进程ID号 |
$! |
后台运行的最后一个进程的ID号 |
$@ |
与$*相同,但是使用时加引号 以"$1" "$2" … "$n" 的形式输出所有参数。 |
$- |
显示Shell使用的当前选项,与set命令功能相同。 |
$? |
显示最后命令的退出状态。0表示没有错误,其他任何值表明有错误。 |
读取脚本参数
# 1. 简单的方式
case $1 in
-h | h)
echo "help"
;;
*)
echo "default"
;;
esac
# 2. 规范化的参数
while getopts "hup:" opt; do
case "$opt" in
h)
usage
exit 0
;;
u)
UPCASE=true
;;
d)
DATE=$OPTARG
;;
esac
done
判断文件
-
文件
if [ -f path ]
-
链接
if [ -L path ]
-
目录
if [ -d path ]
-
整数比较
-eq
等于,如:if [ "$a" -eq "$b" ]-ne
不等于,如:if [ "$a" -ne "$b" ]-gt
大于,如:if [ "$a" -gt "$b" ]-ge
大于等于,如:if [ "$a" -ge "$b" ]-lt
小于,如:if [ "$a" -lt "$b" ]-le
小于等于,如:if [ "$a" -le "$b" ]大于
(需要双括号),如:(("$a" > "$b"))>=
大于等于(需要双括号),如:(("$a" >= "$b"))
case $content in
-h|h)
echo "help"
;;
*)
echo "前面全部不匹配才会执行"
;;
esac
简易循环
for i in $(seq 1 5); do
echo $i
done
i=1
while [ "$i" -le 10 ];do
echo $i
i=$(($i+1))
done
Shell的函数只能返回整型数据类型
simple(){
echo "simple"
}
while IFS= read -r -u3 line; do
echo "$line"
done 3< "$2"
[block]
name=myth
- 如果没有
[block]
这样的声明就可以当sh用, 直接 source file 就加载了配置内容
更为直观, 简单
学习怎么使用的话, 可以看上面的博客(虽然有点简陋), 但是如果是 oh-my-zsh 的用户, 可以直接看别人的插件, 模仿就行了, 例如 redis-cli 插件的自动补全, 就很简单直接
#compdef redis-cli rec
这第一行很重要, 定义了是对哪个命令或脚本的自动补全
- 获取当前shell脚本的绝对路径
basepath=$(cd `dirname $0`; pwd)
- 命令嵌套 只要在 命令中用 两个反引号 `` 将子命令包住即可
- 检查当前用户为Root用户
if [ $(id -u) != "0" ]; then printf $red"Please use root to run this script\n"$end exit 1 fi
- kill 脚本进程
id=`ps -ef | grep "WithRedis.py" | grep -v "grep" | grep -v "\-d" | awk '{print $2}'` if [ "${id}1" = "1" ];then printf $red"not exist background running script\n"$end else kill -9 $id fi
- 得到脚本绝对路径; 如果只是执行 pwd 只是得到执行脚本时的当前绝对路径而已
basepath=$(cd \`dirname $0\`; pwd)