linux学习:shell脚本

目录

要求

注意

变量

变量的定义和赋值

变量的引用

变量的种类

设置环境变量

特殊符号

引号

竖杠(管道)

和大于小于号(重定向)

字符串处理

测试语句

脚本语法

判断

循环

函数

trap


前提

要把 Shell 命令放到一个“脚本”当中,有一个前提:脚本的第一行必须写成类似这样 的格式: #!/bin/bash

这是给系统指定一款 Shell 解释器

注意

脚本文件缺省是没有执行权限的,要使得脚本可以执行必须给他添加权限

赋值号的两边没有空格!在 Shell 脚本中,任何时候要给变量赋值,赋值号两 边一定不能有空格

变量名也有类似于 C 语言那样的规定:只能包含英文字母和数字,且不能以数字开 头

变量

变量的定义和赋值

myname=”Michael Jackson”

变量的引用

使用变量时,需要在变量的前面加一个美元符号

变量的种类

  • A 普通的用户自定义变量,比如上面的 myname。
  • B 系统预定义好的环境变量,比如 PATH。
  • C 命令行变量,比如$#、$*等。

系统的环境变量可以通过如下命令来查看

  • env

每个环境变量都用大写字母表示,比如 PATH。每个 环境变量都有一个值,就是等号右边的字符串。根据环境变量的不同,它们各自的含义不同

设置环境变量

  • export PATH=dir/

PATH 环境变量的作用是保存系统中可执行程序或脚本的所在路径,因此它的值都是 一些以分号隔开的目录,我们经常的使用办法是:不改变其原有的值,而给它再增加一个我们自己需要设置的目录 dir/,因此更有用的命令可能是类似于以下这样的

  • export PATH=$PATH:dir/

而 Shell 脚本中的所谓命令行变量,指的是在脚本内部使用用户从命令行中传递进来的参数

  • ./example.sh abcd 1234

例如上面语句,脚本名叫 example.sh,咱们在执行他的时候顺便给了他两个参数,分别是 abcd 和 1234,要访问这两个参数以及相关的其他值,就必须使用命令行变量

  • $# :代表命令行参数个数,即 2
  • $* :代表所有的参数,即 abcd 1234
  • $@ :同上
  • $n :第 n 个参数,比如$1
  • $? :代表最后一个命令执行之后的返回值
  • $$ :代表当前 Shell 的进程号 PID

特殊符号

引号

引号有三种,他们是:双引号 “ ”、 单引号 ‘ ’、 反引号(抑音符) ` `

  • 双引号的作用是将一些“单词”括起来形成单个的“值”,还可以包括对变量的引用,还可以是一个命令
  • 被单引号所包含,那么 其内部的任何成分都将被视为普通的字符,而不是变量的引用或者命令
  • 反引号的作用就是在双引号中标识出命令
#!/bin/bash
var=calender
echo "var: date" # 直接打印出 var 和 date
echo "$var: `date`" # 打印出变量 var 的值,以及命令 date 的执行结果
echo '$var: `date`' # 打印出$var: `date`

竖杠(管道)

管道不仅可以连接两个命令,也可以连接多个命令

ls -l | wc
cat /etc/passwd | awk -F=/ '{print $1}' | wc

 这就将三个命令连接起来,每个命令的输出都作为下一个命令的输入,连接起来就能完成强 大的功能

和大于小于号(重定向)

每一个进程在刚开始运行的时候,系统都会为他们默认地打开了三个文件,他们分别是 标准输入、标准输出、标准出错

这三个标准文件对应两个硬件设备:标准输入是键盘,标准输出和标准出错是显示器(是 的,显示器设备被打开了两次,第一次打开为行缓冲类型的标准输出,第二次打开为不缓冲 类型的标准出错)。绝大多数的 Shell 命令,默认的输入输出都是这三个文件

当我们执行命令 ls 的时候,他会默认地将结果打印到显示器上,就是因为 ls 本 来就被设计为将结果往 1 号描述符(即标准输出,当执行成功的时候)或者 2 号描述符(即 标准出错,当执行失败的时候)。而当我们打开普通文件的时候,系统也会帮我们产生一系 列后续的数字(文件描述符)来表示这些文件,比如我们紧跟着打开了文件 a.txt 和 b.doc,

