传奇广告查询第一站 同步54.com

shell脚本简略
原创 于2026-01-05 18:07:00发布
26 阅读
0
0

目录

概述

脚本

头部解释器

脚本信息注释

脚本参数

变量申明

基本变量

脚本路径

时间

数组

脚本主体

单引号、倒引号和双引号的区别

测试语句

循环语句

条件语句

分支语句

函数


概述

shell脚本不管是开发还是运维,肯定都会遇到的。本文作者从脚本长啥样子来稍微阐述一下,本文对于那些熟悉linux系统命令但是没有写过脚本的人比较合适,内容上面比较简单,基本覆盖了常见的一些脚本的内容,但是具体的脚本功能需要配合不同的软件命令来进行针对性编写。

脚本

头部解释器

一般脚本的第一行都会带上

需要bash解释器的脚本

#! /bin/bash

需要python解释器的脚本

#! /bin/python 或者 #! /usr/bin/python

此外还有其他的一些类型,可以从/etc/shells里查到其他的解释器。其中最常用的几种是: Bourne shell (sh)、C shell (csh) 和 Korn shell (ksh), 各有优缺点。Bourne shell 是 UNIX 最初使用的 shell,并且在每种 UNIX 上都可以使用, 在 shell 编程方面相当优秀,但在处理与用户的交互方面做得不如其他几种shell。Linux 操作系统缺省的 shell 是Bourne Again shell,它是 Bourne shell 的扩展,简称 Bash,与 Bourne shell 完全向后兼容,并且在Bourne shell 的基础上增加、增强了很多特性。Bash放在/bin/bash中,它有许多特色,可以提供如命令补全、命令编辑和命令历史表等功能,它还包含了很多 C shell 和 Korn shell 中的优点,有灵活和强大的编程接口,同时又有很友好的用户界面。

/bin/sh /sbin/nologin /bin/dash /bin/tcsh /bin/csh 

应该说, /bin/sh 与 /bin/bash 虽然大体上没什么区别, 但仍存在不同的标准. 标记为 “#!/bin/sh” 的脚本不应使用任何 POSIX 没有规定的特性 (如 let 等命令, 但 “#!/bin/bash” 可以). Debian 曾经采用 /bin/bash 更改 /bin/dash,目的使用更少的磁盘空间、提供较少的功能、获取更快的速度。但是后来经过 shell 脚本测试存在运行问题。因为原先在 bash shell 下可以运行的 shell script (shell 脚本),在 /bin/sh 下还是会出现一些意想不到的问题,不是100%的兼用。

脚本信息注释

这部分是脚本的作用和编写时间,包含编写的人,修改的内容等等,没有固定的格式,但是建议这部分作为脚本的,可以参考以下格式

#-------------------------------------------- # name:shell.sh # author:xxxx # date:2019-06-01 # description:xxxxxxx # parameter:$0:xxx $1:xxx $2:xxx #-------------------------------------------- ##### 用户配置区 开始 ##### # # # 这里可以添加脚本描述信息 # # ##### 用户配置区 结束 ##### --------------------- 

脚本参数

脚本参数是指用户运行脚本后面带的参数,为了让脚本的灵活性更强,通常会在脚本后面加入参数传递给脚本进行执行。但是建议脚本后面的参数不要超过三个。脚本后面的参数依此是$1 $2 $3等等,$0是脚本本身的字符串名。但是注意为了让脚本出错概率较小,在脚本中对后面的参数个数进行检查,如限制一个参数。$#输出的是脚本后面所有的参数个数。

$n :表示第几个参数,$1 表示第一个参数,$2 表示第二个参数,依此类推 $0:当前程序的名称 $# :传递给程序的总的参数数目 $? :上一个代码或者shell程序在shell中退出的情况,如果正常退出则返回0,反之为非0值。 $* :传递给程序的所有参数组成的字符串。 $@ :以"参数1" "参数2" ... 形式保存所有参数 $$ :本程序的(进程ID号)PID $! :上一个命令的PID

例子:

