Skip to main content
 Web开发网 » 站长学院 » 织梦教程

Shell编程(上)丨你不知道的玩法~

2021年11月15日5700百度已收录

一、前言

1 为什么学Shell?什么是shell?

Shell脚本语言是实现Linux/UNIX系统管理及自动化运维所必备的重要工具, Linux/UNIX系统的底层及基础应用软件的核心大都涉及Shell脚本的内容。每一个合格 的Linux系统管理员或运维工程师,都需要能够熟练地编写Shell脚本语言,并能够阅 读系统及各类软件附带的Shell脚本内容。只有这样才能提升运维人员的工作效率,适 应曰益复杂的工作环境,减少不必要的重复工作,从而为个人的职场发展奠定较好的基础。

Shell是一个命令解释器,它在操作系统的最外层,负责直接与用户对话,把用户的输入解释给操作系统,并处理各种各样的操作系统的输出结果,输出屏幕返回给用户。

2 什么是Shell脚本?

命令、变量和流程控制语句等有机的结合起来。

shell脚本擅长处理纯文本类型的数据,而linux中,几乎所有的配置文件,日志,都是纯文本类型文件。

3 脚本语言的种类

① 编译型语言

定义:指用专用的编译器,针对特定的操作平台(操作系统)将某种高级语言源代码一次性翻译成可被硬件平台直接运行的二进制机器码(具有操作数,指令、及相应的格式),这个过程叫做编译(./configure make makeinstall );编译好的可执行性文件(.exe),可在相对应的平台上运行(移植性差,但运行效率高)。

典型的编译型语言有, C语言、C++等。

② 解释型语言

定义:指用专门解释器对源程序逐行解释成特定平台的机器码并立即执行的语言;相当于把编译型语言的编译链接过程混到一起同时完成的。

解释型语言执行效率较低,且不能脱离解释器运行,但它的跨平台型比较容易,只需提供特定解释器即可。

常见的解释型语言有, Python(同时是脚本语言)与Ruby等。

③ 脚本语言

定义:为了缩短传统的编写-编译-链接-运行(edit-compile-link-run)过程而创建的计算机编程语言。

特点:程序代码即是最终的执行文件,只是这个过程需要解释器的参与,所以说脚本语言与解释型语言有很大的联系。脚本语言通常是被解释执行的,而且程序是文本文件。

典型的脚本语言有,JavaScript,Python,shell等。

其他常用的脚本语句种类:

PHP是网页程序,也是脚本语言。是一款更专注于web页面开发(前端展示)的脚本语言,例如:Dedecms,discuz。PHP程序也可以处理系统日志,配置文件等,php也可以调用系统命令。

Perl脚本语言。比shell脚本强大很多,语法灵活、复杂,实现方式很多,不易读,团队协作困难,但仍不失为很好的脚本语言,存世大量的程序软件。MHA高可用Perl写的。

Python,不但可以做脚本程序开发,也可以实现web程序以及软件的开发。近两年越来越多的公司都会要求会Python。

Shell脚本与php/perl/python语言的区别和优势?

shell脚本的优势在于处理操作系统底层的业务 (linux系统内部的应用都是shell脚本完成)因为有大量的linux系统命令为它做支撑。2000多个命令都是shell脚本编程的有力支撑,特别是grep、awk、sed等。例如:一键软件安装、优化、监控报警脚本,常规的业务应用,shell开发更简单快速,符合运维的简单、易用、高效原则。

PHP、Python优势在于开发运维工具以及web界面的管理工具,web业务的开发等。处理一键软件安装、优化,报警脚本。常规业务的应用等php/python也是能够做到的。但是开发效率和复杂比用shell就差很多了。

系统环境说明

[root@oldboy scripts]# cat /etc/redhat-release CentOS Linux release 7.4.1708 (Core) [root@oldboy scripts]# uname -r3.10.0-693.el7.x86_64[root@oldboy scripts]# getenforce Disabled[root@oldboy scripts]# systemctl status firewalld.service ● firewalld.service - firewalld - dynamic firewall daemon Loaded: loaded (/usr/lib/systemd/system/firewalld.service; disabled; vendor preset: enabled) Active: inactive (dead) Docs: man:firewalld(1) 4 系统中的shell

查看系统中的命解释器

[root@oldboy ~]# cat /etc/shells /bin/sh/bin/bash/sbin/nologin/usr/bin/sh/usr/bin/bash/usr/sbin/nologin常用操作系统的默认shell