.假如需要将 ls 命令的成功的输出结果(本来会被默认地输送到 1 号文件描述符的信 息)重定向到 a.txt 文件中去

ls 1> a.txt

假如需要将 ls 命令的失败的输出结果(本来会被默认地输送到 2 号文件描述符的信息)重定向到 a.txt 文件中去

ls notexist 2> a.txt (notexist 是一个不存在的文件,所以 ls
命令执行会失败)

重定向标准输入也是类似的,比如直接执行 echo 命令,他将会默认地从标准输入(即 键盘)读取信息,然后打印出来。但是我们可以将标准输入重定向为 b.doc 文件:

echo 0< b.doc

 标准输入输出设备文件描述符要写成&0、&1 和&2。比 如要将一句话输出到标准出错设备中去

echo “hello world” 1>&2

字符串处理

计算一个字符串的字符个数

var="apple tree"
echo "${#var}"
10

删除一个字符串左边部分字符

path="/etc/rc0.d/K20openbsd-inetd"
level=${path#/etc/rc[0-9].d/[SK]}
echo $level
20openbsd-inetd

删除一个字符串右边部分字符

path="/etc/rc0.d/K20openbsd-inetd"
level=${path#/etc/rc[0-9].d/[SK]}
level=${level%%[a-zA-Z]*}
echo $level
20

注意

两个%%表示贪婪匹配,具体含义是:使用通配符[a-zA-Z]*从右向左“尽可能多地” 匹配字符(贪婪原则)。如果只写一个%,则无贪婪原则,那么[a-zA-Z]*将按照最少原则 匹配,即匹配 0 个字符(因为方括号星号*的含义是 0 个或多个字符)。这个道理对于删除 左边字符的井号#也是适用的:双井号##代表从左到右的贪婪匹配

测试语句

有一个叫 test 的命令,专门用来实现所谓的测试语句,测试语句可以测试很多不同的 情形

例子1

要判断一个文件 file 是否存在,而且可读,如果都满足的话就将其显示 在屏幕上,脚本可以写成这样

#!/bin/bash
if test -e file && test -r file
thencat file
fi
等价于
#!/bin/bash
if [ -e file ] && [ -r file ]
thencat file
fi

依靠 test 语句来决定是否要执行 cat 命令,这是脚本语言中最简单的条 件判断语句,根 C 语言的 if-else 结构很类似

  • 方括号的左右两边都必须有空格

脚本语法

判断

if [ -e file ] && [ -r file ]
thencat file
fi
  • 每一个 if 语句都有一个 fi(即倒过来写的 if)作为结束标记。
  • 分支结构中使用 then 作为起始语句。
  • 当且仅当 if 语句后面的语句执行结果为真(即为 0)时,then 以下的语句才会被 执行。当然,if 语句还可以跟 else 配对使用 
if [ -e file ] && [ -r file ]
thencat file # 如果文件存在且可读,则显示该文件内容
elif [ -e file ]
thenchmod u+r filecat file # 如果文件存在但不可读,则加了读权限之后再显示其内容
elsetouch file # 如果文件不存在,则创建该空文件
fi
  •  elif 而不是 else if,其后也要跟 if 一样紧随 then 语句。如果是多路分支,可以使用 case 语句,这个类似于 C 语言中的 switch 语句。比如实现这么一个功能:要求用户输入 一个数字,判断如果输入的是 1,则输出 one,如果输入的是 2,则输出 two,输入其他数 字则输出 unkown,
read VAR # 从键盘接收一个用户输入
case $VAR in # 判断用户输入的值$VAR1) echo “one” # 如果$VAR 的值为 1,则显示 one;; # 每个分支都必须以双分号作为结束(最后一个分支除外)2) echo “two” 6 ;;*) echo “unknown” # 星号*是 Shell 中的通配符,代表任意字符。
esac
  • 变量 VAR 的值实际上是字符串,因此上述代码中的 1) 也可写成 “1”)
  • 整个 case 结构必须 esac 作为结束

