Linux学习2

本文详细介绍了Linuxshell编程中的基本语法,包括如何设置可执行权限,使用注释,变量的定义、类型和操作,字符串处理,数组的使用,expr命令,read命令,echo命令的用法,test和if-else条件判断,以及case语句的使用。
摘要由CSDN通过智能技术生成

三、shell语法

接下来只会介绍一些常见的shell语法的使用

1.Linux系统中一般默认使用bash,所以接下来讲解bash中的语法。

文件开头需要写#! /bin/bash,指明bash为脚本解释器。
作为可执行文件执行需要权限,查看权限----ls -l test.sh,修改权限----chmod +x test.sh
  • 执行示例:
#! /bin/bash
echo "Hello World!"
  • 不同的运行方式
acs@9e0ebfcd82d7:~$ chmod +x test.sh  # 使脚本具有可执行权限
acs@9e0ebfcd82d7:~$ ./test.sh  # 当前路径下执行
Hello World!  # 脚本输出
acs@9e0ebfcd82d7:~$ /home/acs/test.sh  # 绝对路径下执行
Hello World!  # 脚本输出
acs@9e0ebfcd82d7:~$ ~/test.sh  # 家目录路径下执行
Hello World!  # 脚本输出
  • 用解释器执行
acs@9e0ebfcd82d7:~$ bash test.sh
Hello World!  # 脚本输出
2、注释
  • 单行注释
# 这是一行注释
echo 'Hello World'  #  这也是注释
  • 多行注释(其中EOF可以替换)
:<<EOF
第一行注释
第二行注释
第三行注释
EOF
3、变量
  • 定义变量
    • 定义变量直接使用即可,例如name1=''LXLY,字符串可以双引号、单引号、不加引号三种方式都可以
  • 使用变量
    • 例如:
    name=LXLY
    echo $name # 输出LXLY
    echo ${name} # 输出LXLY
    echo ${name}9999 # 输出LXLY9999
    
  • 只读变量
    • 使用readonly或者declare可以将变量变为只读。
name=yxc
readonly name
declare -r name  # 两种写法均可

name=abc  # 会报错,因为此时name只读
  • 删除变量
    • unset可以删除变量
name=yxc
unset name
echo $name  # 输出空行
  • 变量类型
    • 自定义变量(局部变量):子进程不能访问的变量
    • 环境变量(全局变量):子进程可以访问的变量
    • 自定义变量改成环境变量:
acs@9e0ebfcd82d7:~$ name=yxc  # 定义变量
acs@9e0ebfcd82d7:~$ export name  # 第一种方法
acs@9e0ebfcd82d7:~$ declare -x name  # 第二种方法
  • 环境变量改为自定义变量:
acs@9e0ebfcd82d7:~$ export name=yxc  # 定义环境变量
acs@9e0ebfcd82d7:~$ declare +x name  # 改为自定义变量
4、字符串

字符串可以用单引号,也可以用双引号,也可以不用引号

  • 单引号和双引号的区别:
    • 单引号中的内容会原样输出,不会执行、不会取变量;
    • 双引号中的内容可以执行、可以取变量;
name=LXLY  # 不用引号
echo 'hello, $name \"hh\"'  # 单引号字符串,输出 hello, $name \"hh\"
echo "hello, $name \"hh\""  # 双引号字符串,输出 hello, LXLY "hh"
  • 获取字符串长度