if [ $# -ne 1 ];then echo -e "Usage: ./vacuumgp.sh < dbname > \n " echo -e "Example : ./vacuumgp.sh postgres" exit 8 fi

变量申明

基本变量

基本变量有字符串和数字。shell是一个弱类型的,赋值不需要进行类型声明。

my_path="/usr/local" num=10

脚本路径

为了体现脚本的良好移植性,不管脚本在什么目录下,都能够正确执行不报错,那么文件的路径就很重要,因为一般来说,脚本同时会输出日志等信息,可能还会调用其他同目录的脚本,就需要得到脚本的一个当前目录的变量,用于程序主体调用。

mypath=$(cd `dirname $0`;pwd) cd "${mypath}"

时间

获取今天的日期

`date '+%Y%m%d %H:%M:%S'` 或 `date +%F` 或 $(date '+%Y%m%d %H:%M:%S')

定义文件名

filename="`date +%y%m%d`_etc.tar.gz" 

详细的时间域如下:

格式详细
%H小时(00..23)
%I小时(01..12)
%k小时(0..23)
%l小时(1..12)
%M分(00..59)
%p显示出AM或PM
%r时间(hh:mm:ss AM或PM),12小时
%s从1970年1月1日00:00:00到目前经历的秒数
%S秒(00..59)
%T时间(24小时制)(hh:mm:ss)
%X显示时间的格式(%H:%M:%S)
%Z时区 日期域
%a星期几的简称( Sun..Sat)
%A星期几的全称( Sunday..Saturday)
%b月的简称(Jan..Dec)
%B月的全称(January..December)
%c日期和时间( Mon Nov 8 14:12:46 CST 1999)
%d一个月的第几天(01..31)
%D日期(mm/dd/yy)
%h和%b选项相同
%j一年的第几天(001..366)
%m月(01..12)
%w一个星期的第几天(0代表星期天)
%W一年的第几个星期(00..53,星期一为第一天)
%x显示日期的格式(mm/dd/yy)
%y年的最后两个数字( 1999则是99)
%Y年(例如:1970,1996等)

数组

bash shell只支持一维的数组,使用()来定义,元素之间用空格分开。如

# 定义一个数组 table_list=("table1" "table2" "table3") list=(1 2 3 4 5) #获取数组带下标即可 table_list[0] table_list[2] #打印变量需要使用花括号 echo ${table_list[0]} echo ${table_list[2]} #获取所有的变量 table_list[*] #或者 table_list[@] #打印所有变量,本身是一个迭代器 echo ${table_list[*]} # 打印数组长度 echo ${#table_list[@]} echo ${#table_list[*]} 

脚本主体

单引号、倒引号和双引号的区别

脚本主体中包含了很多的内容,除了逻辑结构,还有大量的命令结构,比如将脚本中执行linux命令结果字符串传入到变量,在脚本中输出格式化字符串等等。最常用的就是下面的三个了。

` :如果被倒引号括起来,表示里面执行的是命令。

“” :如果被双引号括起来,里面出现$(表示取变量名),`表示执行命令,\表示转义,其余的才表示字符串。

'' :如果被单引号括起来,里面表示的全部是字符串。

测试语句

测试语句在实际中使用也非常多,要注意条件测试部分中的空格。在方括号的两侧都有空格,在-f、-lt、=等符号两侧同样也有空格。如果没有这些空格,Shell解释脚本的时候就会出错。

# [ -f "$file" ] : 判断$file是否是一个文件 [ -e "$file" ] : 判断$file是否存在 [ -x "$file" ] : 判断$file是否存在且有可执行权限,同样-r测试文件可读性 [ -n "$a" ] : 判断变量$a是否有值 [ -z "$a" ] : 判断变量$a是否为空字符串 [ $a -lt 3 ] : 判断$a的值是否小于3,同样-gt和-le分别表示大于或小于等于 [ "$a" = "$b" ] : 判断$a和$b的取值是否相等 [ cond1 -a cond2 ] : 判断cond1和cond2是否同时成立,-o表示cond1和cond2有一成立

实际一个例子,判断当前文件夹是否存在配置文件kong.conf,不存在的话就创建

if [ ! -e "kong.conf" ]; then touch kong.conf fi

循环语句

关于循环,日常作者使用比较多的就是遍历数组中的元素,对数组中的元素逐个进行操作。

  • while循环格式
while [ cond1 ] && { || } [ cond2 ] …; do … done

下面列举一个小例子,将hosts文件循环读出来。

while read line; do echo $line; done < /etc/hosts; #另外一个例子,方便理解 i=10; while [[ $i -gt 5 ]]; do echo $i; ((i--)); done;
  • for循环格式
for var in …; do … done #或者 for (( cond1; cond2; cond3 )) do … done

下面仅列出一个小例子。

有一个vmstat的脚本定时将vmstat的结果append到一个文本文件中,但是每21行都会出现表头,对后续exp造成不便。

#!/bin/bash Cnt=$(cat vmstat_169_2020-03-17.log |wc -l) loopCnt=0 t1=0 t2=0 sed -i '24,25d' vmstat_169_2020-03-17.log echo "sed -i '24,25d' tmp.log " echo "-----------------------------------------------------" for(( i=3; i
tables=("table1" "table2" "table3") database="mydb" for table in ${tables[*]} do vacuumdb --analyze --table $table $database echo "table $table has finished vacuum.">>/tmp/pg_vacuum.log done 
  • until循环
until [ cond1 ] && { || } [ cond2 ] …; do … done

下面给一个小例子:

a=10; until [[ $a -lt 0 ]]; do echo $a; ((a--)); done;

条件语句

if条件语句在进行逻辑判断时非常有用。

格式:

if …; then … elif …; then … else … fi

实际例子:这个例子功能是检查目标文件夹是否存在输入日期的文件夹,如果存在则往下执行,如果不存在则创建。

read -p "Please input the Date(eg.20180717):" data_date if [ ! -d /data1/remote/$IP/${data_date:0:4}/${data_date:0:6} ]; then echo "The target folder is not exist,creating..." mkdir -p /data1/remote/$IP/${data_date:0:4}/${data_date:0:6} echo "Target folder is created!" fi 

需要注意的是:

1、if[ ] 条件内容一定要写在两个空格之间。

2、if[ ] 之后的分号一定要有

分支语句

case/esac语句,这个在实际中出现的场景是脚本和用户进行简单的交互,例如序号选择,yes/no等。可以在linux系统中使用help case来查看用法。

case var in pattern 1 ) … ;; pattern 2 ) … ;; *) … ;; esac 