循环

实现打印 1 到 100 的功能

declare -i n=0 # 在定义变量 n 前面加上 declare -i 表示该变量为数值
while [ $n -le 100 ] # 如果 n 的值小于等于 100,则循
do # 循环体用 do 和 done 包含起来echo “$n” n=$n+1 # 使 n 的值加 1
don
declare -i n=0
until [ $n -gt 100 ] # 如果 n 的值大于 100,则退出循环
doecho “$n” n=$n+1
done

列出当前目录下每个普通文件所包含的行数

files=`ls` # 在当前目录下执行 ls,将所有的文件名保存在变量 files 中
for a in $files # 循环地将 files 里面的每个单词赋给 a,赋完则退出循环
doif [ -f $a ] # 如果文件$a 是一个普通文件,那么就计算他的行数thenwc -l $afi
done
  • for 循环中,in 后面接的是一个字符串,字符串里面包含几个单词循环体就执行几遍, 每执行一遍 a 的值都轮换地等于字符串里边的各个

函数

编写一个可以检测某用户是否在线的函数

check_user( ) # 定义一个函数 check_user( ),注意括号里面没有空格
{if [ $1 = "quit" ] # 若函数的第一个参数$1 为“quit”,则立即结束脚本thenexitfiUSER=`who | grep $1 | wc -l`if [ $USER -eq 0 ]thenreturn 0 # 判断用户$1 是否在线,是则返回 1,否则返回 0elsereturn 1fi
}
while true
doecho -n "input a user name:" read USERcheck_user $USER # 调用 check_user,并传递参数$USERif [ $? -eq 1 ] # 判断 check_user 的返回值$?是否为 1thenecho "[$USERNAME] online." elseecho "[$USERNAME] offline." fi
done
  • 函数的定义中,括号里面不能写任何东西(第一行);函数必须定义在调用之前; 给函数传参的时候,传递的参数在函数的定义里用$n 来表示第 n 个参数。$?代表函数调用的返回值

trap

脚本中经常有信号处理的语句,最常见的情况是:当脚本收到某个信号的时候,需要处理一些清理工作,然后再退出,类似于 POSIX 编程中的信号处理。脚本中使用 trap 来达到这个目的。

trap “” INT
  • 当脚本收到信号 SIGINT 时,忽略该信号。在 Linux 中所支持的 信号可以使用命令 trap -l 来查看。信号名称的前缀要省略。trap 除了可以“忽略”信号

也可以“捕获”信号

trap do_something INT QUIT HUP
  • 当 脚 本 收 到 INT 、 QUIT 或 者 HUP 信 号 时 执 行 函 数 do_something

还可以指定脚本正常退出时的默认动作

trap on_exit EXIT
  • 当脚本正常退出时,执行函数 on_exit

当一 Shell 脚本 收到某一个信号的时候,我们需要该脚本立即终止,且要执行正常退出时的清理函数

trap on_exit EXIT
trap “:” INT HUP
  • 当脚本正常退出时执行函数 on_exit,当脚本收到信号 INT 或者 HUP 时执行空指令(此处冒号代表一个空指令,如果没有冒号,脚本将完全忽略该信 号,不做响应,不能立即退出),完了之后正常退出,此时触发 EXIT 从而执行函数 on_exit。

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/599344.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

从0到1搭建文档库——sphinx + git + read the docs

sphinx git read the docs 目录 一、sphinx 1 sphinx的安装 2 本地构建文件框架 1&#xff09;创建基本框架&#xff08;生成index.rst &#xff1b;conf.py&#xff09; conf.py默认内容 index.rst默认内容 2&#xff09;生成页面&#xff08;Windows系统下&#xf…

windows一键休眠,一键唤醒

1.使windows睡眠不可用&#xff0c;cmd以管理员身份运行&#xff1a; powercfg.exe /hibernate off 2.桌面创建快捷键 Rundll32.exe Powrprof.dll,SetSuspendState Sleep