1.Linux是Bourne Again shell(bash)2.Solaris和FreeBSD缺省的是Bourne shell(sh)3.AIX下是Korn Shell(ksh)4.HP-UX缺省的是POSIX shell(sh)[root@oldboy ~]# echo $SHELL/bin/bashbash版本

[root@oldboy scripts]# bash -versionGNU bash, 版本 4.2.46(2)-release (x86_64-redhat-linux-gnu)Copyright (C) 2011 Free Software Foundation, Inc.许可证 GPLv3+: GNU GPL 许可证版本3或者更高 < 破壳漏洞

使用 命令 env x='() { :;}; echo be careful' bash -c "echo this is a test"如果返回结果为一行,则为正常, [root@oldboy ~]# env x='() { :;}; echo be careful' bash -c "echo this is a test" this is a test #解决办法 升级当前的bash版本 yum install update bashsh与bash 的关系

[root@oldboy ~]# ll /bin/shlrwxrwxrwx. 1 root root 4 11月 13 11:15 /bin/sh -> bash/bin与 /user/bin 的关系

[root@oldboy ~]# ll /bin -dlrwxrwxrwx. 1 root root 7 11月 13 11:15 /bin -> usr/bin二、脚本书写规范

1 脚本统一存放目录

[root@oldboy ~]# mkdir -p /server/scripts/[root@oldboy ~]# cd /server/scripts/2 选择解释器

注意格式:其中开头的"#!"字符又称为幻数,在执行bash脚本的时候,内核会根据"#!"后的解释器来确定该用那个程序解释这个脚本中的内容。[root@oldboy scripts]# head -1 /etc/init.d/*==> /etc/init.d/functions <==# -*-Shell-script-*-==> /etc/init.d/netconsole <==#!/bin/bash==> /etc/init.d/network <==#! /bin/bash3 编辑脚本使用vim

使用 .vimrc 文件,能够快速的生成开头的注释信息

[root@oldboy scripts]# cat ~/.vimrc autocmd BufNewFile *.py,*.cc,*.sh,*.java exec ":call SetTitle()"func SetTitle() if expand("%:e") == 'sh' call setline(1,"#!/bin/bash") call setline(2, "##############################################################") call setline(3, "# File Name: ".expand("%")) call setline(4, "# Version: V1.0") call setline(5, "# Author: oldboy") call setline(6, "# Organization: 使用后的效果

[root@oldboy scripts]# cat scripts_test.sh #!/bin/bash############################################################### File Name: scripts_test.sh# Version: V1.0# Author: oldboy# Organization: 脚本中,跟在#后面的内容表示注释。注释部分不会被执行,仅给人看。注释可以自成一行,也可以跟在命令后面,与命令同行。要养成写注释的习惯,方便自己与他人。

最好不用中文注释,因为在不同字符集的系统会出现乱码。(字符集为zh_CN.UTF-8,为中文)。

4 文件名规范

名字要有意义,并且结尾以 .sh 结束

5 开发的规范和习惯小结

1) 放在统一的目录

2) 脚本以.sh为扩展名

3) 开头指定脚本解释器。

4) 开头加版本版权等信息,可配置~/.vimrc文件自动添加。

5) 脚本不要用中文注释,尽量用英文注释。

6) 代码书写优秀习惯

a、成对的内容一次性写出来,防止遗漏,如[ ]、' '、" "等

b、[ ]两端要有空格,先输入[ ],退格,输入2个空格,再退格写。

c、流程控制语句一次书写完,再添加内容。(if 条件 ; then 内容;fi)ddd

d、通过缩进让代码易读。

f、脚本中的引号都是英文状态下的引号,其他字符也是英文状态。

三、shell脚本的执行

1 执行脚本的办法

sh/bash scripts.sh chown +x ./scripts.sh && ./scripts.sh source scripts.sh. (空格) scripts.shcat oldboyedu.sh |bash # 效率较低source 与 . (点) 的作用

soucre命令

[root@oldboy ~]# help source |head -2source: source 文件名 [参数] 在当前 shell 中执行一个文件中的命令。. (点)

[root@oldboy scripts]# help . |head -2.: . 文件名 [参数] 在当前 shell 中执行一个文件中的命令。。2 sh 于 source的区别

Shell编程(上)丨你不知道的玩法~  dedecms技巧 第1张

