最近发现shell脚本在日常工做中简直算一把瑞士军刀,不少场景下用shell脚本能实现经常使用的简单需求,而以前都没怎么学习过shell,就趁机把shell相关的语法和常见用法总结了下,方便之后随时回顾和查阅!html
其实shell的开始并不难,第一行以下:
#!/bin/bash
叫作shebangpython
shell的注释以#
开头(这点相似于python)linux
变量定义:
var_test1="hello"
注意定义和赋值时=
两边均不能有空格,且变量名前不能加$
!
变量使用:
echo $var_test1
或
echo ${var_test1}
或:
str1="${var_test1} world"
加上美圆符后的变量会被替换为其值。注意,单引号中的不会被替换
命令执行结果赋给一个变量:web
lines=`wc -l 1.txt` # 这里lines就被赋值为了1.txt的行数 lines="$(wc -l 1.txt)" # 单撇号不能嵌套,可用这种包含在内
格式化变量:
echo 23 | awk {printf("%05d", $0);} # 00023
字符串截取:
${var_test1:1:3}
会获得"ell"
,即从索引1开始取3个字符正则表达式
shell中的输出直接用echo便可
echo "hello word"
等同于
echo hello word
shell
echo "What's your name?" read Name # 这里不须要声明新变量 echo Hello, $Name!
&&
: 知足短路原则,即第一个命令返回失败时不会执行第二个命令||
: 知足短路原则,即第一个命令返回成功时不会执行第二个命令;
: 不短路,无论第一个命令是否返回成功,均会执行第二个命令echo "Always executed" || echo "Only executed if first command fails" echo "Always executed" && echo "Only executed if first command does NOT fail" echo "Always executed1" ; echo "Always executed2"
cmd1 | cmd2
: 通道,第一个命令的输出会直接做为第二个命令的标准输入来执行
python test.py < in.txt
: 输入重定向,将从in.txt读取输入而非标准输入流
echo "hello" > out.txt
: 输出重定向,将前一个命令的输出定向到文件out.txt中(建立新文件并写入)
echo "hello" >> append.txt
: 输出重定向,将前一个命令的输出定向到文件append.txt中(追加到文件末尾)
echo "hello" 2> err.txt
: 错误流重定向,将前一个命令的错误流输出定向到文件err.txt中(建立新文件并写入)
echo "hello" 2>> err.txt
: 错误流重定向,将前一个命令的错误流输出定向到文件err.txt中(追加到文件末尾)bash
最简单的判断逻辑是:app
if [ condition1 ]; then statement1 elif [ condition2 ]; then statement2 else statement3 fi # 或者用test if test condition1 ; then statement1 fi
注意:condition两边与[]之间必须至少有一个空格less
shell中判断条件不一样于其余语言中的<,>,==,<=,>=那么直接,常见的有下列几种:oop
-eq 等于(==) -ne 不等于(!=) -gt 大于(\>) -lt 小于(\<) -le 小于等于 -ge 大于等于 -z 空串 -n 非空串 == 两个字符相等 != 两个字符不等
-a 且 -o 或
[[ ... ]]
[[ hello == hell? ]]
结果为真[[ ]]
条件判断结构中,可是若是出如今 [ ]
结构中的话,会报错。好比能够直接使用 if [[ $a != 1 && $a <= 5 ]]
, 若是不使用双括号, 则为 if [ $a -ne 1] && [ $a -le 5 ]
或者 if [ $a -ne 1 -a $a -le 5 ]
-f 判断后面是否为一个文件 -d 判断后面是否为一个目录 -e 判断后面对应的文件是否存在 -s 判断文件是否存在且不为空
case "$Variable" in # 列出须要匹配的字符串 0) echo "There is a zero.";; 1) echo "There is a one.";; *) echo "It is not null.";; esac
# {1..3} == `seq 1 3` for Variable in {1..3} do echo "$Variable" done
或传统的"for循环",但须要加两层括号(两层的小括号内能够写C语言中的语句):
for ((a=1; a <= 3; a++)) do echo $a done
在其余命令的结果上执行for循环:
for Output in $(ls) do cat "$Output" done
while [ condition ] do echo "loop body here..." break done
$# 命令行参数个数 $0 当前脚本名 $n 第n个参数值,n可取1,2,3... $@ 全部命令行参数 $? 上一个命令的返回值
# 打印每行中 ',' 以前内容 cut -d ',' -f 1 file.txt # 将 file.txt 文件全部 'okay' 替换为 'great', (兼容正则表达式) sed -i 's/okay/great/g' file.txt # shell中不支持浮点数除法运算,可以使用awk实现浮点除法 a=3 b=4 c=`awk 'BEGIN{printf "%.2f",('$a'/'$b')}'` # 单引号内的变量不能被替换,所以须要将变量单独放在引号外 sort -k 指定比较的列(从1开始) -n 数值比较 -r 倒序 -o filename 输出到文件(可用此选项输出到输入文件) -f 不区分大小写排序 -c 检查是否已排序好,若是未排序好则输出第一个未按序的行 -M 按月份排序 -b 忽略前导空格 -u 获得不重复的行 多列排序:sort -k1,1 -k3nr,3
命令 | 做用 |
---|---|
${#string} |
$string 的长度 |
${string:position} |
在$string 中, 从位置$position 开始提取子串 |
${string:position:length} |
在$string 中, 从位置$position 开始提取长度为$length 的子串 |
${string#substring} |
从变量$string 的开头, 删除最短匹配$substring 的子串 |
${string##substring} |
从变量$string 的开头, 删除最长匹配$substring 的子串 |
${string%substring} |
从变量$string 的结尾, 删除最短匹配$substring 的子串 |
${string%%substring} |
从变量$string 的结尾, 删除最长匹配$substring 的子串 |
${string/substring/replacement} |
使用$replacement , 来代替第一个匹配的$substring |
${string//substring/replacement} |
使用$replacement , 代替全部匹配的$substring |
${string/#substring/replacement} |
若是$string 的前缀匹配$substring , 那么就用$replacement 来代替匹配到的$substring |
${string/%substring/replacement} |
若是$string 的后缀匹配$substring , 那么就用$replacement 来代替匹配到的$substring |
a="one,two,three,four" # 要将$a分割开,能够这样: OLD_IFS="$IFS" IFS="," arr=($a) IFS="$OLD_IFS" for s in ${arr[@]} do echo "$s" done # 后面的能够直接改成只取${arr[1]}
加粗的部分支持管道接收标准输入流。更多命令可参考:Linux命令大全
本文简要介绍了下shell入门的一些用法,若有不对之处,欢迎你们指正。我也是不久前才开始真正使用shell,这是个强大而又复杂的语言,你们有比较好的shell资料也欢迎留言,共同窗习,谢谢!
转载请注明出处:Shell常见用法小记