假期周进度报告2
shell编程
基础
-
命令语言,程序设计语言,用于访问操作系统内核服务
-
常见种类:bash
-
文件后缀:.sh,拓展名不影响脚本执行
-
linux命令也是shell语句
-
运行:
# 方法一:作为可执行程序 $ chmod +x ./test.sh #使脚本具有执行权限 $ ./test.sh #执行脚本 运行二进制一定要写成 ./test.sh,而不是 test.sh test.sh系统会去 PATH 里寻找test.sh ./test.sh系统就在当前目录找 # 方法二:作为解释器参数 /bin/sh test.sh /bin/php test.php 这种方法不需要在文件首行指定解释器
-
严格规定空格
变量和符号(+-*/等)、命令(echo,expr等)之间有空格
赋值没有空格
-
严格使用符号
''不转义字符串
“”转义字符串
``命令
{}变量边界区分
[]符号优先级
数据类型
变量
类型
- 1) 局部变量 脚本或命令中定义,当前shell实例中有效
- 2) 环境变量 所有的程序,包括shell启动的程序,都能访问环境变量,有些程序需要环境变量来保证其正常运行。必要的时候shell脚本也可以定义环境变量。
- 3) shell变量 shell变量是由shell程序设置的特殊变量。shell变量中有一部分是环境变量,有一部分是局部变量,这些变量保证了shell的正常运行
声明
# 显式赋值
your_name="runoob.com"
> 变量名和等号之间不能有空格
> 命名只能使用英文字母,数字和下划线,首个字符不能以数字开头。
> 中间不能有空格,可以使用下划线(_)。
> 不能使用标点符号。
> 不能使用bash里的关键字(可用help命令查看保留关键字)
# 语句赋值
for file in `ls /etc`
for file in $(ls /etc)
# 只读变量
your_name="runoob.com"
readonly yourname
使用
# 使用一个定义过的变量,$变量
echo $your_name
echo ${your_name}
echo "I am good at ${skill}Script"
> {}是为了帮助解释器识别变量的边界
> 赋值时不用加$
删除
unset 变量
> 只读变量不能删除
linux命令赋给变量
alias new_name='command'
字符串
-
单引号:原样输出
-
双引号:可以有变量,里面的其他引号\转义
-
拼接:单套单,双套双
your_name="runoob" # 使用双引号拼接 √ greeting="hello, "$your_name" !" √ greeting_1="hello, ${your_name} !" echo $greeting $greeting_1 # 使用单引号拼接 √ greeting_2='hello, '$your_name' !' × greeting_3='hello, ${your_name} !' echo $greeting_2 $greeting_3 输出结果为: hello, runoob ! hello, runoob ! hello, runoob ! hello, ${your_name} !
-
长度
${#变量}
-
切片
${变量:start:end}
-
查找字符
`expr index "$string" io` `是反引号,而不是单引号' 只能查找单个字符位置,io哪个先出现计算哪个
-
特殊转移字符
\n 换行 \c 不换行
数组
-
只支持一维
-
赋值
变量=(v1 v2 v3) 变量[下标]=value
-
读取
${数组[下标]} ${数组[@/*]} @/*表示所有元素
-
长度
# 数组长度 ${#数组[@/*]} # 元素长度 ${#数组[下标]}
-
拼接
array_new=(${array1[@]} ${array2[@]}) array_new=(${array1[*]} ${array2[*]})
-
删除
unset array_name[index]
高级
输入
# 从标准输入中读取一行指定给 shell 变量
read 变量
输出
echo
echo [-e] 变量/字符串 [> file]
# 可选参数
-e 开启转义
>重定向
printf
# printf
printf format-string [arguments...]
format-string: 为格式控制字符串
arguments: 为参数列表。
格式化字符串
eg "%-10s %-8s %-4.2f\n" 郭芙 女 47.9876
分析 -:左对齐,没有默认右对齐
10:宽度
s:字符类型
# 传入参数多余占位符
eg printf %s abc def
显示 abcdef
eg "%s %s %s\n" a b c d e f g h i j
显示 a b c
d e f
g h i
# 传入参数不够
%s 用NULL代替
%d 用 0 代替
转义
\a 警告字符,通常为ASCII的BEL字符
\b 后退
\c 抑制(不显示)输出结果中任何结尾的换行字符(只在%b格式指示符控制下的参数字符串中有效),而且,任何留在参数里的字符、任何接下来的参数以及任何留在格式字符串中的字符,都被忽略
\f 换页(formfeed)
\n 换行
\r 回车(Carriage return)
\t 水平制表符
\v 垂直制表符
\\ 一个字面上的反斜杠字符
\ddd 表示1到3位数八进制值的字符。仅在格式字符串中有效
\0ddd 表示1到3位的八进制值字符
传参
# 脚本中
$num
# 执行时
./...sh args1 args2
$# 传递到脚本的参数个数
$$ 脚本运行的当前进程ID号
$! 后台运行的最后一个进程的ID号
$- 显示Shell使用的当前选项,与set命令功能相同。
$? 显示最后命令的退出状态。0表示没有错误,其他任何值表明有错误。
$* 以一个字符串显示所有向脚本传递的参数。
"$*"--"$1 $2 … $n"的形式输出所有参数。
$@ 使用时加引号,并在引号中返回每个参数。
"$@"--"$1" "$2" … "$n"的形式输出所有参数。
运算
-
算数运算符
-
条件运算符
-
关系运算符
-
布尔运算符
-
字符串运算符
-
文件测试运算符
`expr $a + $b` # 表达式和运算符之间要有空格 # + - * / % [ $a == $b ] # 表达式要放在方括号之间,并且要有空格。 # 关系 -eq 相等 == -ne 不相等 != -gt 大于 -lt 小于 -ge 大于等于 -le 小于等于 # 布尔 -o 或 -a 与 [[ 表达式1 && 表达式2 ]] && 和 || 或 其他形式 [ -z $a ] 长度为0 [ -n "$a" ] 长度为 0" [ $a ] 字符串不为空" 文件检查符号 file='path' [ -r $file ] -f file 检测文件是否是普通文件 -b file 检测文件是否是块设备文件 -c file 检测文件是否是字符设备文件 -d file 检测文件是否是目录 -p file 检测文件是否是有名管道 -g file 检测文件是否设置了 SGID 位 -k file 检测文件是否设置了粘着位(Sticky Bit) -u file 检测文件是否设置了 SUID 位 -r file 检测文件是否可读 -w file 检测文件是否可写 -x file 检测文件是否可执行 -s file 检测文件是否为空(文件大小是否大于0) -e file 检测文件(包括目录)是否存在 -S: 判断某文件是否 socket。 -L: 检测文件是否存在并且是一个符号链接。
注释
# 单行注释
#
# 多行注释
借助函数,放在函数中但不调用
:<<EOF/'/!
注释内容...
注释内容...
注释内容...
EOF/'/!
# 声明解释脚本文件的shell程序的路径
#!/bin/sh
流程控制
循环
# 循环
# 固定次数/容器一次操作
for x in list
do
done
for x in list;do;done
# 直到不满足条件
while
do
done
while;do;done
# 直到满足条件
until
do
done
until;do;done
# 死循环
for (( ; ; ))
while true
结合条件选择使用
结合break,continue使用
判断
# 判断
if
then
elif
then
else
fi
if;then;fi
条件选择
case 值 in
模式1)
command
;;
模式2)
command
;;
*)
else-command
esac
选择循环
select in
Shell独有的一种循环
select variable in value_list
do
statements
done
eg:
#!/bin/bash
echo "What is your favourite OS?"
select name in "Linux" "Windows" "Mac OS" "UNIX" "Android"
do
echo $name
done
echo "You have selected $name"
运行结果:
What is your favourite OS?
1) Linux
2) Windows
3) Mac OS
4) UNIX
5) Android
#? 4↙
You have selected UNIX
#? 1↙
You have selected Linux
#? 9↙
You have selected
#? 2↙
You have selected Windows
#?^D
函数
funname()
{
action
$1
$2
return
}
# 调用
function args1 args2
$?获取返回值,不需要把函数赋值给变量
# 按序号位置对应传参,大于10时加{}分割,不能时$10,需要是${10}
重定向
# 一般重定向
n > file 将输出重定向到 file。
n < file 将输入重定向到 file。
n < file1 >file2 从1中读取执行n,并将结果储存到2
n >> file 将输出以追加的方式重定向到 file。
n >& m 将输出文件 m 和 n 合并。
n <& m 将输入文件 m 和 n 合并。
<< tag 将开始标记 tag 和结束标记 tag 之间的内容作为输入。
# Here Document重定向
command << delimiter
document
delimiter
将两个 delimiter 之间的内容(document) 作为输入传递给 command
结尾的delimiter 一定要顶格写,开始的delimiter前后的空格会被忽略掉。
# 禁止输出重定向
$ command > /dev/null
/dev/null 是一个特殊的文件,写入到它的内容都会被丢弃
文件引用
. filename
# 注意点号(.)和文件名中间有一空格
或
source filename
命令
# 放在``中
date 显示当前时间
expr 执行运算
# 不才``中
test 检查条件是否成立
* 关系运算符 test $[num1] -eq $[num2]
* 字符串运算符 test $num1 = $num2
* 文件运算符 test -e ./bash
* 逻辑运算符 test -e ./notFile -o -e ./bash
let "命令" 执行一个或多个表达式,变量计算中不需要加上 $ 来表示变量