[root@oldboy scripts]# sh oldboy_test.sh Hello World![root@oldboy scripts]# echo $oldboy# sh 新建一个Shell窗口(新建一个进程)执行一个文件中的命令。[root@oldboy scripts]# source oldboy_test.sh Hello World![root@oldboy scripts]# echo $oldboyHello World!四、Shell的变量

1 什么是变量

变量可以分为两类:环境变量(全局变量)和普通变量(局部变量)

环境变量也可称为全局变量,可以在创建他们的Shell及其派生出来的任意子进程shell中使用,环境变量又可分为自定义环境变量和Bash内置的环境变量

普通变量也可称为局部变量,只能在创建他们的Shell函数或Shell脚本中使用。普通变量一般是由开发者用户开发脚本程序时创建的。

特殊变量

2 环境变量

使用 env/declare/set/export -p 命令查看系统中的环境变量,这三个命令的的输出方式稍有不同。

[root@oldboy scripts]# envXDG_SESSION_ID=1HOSTNAME=oldboyTERM=linuxSHELL=/bin/bashHISTSIZE=1000SSH_CLIENT=10.0.0.1 5537 22SSH_TTY=/dev/pts/0USER=root~~~输出一个系统中的 环境变量

[root@oldboy ~]# echo $LANGzh_CN.UTF-83 普通变量

本地变量在用户当前的Shell生存期的脚本中使用。例如,本地变量OLDBOY取值为bingbing,这个值在用户当前Shell生存期中有意义。如果在Shell中启动另一个进程或退出,本地变量值将无效

定义普通变量实践

[root@oldboy ~]# a=1[root@oldboy ~]# b='2'[root@oldboy ~]# c="3"[root@oldboy ~]# echo "$a"1[root@oldboy ~]# echo "$b"2[root@oldboy ~]# echo "${c}"提示:$变量名表示输出变量,可以用$c和${c}两种用法小结:连续普通字符串内容赋值给变量,不管用什么引号或者不用引号,它的内容是什么,打印变量就输出什么4 export命令

[root@oldboy ~]# help export export: export [-fn] [名称[=值] ...] 或 export -p为 shell 变量设定导出属性。标记每个 NAME 名称为自动导出到后续命令执行的环境。如果提供了,VALUE,则导出前将 VALUE 作为赋值。export命令的说明

当前shell窗口及子shell窗口生效在新开的shell窗口不会生效,生效需要写入配置文件# 定义变量

[root@oldboy scripts]# OLDBOY=oldboy[root@oldboy scripts]# export OLDBOY1=1# 当前窗口查看

[root@oldboy scripts]# echo $OLDBOYoldboy[root@oldboy scripts]# echo $OLDBOY11# 编写测试脚本

[root@oldboy scripts]# vim quanju.sh #!/bin/bashecho $OLDBOYecho $OLDBOY1# 使用sh执行

[root@oldboy scripts]# sh quanju.sh 1# 使用source 执行

[root@oldboy scripts]# source quanju.sh oldboy15 环境变量相关配置文件

/etc/proflie/etc/bashrc~/.bashrc~/.bash_profile/etc/proflie.d/ # 目录四文件读取顺序(CentOS6和7都一样)

① /etc/profile② ~/.bash_profile③ ~/.bashrc④ /etc/bashrcShell编程(上)丨你不知道的玩法~  dedecms技巧 第2张

文件读取过程示意图

验证四文件读取顺序的方法

sed -i '1a echo "$(date +%T-%s) /etc/profile1" >>/tmp/oldboy' /etc/profilesed -i '$a echo "$(date +%T-%s) /etc/profile2" >>/tmp/oldboy' /etc/profilesed -i '1a echo "$(date +%T-%s) /etc/bashrc1" >>/tmp/oldboy' /etc/bashrcsed -i '$a echo "$(date +%T-%s) /etc/bashrc2" >>/tmp/oldboy' /etc/bashrcsed -i '1a echo "$(date +%T-%s) ~/.bashrc1" >>/tmp/oldboy' ~/.bashrcsed -i '$a echo "$(date +%T-%s) ~/.bashrc2" >>/tmp/oldboy' ~/.bashrcsed -i '1a echo "$(date +%T-%s) ~/.bash_profile1" >>/tmp/oldboy' ~/.bash_profilesed -i '$a echo "$(date +%T-%s) ~/.bash_profile2" >>/tmp/oldboy' ~/.bash_profile6 环境变量的知识小结

