假期周进度报告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 "命令" 执行一个或多个表达式,变量计算中不需要加上 $ 来表示变量
posted @ 2020-08-12 13:36  ziyuliu  阅读(121)  评论(0编辑  收藏  举报