例子,由脚本的第一个参数来判断执行哪一部分的脚本

case $1 in start | begin) echo "start something" ;; stop | end) echo "stop something" ;; *) echo "Ignorant" ;; esac

有一个更为常用的select语句配合使用,用户可以从一组不同的值中进行选择。

格式:

select var in …; do break; done

例子,这里需要注意的是,用户输入的是1,2或3

select ch in "postgresql" "mysql" "exit"; do case $ch in "postgresql") echo "start install postgresql" ;; "mysql") echo "start install mysql" ;; "exit") echo "exit" break; ;; *) echo "error,please select "1~3"" ;; esac done;

函数

函数的一个好处就是可以重复调用,但是一般来说,代码中完成的功能比较简单和单一,但是对于一些较大的脚本来说,函数是必不可少的。

下面举一个非常贱的函数例子,功能是计算所有参数的和。

#!/bin/bash function getsum(){ local sum=0 for n in $@ do ((sum+=n)) done return $sum } getsum 10 20 55 15 #调用函数并传递参数 echo $?

注:$@表示函数的所有参数,$?表示函数的退出状态(返回值)

管理员
0
0
0
分享
上一篇: android自动秒杀脚本,gg修改器执行脚本按键代码秒杀安卓
下一篇: 网页传奇服务器端,拍拍科技武易传奇神鸟归来商业版+网站
评论
历史记录
回顶部
浏览时间 游戏名称 游戏IP 开区网址
注册GM1论坛账号
  • 上传头像
注册

已有账号,

微信扫码登录
重置密码
重置密码

注册

绑定关联手机号
关联手机号