参考链接:
目录
基础知识准备
文本命令
cat
cat
命令的常用功能是将文件内容打印并输出到屏幕上。
cool@cool:~/Shell_Learning$ vim demo.txt
cool@cool:~/Shell_Learning$ cat demo.txt
hello
world
this is a test
除此之外,cat
命令也能创建和编辑新文件,由于该用法用的不多,所以作者就不在这里展开讨论了。
echo
echo命令能将指定文本显示在Linux命令行上,或者通过重定向符写入到指定的文件中。
>
为重定向符号,可以将内容追加到文件中,如果有同名文件会先清除原文件所有内容>>
为追加重定向符号,不会清除文件的内容,而是将对应内容追加到文件的末尾行
cool@cool:~/Shell_Learning$ echo "hello"
hello
cool@cool:~/Shell_Learning$ echo "hello">hello.txt
cool@cool:~/Shell_Learning$ cat hello.txt
hello
cool@cool:~/Shell_Learning$ echo "world">>hello.txt
cool@cool:~/Shell_Learning$ cat hello.txt
hello
world
grep
grep命令是Linux系统中最重要的命令之一,其功能是从文本文件或管道数据流中筛选匹配的行及数据。
sed/awk
这两个文本编辑器学习起来难度比较大,且学了不常用容易忘,下面是几篇相关文章:
vim编辑器
vim文本编辑器,是由 vi 发展演变过来的文本编辑器,使用简单、功能强大、是 Linux 众多发行版的默认文本编辑器。
vim编辑器主要有三种工作模式:
- 命令模式:不能编辑文件,只能通过快捷键进行一些操作(如移动光标、复制、粘贴、删除字符/整行等),打开vim后默认进入命令模式。
- 末行模式:可在末行输入一些命令对文件进行操作(如搜索、替换、保存、退出、高亮等)。
- 编辑模式:在命令模式下按下
i
或a
进入编辑模式,可对文件内容进行编辑。
关于vim编辑器的具体操作可参考:编辑器之神——vim编辑器(详细、完整)-CSDN博客
一、Shell入门介绍
1.1 Shell概述
Shell是一个命令解释器,它的作用是解释执行用户输入的命令及程序等,用户每输入一条命令,Shell就解释执行一条。
Shell存在于操作系统的最外层,负责与用户直接对话,把用户的输入解释给操作系统,并处理各种各样的操作系统的输出结果,然后输出到屏幕返回给用户。输入系统用户名和密码并登录到Linux后的所有操作都是由Shell解释与执行的。
1.2 Shell脚本分类
Shell脚本主要分为两大类:Bourne shell和C shell。Linux系统中的主流Shell是bash。
Bourne shell
Bourne shell包含三个类型:Bourne shell(sh)、Korn shell(ksh)、Bourne Again Shell(bash)
- Bourne shell(sh)是标准的UNIX Shell,很多UNIX系统都配有sh。
- 是Bourne shell(sh)的超集合,并且添加了csh引入的新功能,是目前很多UNIX系统标准配置的Shell。
- Bourne Again Shell(bash)是各种Linux发行版默认配置的Shell,Linux系统上的/bin/sh往往是指向/bin/bash的符号链接。
C shell
C shell包含两个类型:csh和tsh。
- csh支持很多Bourne shell所不支持的功能,例如:作业控制、别名、系统算术、命令历史、命令行编辑等。
- tcsh是csh的增强版,加入了命令补全等功能,在FreeBSD、Mac OS X等系统上替代了csh。
1.3 Shell脚本的创建与执行
可以通过以下命令查看当前Linux系统的Shell支持情况:
cat /etc/shells
在我的Ubuntu20.04执行该命令,结果如下:
cool@cool:~/Shell_Learning$ cat /etc/shells
/bin/sh
/bin/bash
/usr/bin/bash
/bin/rbash
/usr/bin/rbash
/bin/dash
/usr/bin/dash
可以通过以下命令查看当前Linux系统默认的Shell
echo $SHELL
输出结果一般是:/bin/bash
脚本创建
shell脚本文件第一行一般第一行会指出由哪个程序(解释器)来执行脚本中的内容,一般为:
#!/bin/bash
其他内容就根据我们的需求编辑就行,可以是我们在shell终端输入的命令,也可以包含选择语句、循环语句、shell函数等,具体写法不在本章展开。
脚本执行
执行脚本文件一般有4种方法。
bash demo.sh
./demo.sh
source demo.sh
. demo.sh
前两种方式会在一个新的子 shell 中执行demo.sh
脚本。这意味着脚本中的所有变量和环境改变都仅在这个子 shell 中有效,而不会影响当前的 shell。
后两种方式会在当前的 shell 环境中执行demo.sh
脚本。脚本中的所有变量和环境改变都会直接反映在当前的 shell 中。
其中在使用方法2执行脚本文件前,需要给予脚本执行权限:chmod +x demo.sh
二、Shell变量
Shell变量可分为:环境变量(全局变量)和普通变量(局部变量)。
- 环境变量也可称为全局变量,可以在创建它们的Shell及其派生出来的任意子进程Shell中使用,环境变量又可分为自定义环境变量和bash内置的环境变量。
- 普通变量也可称为局部变量,只能在创建它们的Shell函数或Shell脚本中使用。普通变量一般由开发者在开发脚本程序时创建。
2.1 环境变量
常见的系统环境变量:
$0 当前脚本的名称;
$n 当前脚本的第n个参数,n=1,2,…9;
$* 当前脚本的所有参数(不包括程序本身);
$# 当前脚本的参数个数(不包括程序本身);
$? 前一个命令或程序执行完后的状态,返回0表示执行成功;
$$ 程序本身的PID号。
PATH 命令所示路径,以冒号为分割;
HOME 打印用户家目录;
SHELL 显示当前Shell类型;
USER 打印当前用户名;
ID 打印当前用户id信息;
PWD 显示当前所在路径;
TERM 打印当前终端类型;
HOSTNAME 显示当前主机名;
HISTSIZE 历史命令大小,可通过 HISTTIMEFORMAT 变量设置命令执行时间;
RANDOM 随机生成一个 0 至 32767 的整数;
HOSTNAME 主机名
环境变量可以在终端中设置和创建,但是终端关闭后变量值就会丢失,如果想永久保存环境变量,可以将变量配置在:
- 用户家目录下的bash_profile或bashrc文件中
- 全局配置/etc/bashrc或/etc/profile文件中定义。
自定义环境变量的方式:
# 按照系统规范,所有环境变量的名字均采用大写形式。等号两边不能有空格
export NUM=1
# 或者
NUM=100
export NUM
查看环境变量:
env
查看当前Shell的所有变量,包括全局变量和局部变量:
set
取消定义的环境变量:
unset NUM
2.2 普通变量
普通变量(本地变量/局部变量)在用户当前Shell生存期的脚本中使用,这个值只在用户当前Shell生存期中有意义。如果在Shell中启动另一个进程或退出,那么变量的值将会无效。
为普通变量赋值有三种方法:
变量=value
变量='value'
变量="value"
- 不加任何引号直接定义变量的内容,值里有变量的会被解析后再输出。
- 通过单引号定义。这种定义方式的特点是:输出变量内容时单引号里是什么就输出什么,即使内容中有变量和命令,也会把它们原样输出。
- 通过双引号定义变量。这种定义方式的特点是:输出变量内容时引号里的变量及命令会经过解析后再输出内容。
-
2.3 Shell变量进阶
- 将命令结果赋值给变量
变量=`ls`
变量=$(ls)
# 示例:
a=`ls`
echo $a
- 当变量后面连接有其他字符的时候,必须给变量加上大括号{}
- 连续字母、数字打印
echo {a..z}
# 输出为:a b c d e f g h i j k l m n o p q r s t u v w x y z
echo {1..9}
# 输出为:1 2 3 4 5 6 7 8 9
$?
:用于获取执行上一个指令的执行状态返回值,0表示成功,非0表示失败。
三、Shell运算符与条件测试
仅列举几个常用的:
在[] 中使用的比较符号 | 在(()) 和[[]] 中使用的比较符号 | 说明 |
---|---|---|
-eq | ==或= | 相等 |
-ne | != | 不相等 |
-gt | > | 大于 |
-ge | >= | 大于等于 |
-lt | < | 小于 |
-le | <= | 小于等于 |
二元数字在[] 中使用-gt、-le类符号的比较: |
[ 2 -gt 1 ] && echo 1 || echo 0 # 输出为1
[ 2 -lt 1 ] && echo 1 || echo 0 # 输出为0
二元数字在(())
中的比较。
((3>2)) && echo 1 || echo 0
((3<2)) && echo 1 || echo 0
((3==2)) && echo 1 || echo 0
文件测试表达式:(这些操作符号对于[[]]
、[]
、test
的测试表达式是通用的)
操作符 | 说明 |
---|---|
-d | 文件存在且为目录则为真,则测试表达式成立 |
-f | 文件存在且为普通文件则为真,则测试表达式成立 |
-e | 文件存在则为真,-e不辨别是目录还是文件 |
字符串测试操作符:
常用字符串测试操作符 | 说明 |
---|---|
-n “字符串” | 若字符串的长度不为0,则为真 |
-z “字符串” | 若字符串的长度为0,则为真 |
“字符串1”=“字符串2” | 若字符串1等于字符串2,则为真;可使用"==“替代”=" |
“字符串1”!=“字符串2” | 若字符串1不等于字符串2,则为真 |
四、Shell选择语句、循环语句与函数
4.1 选择语句
4.1.1 if语句
语法如下:
# 单分支结构
if <条件表达式>;then
指令
fi
# 双分支结构
if <条件表达式>;then
指令
else
指令
fi
# 例如:
if [ -f "$file1" ];then
echo 1
else
echo 0
fi
4.1.2 read命令
- -p prompt:设置提示信息。
- -t timeout:设置输入等待的时间,单位默认为秒。
read -t 10 -p "input a number:" num # 变量前需要有空格 -t 10 等待10s
echo $num
read -t 10 -p "input two number:" num1 num2 # 变量前需要有空格
echo $num1 $num2
4.1.3 case语句
case条件语句相当于多分支的if/elif/else条件语句,但是它比这些条件语句看起来更规范更工整。
语法格式如下:
case "变量" in
值1)
指令..
;;
值2)
指令..
;;
*)
指令..
esac
# 示例:
#!/bin/bash
read -p "input a number:" num
case "$num" in
1)
echo "the num is 1"
;;
2)
echo "the num is 2"
;;
[3-9])
echo "the num is $num"
;;
*)
echo "error!"
;;
esac
4.2 循环语句
4.2.1 while循环
while <条件表达式>
do
指令
done
# 示例:
while true # 条件为真一直运行
do
uptime
sleep 2 # 让程序暂停2秒,控制循环的频率,否则会消耗大量的系统资源
done
4.2.2 for循环
for 变量名 in 变量取值列表
do
指令
done
4.3 shell函数
标准写法:
# 函数声明:
function 函数名() {
指令
return n
}
# 函数调用:
# 执行不带参的函数时,直接输入函数名
function
# 执行带参的函数时
function 参数1 参数2
shell中的break
、continue
、exit
、return
:
break n
:如果省略n,则表示跳出整个循环,n表示跳出循环的层数。continue n
:如果省略n,则表示跳过本次循环,进入下一次循环;n表示退到第n层继续循环。exit n
:退出当前shell程序,n为程序执行的状态返回值。在下一个shell中可以通过$?
接收exit的返回值n。return n
:作为函数的返回值。
五、应用实例
- 脚本生成一个 100 以内的随机数,提示用户猜数字,根据用户的输入,提示用户猜对了,猜小了或猜大了,直至用户猜对脚本结束。
#!/bin/bash
num=$[RANDOM%100+1]
while true
read -p "猜一个1到100之间的数:" guess
do
if [ $num -gt $guess ];then
echo "猜小了"
elif [ $num -lt $guess ];then
echo "猜大了"
elif [ $num -eq $guess ];then
echo "恭喜你!猜对了!"
break
else
echo "请按要求输入数字!"
fi
done
- 复制文件并显示进度条。
#!/bin/bash
jindu(){
while true
do
echo -n "#"
sleep 0.2
done
}
cur_path=$PWD
echo "开始创建新文件夹..."
mkdir $cur_path/jindu1 $cur_path/jindu2
if [ $? -eq 0 ];then
file1_path=$cur_path/jindu1
file2_path=$cur_path/jindu2
echo "新文件夹创建成功!"
fi
touch $file1_path/test111.txt
jindu & # 这里的&表示将jindu函数放在后台运行
JINDU_PID=$!
cp -a $file1_path $file2_path
sleep 2
kill $JINDU_PID
echo -e "\n拷贝完成!"
sleep 0.5
rm -rf $file1_path $file2_path
if [ $? -eq 0 ];then
echo "脚本运行完毕,已自动将文件删除"
fi