❶ 变量名通常要大写。

❷ 变量可以在自身的Shell及子Shell中使用。

❸ 常用export来定义环境变量。

❹ 执行env默认可以显示所有的环境变量名称及对应的值。

❺ 输出时用“$变量名”,取消时用“unset变量名”。

❻ 书写crond定时任务时要注意,脚本要用到的环境变量最好先在所执行的Shell脚本中重新定义。

❼ 如果希望环境变量永久生效,则可以将其放在用户环境变量文件或全局环境变量文件里。

7 变量中引号的使用

只有在变量的值中有空格的时候,会使用引号。单引号与双引号的区别在于,是否能够解析特殊符号。[root@oldboy ~]# name=oldboyedu[root@oldboy ~]# name2='oldboy'[root@oldboy ~]# name3=" 普通变量的要求

1) 内容是纯数字、简单的连续字符(内容中不带任何空格)时,定义时可以不加任何引号,例如:

a.oldboyAge=22b.NETWORKING=yes2) 没有特殊情况时,字符串一律用双引号定义赋值,特别是多个字符串中间有空格时,例如:

a.NFSD_MODULE="no load"b.MyName="Oldboy is a handsome boy."3) 当变量里的内容需要原样输出时,要用单引号(M),这样的需求极少,例如:

a.OLDBOY_NAME='OLDBOY'变量使用反引号赋值

[root@oldboy scripts]# time=`date`[root@oldboy scripts]# echo $time2017年 12月 05日 星期二 09:02:06 CST[root@oldboy scripts]# file=`ls`[root@oldboy scripts]# echo $fileoldboy_test.sh panduan.sh quanju.sh yhk.sh使用${}

打印变量的时候防止出现“金庸新著”的问题

[root@oldboy scripts]# time=`date`[root@oldboy scripts]# echo $time_day[root@oldboy scripts]# echo ${time}_day2017年 12月 05日 星期二 09:02:06 CST_day[root@oldboy scripts]# echo $time-day2017年 12月 05日 星期二 09:02:06 CST-day编写脚本测试${}

# 使用脚本测试 [root@oldboy scripts]# vim bianliang.sh #!/bin/bash ############################################################# # File Name: bianliang.sh # Version: V1.0 # Author: oldboy # Organization: 定义变量名技巧

1. 变量名只能为字母、数字或下划线,只能以字母或下划线开头。

2. 变量名的定义要有一定的规范,并且要见名知意。

示例:

oldboyAge=22 #<==每个单词的首字母大写的写法oldboy_age=22 #<==单词之间用"_"的写法oldboyAgeSex=man #<==驼峰语法:首个单词的首字母小写,其余单词首字母大写oldboyAGE=22 #<==单词全大写的写法3. 一般的变量定义、赋值常用双引号;简单连续的字符串可以不加引号;希望原样输出时使用单引号。

4. 希望变量的内容是命令的解析结果时,要用反引号'',或者用$()把命令括起来再赋值。

五、特殊变量

1 位置变量

常用的特殊位置参数说明

位置变量作用说明$0 获取当前执行的shell脚本的文件名,如果执行脚本带路径那么就包括脚本路径。$n获取当前执行的shell脚本的第n个参数值,n=1..9,当n为0时表示脚本的文件名,如果n大于9用大括号括起来{10},参数以空格隔开。$#获取当前执行的shell脚本后面接的参数的总个数$*获取当前shell的所有传参的参数,不加引号同$@;如果给$*加上双引号,例如: “$*”,则表示将所有的参数视为单个字符串,相当于“112$3”。$@获取当前shell的所有传参的参数,不加引号同$*;如果给$@加上双引号,例如: “$@”,则表示将所有参数视为不同的独立字符串,相当于“$1” “$2” “$3” “……”,这是将参数传递给其他程序的最佳方式,因为他会保留所有内嵌在每个参数里的任何空白。当“$*”和“$@”都加双引号时,两者有区别,都不加双引号时,两者无区别。

0,0,1.$2 ~ 参数实践

[root@oldboy scripts]# vim chanshu.sh#!/bin/bash############################################################# # File Name: chanshu.sh# Version: V1.0# Author: oldboy# Organization: $# 参数实践