零信任安全模型:构建未来数字世界的安全基石

在数字化转型的浪潮中&#xff0c;云原生技术已成为推动企业创新和灵活性的关键力量&#x1f4a1;。然而&#xff0c;随着技术的进步和应用的广泛&#xff0c;网络安全威胁也日益严峻&#x1f513;&#xff0c;传统的网络安全模型已经难以应对复杂多变的网络环境。在这样的背景…

一个浮动绝对居中的tailwindcss

今天改进图片组件&#xff0c;遇到个SVG绝对居中的问题。想起之前大概是通过top left来实现&#xff0c;由于组件的宽高需要动态输入。不能定死宽高&#xff0c;于是想起来问GPT。刚开始老是给一些很菜的代码&#xff0c;不是我想要的 气不打一处来&#xff0c;索性给他限死框框…

积木报表Excel数据量大导出慢导不出问题、大量数据导不出问题优化方案和分析解决思路(优化前一万多导出失败,优化后支持百万级跨库表导出)

文章目录 积木报表Excel数据量大导出慢导不出问题、大量数据导不出问题优化方案和分析解决思路&#xff08;优化前一万多导出失败&#xff0c;优化后支持百万级跨库表导出&#xff09;优化结果需求背景和解决方案的思考解决方案流程描述&#xff1a;关键代码引入easy excel新建…

uniapp中uni.navigateTo传递变量

效果展示&#xff1a; 核心代码&#xff1a; uniapp中uni.navigateTo传递变量 methods: {changePages(item) {setDatas("maintenanceFunName", JSON.stringify(item)).then((res) > {uni.navigateTo({url: /pages/PMS/maintenance/maintenanceTypes/maintenanceT…

KNN课堂(分类课堂(可用kd树/特征归一化提高精度)))

实验代码&#xff1a; # 导入所需要的库 import numpy as np import pandas as pd from sklearn.ensemble import RandomForestClassifier from sklearn.model_selection import train_test_split from sklearn.neighbors import KNeighborsClassifier # 导入数据集 df pd.…

软件无线电系列——CIC滤波器

本节目录 一、CIC滤波器 1、积分器 2、梳状滤波器 3、CIC滤波器的特性 二、单级CIC和多级CIC滤波器本节内容 一、CIC滤波器 CIC滤波器&#xff0c;英文名称为Cascade Integrator Comb&#xff0c;中文全称为级联积分梳状滤波器&#xff0c;从字面来看就是将积分器与梳状滤波器…

ES学习日记(十一)-------Java操作ES之基本操作

前言 此篇博客还是一些基础操作&#xff0c;没什么可写的&#xff0c;需要的同学直接抄作业进行测试就可以 上一节写了连接和测试新增操作,这一节写java操作ES的基本操作,也就是增删改查,在这里补充一点知识,我们之前用了指定的索引进行指定添加 有一个情况是,如果我们指定了…

Visual Studio 配置代码风格审查工具cpplint

文章目录 一、Visual Studio 配置代码风格审查工具cpplint1、安装2、运行3、集成到Visual Studio4、集成到Git 前言 cpplint是一个用于检查C代码风格的工具&#xff0c;它可以帮助我们发现潜在的编码问题&#xff0c;提高代码质量。cpplint遵循Google的C编码规范&#xff0c;通…

js通过Object.defineProperty实现数据响应式

目录 数据响应式属性描述符propertyResponsive 依赖收集依赖队列寻找依赖 观察器 派发更新Observer完整代码关于数据响应式关于Object.defineProperty的限制 数据响应式 假设我们现在有这么一个页面 <!DOCTYPE html> <html lang"en"><head><m…

【JavaScript】原型链/作用域/this指针/闭包

1.原型链 参考资料&#xff1a;Annotated ES5 ECMAScript起初并不支持如C、Smalltalk 或 Java 中“类”的形式创建对象&#xff0c;而是通过字面量表示法或者构造函数创建对象。每个构造函数都是一个具有名为“prototype”的属性的函数&#xff0c;该属性用于实现基于原型的继…