name="LXLY"
echo ${#name}  # 输出4
  • 提取字符串
name="hello, LXLY"
echo ${name:0:5}  # 提取从0开始的5个字符
5、数组

数组中可以存放多个不同类型的值,只支持一维数组,初始化时不需要指明数组大小。
数组下标从0开始

  • 定义
    数组用小括号表示,元素之间用空格隔开。例如:
    array=(1 abc "def" yxc)
    也可以直接定义数组中某个元素的值:
array[0]=1
array[1]=abc
array[2]="def"
array[3]=yxc
  • 读取数组中某个元素的值
    格式:${array[index]}.例如:
array=(1 abc "def" yxc)
echo ${array[0]}
echo ${array[1]}
echo ${array[2]}
echo ${array[3]}
  • 读取整个数组
${array[@]}  # 第一种写法
${array[*]}  # 第二种写法
array=(1 abc "def" yxc)

echo ${array[@]}  # 第一种写法
echo ${array[*]}  # 第二种写法
  • 获取数组长度
${#array[@]}  # 第一种写法
${#array[*]}  # 第二种写法
array=(1 abc "def" yxc)

echo ${#array[@]}  # 第一种写法
echo ${#array[*]}  # 第二种写法
6、expr命令

expr命令用于求表达式的值,格式为:
expr+表达式
表达式注意:

  • 用空格隔开每一项
  • 用反斜杠放在shell特定的字符前面(发现表达式运行错误时,可以试试转义)
  • 对包含空格和其他特殊字符的字符串要用引号括起来
  • expr会在stdout中输出结果。如果为逻辑关系表达式,则结果为真时,stdout输出1,否则输出0。
  • expr的exit code:如果为逻辑关系表达式,则结果为真时,exit code为0,否则为1。
  1. 当表达式为字符转串时
  • length STRING
    返回STRING的长度
  • index STRING CHARSET
    CHARSET中任意单个字符在STRING中最前面的字符位置,下标从1开始。如果在STRING中完全不存在CHARSET中的字符,则返回0。
  • substr STRING POSITION LENGTH
    返回STRING字符串中从POSITION开始,长度最大为LENGTH的子串。如果POSITIONLENGTH为负数,0或非数值,则返回空字符串。
  1. 当表达式为运算表达式时(+ - * / % ()
  • + -
    • 加减运算。两端参数会转换为整数,如果转换失败则报错
  • / %
    • 乘,除,取模运算。两端参数会转换为整数,如果转换失败则报错
  • ():可以改变优先级,但需要用反斜杠转义
示例:
a=3
b=4

echo `expr $a + $b`  # 输出7
echo `expr $a - $b`  # 输出-1
echo `expr $a \* $b`  # 输出12,*需要转义
echo `expr $a / $b`  # 输出0,整除
echo `expr $a % $b` # 输出3
echo `expr \( $a + 1 \) \* \( $b + 1 \)`  # 输出20,值为(a + 1) * (b + 1)
  1. 逻辑关系表达式
  • |
    • 如果第一个参数非空且非0,则返回第一个参数的值,否则返回第二个参数的值,但要求第二个参数的值也是非空或非0,否则返回0。如果第一个参数是非空或非0时,不会计算第二个参数。
  • &
    • 如果两个参数都非空且非0,则返回第一个参数,否则返回0。如果第一个参为0或为空,则不会计算第二个参数。
  • < <= = == != >= >
    • 比较两端的参数,如果为true,则返回1,否则返回0。”==”是”=”的同义词。”expr”首先尝试将两端参数转换为整数,并做算术比较,如果转换失败,则按字符集排序规则做字符比较。
  • ():可以改变优先级,但需要用反斜杠转义
    示例:
a=3
b=4

echo `expr $a \> $b`  # 输出0,>需要转义
echo `expr $a '<' $b`  # 输出1,也可以将特殊字符用引号引起来
echo `expr $a '>=' $b`  # 输出0
echo `expr $a \<\= $b`  # 输出1

c=0
d=5

echo `expr $c \& $d`  # 输出0
echo `expr $a \& $b`  # 输出3
echo `expr $c \| $d`  # 输出5
echo `expr $a \| $b`  # 输出3
7、read命令

read命令用于从标准输入中读取单行数据。当读到文件结束符时,exit code为1,否则为0。
参数说明

  • -p:后面可以接提示信息
  • -t:后面跟秒数,定义输入字符的等待时间,超过等待时间后会自动忽略此命令

实例(在终端中作为辅助输入):

acs@9e0ebfcd82d7:~$ read name  # 读入name的值
acwing yxc  # 标准输入
acs@9e0ebfcd82d7:~$ echo $name  # 输出name的值
acwing yxc  #标准输出
acs@9e0ebfcd82d7:~$ read -p "Please input your name: " -t 30 name  # 读入name的值,等待时间30秒
Please input your name: acwing yxc  # 标准输入
acs@9e0ebfcd82d7:~$ echo $name  # 输出name的值
acwing yxc  # 标准输出
8、echo(输出字符串)

echo用于输出字符串。命令格式:
echo STRING

  1. 显示普通字符串
echo "Hello AC Terminal"
echo Hello AC Terminal  # 引号可以省略
  1. 显示转义字符
echo "\"Hello AC Terminal\""  # 注意只能使用双引号,如果使用单引号,则不转义
echo \"Hello AC Terminal\"  # 也可以省略双引号
  1. 显示变量
name=yxc
echo "My name is $name"  # 输出 My name is yxc
  1. 显示换行
echo -e "Hi\n"  # -e 开启转义
echo "LXLY"

# 输出结果
Hi

LXLY
  1. 显示不换行
echo -e "Hi \c" # -e 开启转义 \c 不换行
echo "LXLY"

# 直接输出结果
Hi LXLY
  1. 显示结果定向至文件
echo "Hello World" > output.txt  # 将内容以覆盖的方式输出到output.txt中
  1. 原样输出字符串,不进行转义或取变量(用单引号)
name=LXLY
echo '$name\"'

# 输出结果
$name\"
  1. 显示命令的执行结果
echo `date`

# 输出结果:
Wed Sep 1 11:45:33 CST 2021
注意:\\ \a \b \c \d \e \f \n \r \t \v 这些是要在有 - e 的时候才能起作用, 其他时候的转义是不用- e也能转义的。 man echo 里特俗
9、printf

printf命令用于格式化输出,类似于C/C++中的printf函数,默认不会在字符串末尾添加换行符。

  • 命令格式:
    printf format-string [arguments...]
  • 用法示例
    脚本内容:
printf "%10d.\n" 123  # 占10位,右对齐
printf "%-10.2f.\n" 123.123321  # 占10位,保留2位小数,左对齐
printf "My name is %s\n" "yxc"  # 格式化输出字符串
printf "%d * %d = %d\n"  2 3 `expr 2 \* 3` # 表达式的值作为参数

# 输出结果:
       123.
123.12    .
My name is yxc
2 * 3 = 6
10、test命令与判断符号[]

首先了解逻辑运算符&&和||

  • && 表示与,|| 表示或
  • 二者具有短路原则:
    expr1 && expr2:当expr1为假时,直接忽略expr2
    expr1 || expr2:当expr1为真时,直接忽略expr2
  • 表达式的exit code为0,表示真;为非零,表示假。(与C/C++中的定义相反)
  1. test命令
    在命令行中输入man test,可以查看test命令的用法。
    test命令用于判断文件类型,以及对变量做比较。
    test命令用exit code返回结果,而不是使用stdout。0表示真,非0表示假。
    例如:
test 2 -lt 3  # 为真,返回值为0
echo $?  # 输出上个命令的返回值,输出0

acs@9e0ebfcd82d7:~$ ls  # 列出当前目录下的所有文件
homework  output.txt  test.sh  tmp
acs@9e0ebfcd82d7:~$ test -e test.sh && echo "exist" || echo "Not exist"
exist  # test.sh 文件存在
acs@9e0ebfcd82d7:~$ test -e test2.sh && echo "exist" || echo "Not exist"
Not exist  # testh2.sh 文件不存在
  1. 文件类型判断
    命令格式:
    test -e filename # 判断文件是否存在
-e-f-d
文件是否存在是否为文件是否为目录
  1. 文件权限判断
    命令格式:
    test -r filename # 判断文件是否可读
测试参数代表意义
-r文件是否可读
-w文件是否可写
-x文件是否可执行
-s是否为非空文件
  1. 整数间的比较
    test $a -eq $b # a是否等于b
测试参数代表意义
-eqa是否等于b
-gta是否大于b
-lta是否小于b
-gea是否大于等于b
-lea是否小于等于b
  1. 字符串比较
测试参数代表意义
test -z STRING判断STRING是否为空,如果为空,则返回true
test -n STRING判断STRING是否非空,如果非空,则返回true(-n可以省略)
test str1 == str2判断str1是否等于str2
test str1 != str2判断str1是否不等于str2
  1. 多重条件判定
    命令格式:
    test -r filename -a -x filename
测试参数代表意义
-a两条件是否同时成立
-o两条件是否至少一个成立
!取反。如 test ! -x file,当file不可执行时,返回true
  1. 判断符号[]
    []与test用法几乎一模一样,更常用于if语句中。另外[[]][]的加强版,支持的特性更多
    **例如:
[ 2 -lt 3 ]  # 为真,返回值为0
echo $?  # 输出上个命令的返回值,输出0


acs@9e0ebfcd82d7:~$ ls  # 列出当前目录下的所有文件
homework  output.txt  test.sh  tmp
acs@9e0ebfcd82d7:~$ [ -e test.sh ] && echo "exist" || echo "Not exist"
exist  # test.sh 文件存在
acs@9e0ebfcd82d7:~$ [ -e test2.sh ] && echo "exist" || echo "Not exist"
Not exist  # testh2.sh 文件不存在

注意:

  • []内的每一项都要用空格隔开
  • 中括号内的变量,最好用双引号括起来
  • 中括号内的常数,最好用单或双引号括起来

例如:

name="acwing LXLY"
[ $name == "acwing LXLY" ]  # 错误,等价于 [ acwing LXLY == "acwing LXLY" ],参数太多
[ "$name" == "acwing LXLY" ]  # 正确
11、判断语句

if…then形式

  • 单个if
if condition
then
    语句1
    语句2
    ...
fi

# 示例:
a=3
b=4

if [ "$a" -lt "$b" ] && [ "$a" -gt 2 ]
then
    echo ${a}在范围内
fi

# 输出结果
3在范围内
  • 单层if-else
if condition
then
    语句1
    语句2
    ...
else
    语句1
    语句2
    ...
fi

# 示例:
a=3
b=4
if ! [ "$a" -lt "$b" ]
then
    echo ${a}不小于${b}
else
    echo ${a}小于${b}
fi

# 输出结果:
3小于4
  • 嵌套模式
if condition
then
    语句1
    语句2
    ...
elif condition
then
    语句1
    语句2
    ...
elif condition
then
    语句1
    语句2
else
    语句1
    语句2
    ...
fi

# 示例:
a=4
if [ $a -eq 1 ]
then
    echo ${a}等于1
elif [ $a -eq 2 ]
then
    echo ${a}等于2
elif [ $a -eq 3 ]
then
    echo ${a}等于3
else
    echo 其他
fi

# 输出结果:
其他
12、case…esac形式(相当于switch语句)
case $变量名称 in
    值1)
        语句1
        语句2
        ...
        ;;  # 类似于C/C++中的break
    值2)
        语句1
        语句2
        ...
        ;;
    *)  # 类似于C/C++中的default
        语句1
        语句2
        ...
        ;;
esac

# 示例:
a=4
case $a in
    1)
        echo ${a}等于1
        ;;  
    2)
        echo ${a}等于2
        ;;  
    3)                                                
        echo ${a}等于3
        ;;  
    *)
        echo 其他
        ;;  
esac

# 输出结果:
其他
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值