[root@oldboy scripts]# vim chanshu.sh############################################################## File Name: chanshu.sh# Version: V1.0# Author: oldboy# Organization: $* 参数实践

[root@oldboy scripts]# vim chanshu.sh############################################################## File Name: chanshu.sh# Version: V1.0# Author: oldboy# Organization: $* 与 $@ 对比实践

[root@oldboy scripts]# set -- "I am" handsome boy..[root@oldboy scripts]# echo $1I am[root@oldboy scripts]# echo $2handsome[root@oldboy scripts]# echo $3boy..[root@oldboy scripts]# echo $*I am handsome boy..[root@oldboy scripts]# echo $@I am handsome boy..[root@oldboy scripts]# for i in $*;do echo $i ;doneIamhandsomeboy..[root@oldboy scripts]# for i in $@;do echo $i ;doneIamhandsomeboy..[root@oldboy scripts]# for i in "$@";do echo $i ;doneI amhandsomeboy..[root@oldboy scripts]# for i in "$*";do echo $i ;doneI am handsome boy..2 进程状态变量

Shell进程的特殊状态变量说明

位置变量作用说明$?获取执行上一个指令的执行状态返回值(0为成功,非零为失败),这个变量最常用$$获取当前执行的Shell脚本的进程号(PID),这个变量不常用,了解即可$!获取上一个在后台工作的进程的进程号(PID),这个变量不常用,了解即可$_获取在此之前执行的命令或脚本的最后一个参数,这个变量不常用,了解即可

进程参数实践

[root@oldboy scripts]# echo $?0[root@oldboy scripts]# echo $$1368[root@oldboy scripts]# echo $![root@oldboy scripts]# echo $_echo3 echo参数说明

参数参数说明-n不要追加换行-e启用下列反斜杠转义的解释-E显式地抑制对于反斜杠转义的解释`echo' 对下列反斜杠字符进行转义:\n换行\r回车\t横向制表符\b退格\v纵向制表符\c抑制更多的输出

六、定义变量的方式

1 三种定义变量的方式

① 直接赋值

② 传参 (传递参数)

③ 交互式设置变量,使用read命令

2 read命令说明

在命令行中使用

[root@oldboy scripts]# read132[root@oldboy scripts]# echo $REPLY 132[root@oldboy scripts]# read oldboy 456[root@oldboy scripts]# echo $oldboy456[root@oldboy scripts]# echo $REPLY 132在脚本中使用

[root@oldboy scripts]# vim oldboy_test.sh #!/bin/bashread -p '请输入:' oldboyecho $oldboy执行结果

[root@oldboy scripts]# sh oldboy_test.sh 请输入:oldboy_oldboyeduoldboy_oldboyeduread命令的帮助说明

[root@oldboy scripts]# read --help-bash: read: --: 无效选项read: 用法:read [-ers] [-a 数组] [-d 分隔符] [-i 缓冲区文字] [-n 读取字符数] [-N 读取字符数] [-p 提示符] [-t 超时] [-u 文件描述符] [-s不显示终端的任何输入] [名称 ...]3 定义方法实践

直接赋值方法

[root@oldboy scripts]# vim bianliang.sh# File Name: bianliang.sh# Version: V1.0# Author: oldboy# Organization: 传参 (传递参数)

[root@oldboy scripts]# vim bianliang.sh############################################################### File Name: bianliang.sh# Version: V1.0# Author: oldboy# Organization:

[root@oldboy scripts]# vim yhk.sh #!/bin/bash############################################################### File Name: yhk.sh# Version: V1.0# Author: oldboy# Organization: 地址

脚本内容↓

[root@oldboy scripts]# cat xiugaizhuji.sh #!/bin/bash############################################################## File Name: jiaohu.sh# Version: V1.0# Author: oldboy# Organization: 脚本测试结果

[root@oldboy scripts]# sh xiugaizhuji.sh请输入主机名:oldboy请输入IP地址的主机位:180是否重启服务器:{yes/no}yes系统将在10秒后重启![root@oldboy scripts]# sh xiugaizhuji.sh请输入主机名:oldboy请输入IP地址的主机位:180是否重启服务器:{yes/no}no请稍后手动重启!本文内容来自 老男孩Linux云计算运维优秀学员课后笔记整理,《跟老男孩学习Linux运维:Shell编程实战》一书Shell编程(上)丨你不知道的玩法~  dedecms技巧 第3张

评论列表暂无评论
发表评论
微信