在进行脚本编写的时候,有时候我们需要判断当前的条件是否符合我们要执行的命令,这时候便可以使用if语句
if语句格式:
单条件判断:
if 判断条件;then
条件判断为真时执行的代码命令
fi
双条件判断:
if 判断条件;then
条件判断为真时执行的代码命令
elif 判断条件2;then
条件判断为真时执行的代码命令
fi
多分支判断:
if 判断条件1;then
条件判断为真时执行的代码命令
elif 判断条件2;then
条件判断为真时执行的代码命令
elif 判断条件3;then
条件判断为真时执行的代码命令
else
上诉条件都不满足时执行的代码命令
fi
*注意:条件判断若是满足就是立即执行当前条件下的代码命令,执行完毕后便不会再去执行其他的条件判断,if语句就结束了。
例子1:编写脚本 createuser.sh,实现如下功能:使用一个用户名做为参数,如果 指定参数的用户存在,就显示其存在,否则添加之;显示添加的用户的id号等信息
#!/bin/bash
if [ -z $1 ]
then
echo "your enter is null,please enter name."
elif [ ! -z $1 ]
then
grep "$1" /etc/passwd &>/dev/null
if [ $? -eq 0 ]
then
echo "user $1 is exits"
else
echo "starting add user"
useradd $1
id $1
fi
fi
上面的执行结果貌似是正确的,但是存在着一些错误,前面我们说了,if的条件判断如果不符合,就会结束或者执行下一个条件判断语句,所以elif中的条件可以不再判断,因为1来判断更好,不会出现这种情况。现在,做出以下修改。
#!/bin/bash
if [ -z $1 ]
then
echo "your enter is null,please enter name."
else
id "$1" &>/dev/null
if [ $? -eq 0 ]
then
echo "user $1 is exits"
else
echo "starting add user"
useradd $1
id $1
fi
fi
在if条件语句中的判断方式有以下几种。使用格式 if [ 条件1 options 条件2 ],注意方括号的前后都含有空格
#1.数值测试
-eq 是否等于
-ne 是否不等于
-gt 是否大于
-ge 是否大于等于
-lt 是否小于
-le 是否小于等于
#2.字符串测试
-z STRING 字符串是否为空,没定义或空为真,不空为假,
-n STRING 字符串是否不空,不空为真,空为假
#3.文件测试
-a FILE:同 -e
-e FILE: 文件存在性测试,存在为真,否则为假
-b FILE:是否存在且为块设备文件
-c FILE:是否存在且为字符设备文件
-d FILE:是否存在且为目录文件
-f FILE:是否存在且为普通文件
-h FILE 或 -L FILE:存在且为符号链接文件
-p FILE:是否存在且为命名管道文件
-S FILE:是否存在且为套接字文件
-r FILE:是否存在且可读
-w FILE:是否存在且可写
-x FILE:是否存在且可执行
FILE1 -nt FILE2 : 文件FILE1是否比文件FILE2新
FILE1 -ot FILE2 : 文件FILE1是否比文件FILE2旧
上面的案例中使用了$1符号,这代表什么呢?
在shell脚本中,$0,$1,$@,$*都代表着不同的含义
$0代表文件名的本身,
$1代表命令行中的第一个参数
$2代表命令行中的第二个参数
$@代表将参数作为单独的个体
$*代表将参数作为一个整体。
看看例子
#!/bin/bash
echo $0
echo $1
echo $2
echo -------------------
for i in "$@";do
echo "$i"
done
for i in "$*";do
echo $i
done
[root@iz2zed4h7bg5dihzfqhmn9z script]# ./test.sh 1 2 3 4
./test.sh
1
2
-------------------
1
2
3
4
1 2 3 4
这是可以看出$@是将命令行中的参数当作单独的个体来输出的,而$*是将命令行中的参数当作整体来输出的。
在有时候进行脚本编写时,每次创建脚本都需要输入#!/bin/bash这一行,重复的工作甚是无聊,那么有没有什么简单的方法可以避免没次都输入这样的一行呢?
vimrc文件:vim编辑器的相关的所有功能开关都可以通过.vimrc文件来设置。
.vimrc文件分为系统配置文件和用户配置文件
可以使用 vim --version来查看两个文件的存放路径
注意:用户配置文件优先于系统配置文件
常用配置
"===================
"适合自己用的vimrc配置文件
"===================
"设置编码
set fileencodings=utf-8,ucs-bom,gb18030,gbk,gb2312,cp936
set termencoding=utf-8
set encoding=utf-8
"保存.vimrc文件时自动重启加载,即让此文件立即生效
autocmd BufWritePost $MYVIMRC source $MYVIMRC
"语法高亮
syntax on
"设置ruler会在右下角显示光标所在的行号和列号,不方便查看,改成设置状态栏显示内容
"set ruler
"设置状态行显示的内容. %F: 显示当前文件的完整路径. %r: 如果readonly,会显示[RO]
"%B: 显示光标下字符的编码值,十六进制. %l:光标所在的行号. %v:光标所在的虚拟列号.
"%P: 显示当前内容在整个文件中的百分比. %H和%M是strftime()函数的参数,获取时间.
set statusline=%F%r\ [HEX=%B][%l,%v,%P]\ %{strftime(\"%H:%M\")}
"显示行号
set nu "等同于 set number
"突出显示当前行
set cursorline "等同于 set cul
"突出显示当前列
set cursorcolumn "等同于 set cuc
"共享剪贴板
set clipboard+=unnamed
"从不备份
set nobackup
"自动保存
set autowrite
"隐藏工具栏
"set guioptions-=T
"隐藏菜单栏
"set guioptions-=m
"高亮显示所有搜索到的内容.后面用map映射快捷键来方便关闭当前搜索的高亮.
"set hlsearch
"光标立刻跳转到搜索到内容
"set incsearch
"搜索到最后匹配的位置后,再次搜索不回到第一个匹配处
"set nowrapscan
"去掉输入错误时的提示声音
set noeb
" 默认按下Esc后,需要等待1秒才生效,设置Esc超时时间为100ms,尽快生效
set ttimeout
set ttimeoutlen=100
"在处理未保存或只读文件的时候,弹出确认
set confirm
"让Backspace键可以往前删除字符.
"Debian系统自带的vim版本会加载一个debian.vim文件,默认已经设置这一项,
"可以正常使用Backspace键.如果使用自己编译的vim版本,并自行配置.vimrc文件,
"可能就没有设置这一项,导致Backspace键用不了,或者时灵时不灵.所以主动配置.
"使回格键(backspace)正常处理indent, eol, start等
set backspace=indent,eol,start
"允许backspace和光标键跨越行边界
"set whichwrap+=<,>,h,l
"去掉有关vi一致性模式,避免操作习惯上的局限.
set nocompatible
"FIXME 在MS-DOS控制台打开vim时,控制台使用鼠标右键来复制粘贴,设置
"全鼠标模式,鼠标右键被映射为visual mode,不能用来复制粘贴,不方便.
"但是如果不设置鼠标模式,会无法使用鼠标滚轮来滚动界面.经过验证,发现
"可以设成普通模式mouse=n来使用鼠标滚轮,也能使用鼠标右键复制粘贴.
" mouse=c/mouse=i模式都不能用鼠标滚轮. Linux下还是要设成 mouse=a
set mouse=n
"set selection=exclusive
"set selectmode=mouse,key
"高亮显示括号匹配
set showmatch
"设置Tab长度为4空格
set tabstop=4
"设置自动缩进长度为4空格
set shiftwidth=4
"自动缩进,这个导致从外面拷贝多行以空格开头的内容时,会有多的缩进,先不设置
"set autoindent
"不要用空格代替制表符
set noexpandtab
"输入tab制表符时,自动替换成空格
"set expandtab
"设置softtabstop有一个好处是可以用Backspace键来一次删除4个空格.
"softtabstop的值为负数,会使用shiftwidth的值,两者保持一致,方便统一缩进.
"set softtabstop=4
"显示空格和tab键
set listchars=tab:>-,trail:-
"1=启动显示状态行, 2=总是显示状态行.设置总是显示状态行,方便看到当前文件名
set laststatus=2
"自动补全
:inoremap ( ()<ESC>i
:inoremap ) <c-r>=ClosePair(')')<CR>
:inoremap { {<CR>}<ESC>O
:inoremap } <c-r>=ClosePair('}')<CR>
:inoremap [ []<ESC>i
:inoremap ] <c-r>=ClosePair(']')<CR>
:inoremap " ""<ESC>i
:inoremap ' ''<ESC>i
function! ClosePair(char)
if getline('.')[col('.') - 1] == a:char
return "\<Right>"
else
return a:char
endif
endfunction
"打开文件类型检测,并载入文件类型插件,为特定文件类型载入相关缩进文
filetype plugin indent on
" 设置自动补全的选项. longest表示只自动补全最大匹配的部分,剩余部分通过
" CTRL-P/CTRL-N来选择匹配项进行补全. menu表示弹出可补全的内容列表.
" 如果有多个匹配,longest选项不会自动选中并完整补全,要多按一次CTRL-P,比较
" 麻烦,不做设置,保持默认设置,vim默认没有设置longest.
"set completeopt=longest,menu "启用这句才会开启自动补全
"设置背景主题
"color asmanian2
"设置字体
"set guifont=Courier_New:h10:cANSI
"设置颜色主题,适用于黑色背景.
colorscheme slate
"=============显示中文帮助
if version >= 603
set helplang=cn
set encoding=utf-8
endif
"=============新建.c,.h,.sh,.java文件,自动插入文件头
autocmd BufNewFile *.cpp,*.[ch],*.sh,*.java exec ":call SetTitle()"
""定义函数SetTitle,自动插入文件头
func SetTitle()
"如果文件类型为.sh文件
if &filetype == 'sh'
call setline(1,"\############################")
call append(line("."), "\# File Name: ".expand("%"))
call append(line(".")+1, "\# Author: LPeng")
call append(line(".")+2, "\# mail: 7526@qq.com")
call append(line(".")+3, "\# Created Time: ".strftime("%c"))
call append(line(".")+4, "\############################")
call append(line(".")+5, "\#!/bin/bash")
call append(line(".")+6, "")
else
call setline(1, "/******************************")
call append(line("."), " > File Name: ".expand("%"))
call append(line(".")+1, " > Author:LPeng")
call append(line(".")+2, " > Mail: 7526@qq.com ")
call append(line(".")+3, " > Created Time: ".strftime("%c"))
call append(line(".")+4, " *****************************/")
call append(line(".")+5, "")
endif
"新建文件后,自动定位到文件末尾
autocmd BufNewFile * normal G
endfunc
现在根据上面的信息来编写生成脚本基本格式的脚本,包括作者,联系方式,版本,时间,描述等
set ts=4
set autowrite
autocmd BufNewFile *.cpp,*.[ch],*.sh,*.java exec ":call SetTitle()"
func SetTitle()
if expand("%:e") == 'sh'
call setline(1,"#!/bin/bash")
call setline(2,"#")
call setline(3,"###################################")
call setline(4,"#Author: xuxiaolong")
call setline(5,"#Date: ".strftime("%Y-%m-%d"))
call setline(6,"#FileName ".expand("%"))
call setline(7,"#Description: The Test script")
call setline(8,"#Mail: longge@163.com")
call setline(9,"#Copyright (C): ".strftime("%Y")." All Rights Reserved")
call setline(10,"###################################")
call setline(11,"")
endif
endfunc
autocmd BufNewFile * normal G
设置好之后创建脚本文件就更方便了
[root@iz2zed4h7bg5dihzfqhmn9z ~]# vim test1.sh
#!/bin/bash
#
###################################
#Author: xuxiaolong
#Date: 2020-12-25
#FileName test1.sh
#Description: The Test script
#Mail: longge@163.com
#Copyright (C): 2020 All Rights Reserved
###################################
再来说说文件查找命令find
find是实时查找工具,通过遍历指定路径完成文件查找
格式:find [ optinons ] [ 查找路径] [查找条件] [处理动作]
查找路径:指定具体目标路径,默认为当前目录
查找条件:指定查找的标准
处理动作:对符合条件的问价做操作,默认为输出打印到当前屏幕
options :-maxdepth level 最大搜索目录深度,指定目录下的文件为第1级
-mindepth level 最小搜索目录深度
1. 按照文件名和inode查找
-name "文件名称":支持使用glob,如:*, ?, [], [^],通配符要加双引号引起来
-iname "文件名称":不区分字母大小写
-inum n 按inode号查找
-samefile name 相同inode号的文件
-links n 链接数为n的文件
-regex “PATTERN”:以PATTERN匹配整个文件路径,而非文件名称
2.按照属主、属组查找
-user USERNAME:查找属主为指定用户(UID)的文件
-group GRPNAME: 查找属组为指定组(GID)的文件
-uid UserID:查找属主为指定的UID号的文件
-gid GroupID:查找属组为指定的GID号的文件
-nouser:查找没有属主的文件
-nogroup:查找没有属组的文件
3.按照文件的类型来查找
-type TYPE
TYPE可以是以下形式:
f: 普通文件
d: 目录文件
l: 符号链接文件
s:套接字文件
b: 块设备文件
c: 字符设备文件
p: 管道文件
4. 组合条件
与:-a ,默认多个条件是与关系
或:-o
非:-not !
5.按照文件大小来查找
-size [+|-]#UNIT
常用单位:k, M, G,c(byte),注意大小写敏感
#UNIT: (#-1, #]
如:6k 表示(5k,6k]
-#UNIT:[0,#-1]
如:-6k 表示[0,5k]
+#UNIT:(#,∞)
如:+6k 表示(6k,∞)
6.按照文件权限来查找
-perm [/|-]MODE
MODE: 精确权限匹配
/MODE:任何一类(u,g,o)对象的权限中只要能一位匹配即可,或关系,+ 从CentOS 7开始淘汰
-MODE:每一类对象都必须同时拥有指定权限,与关系
0 表示不关注
7.按照文件时间查找
以“天”为单位
-atime [+|-]#
#: [#,#+1)
+#: [#+1,∞]
-#: [0,#)
-mtime
-ctime
以“分钟”为单位
-amin
-mmin
-cmin
8.处理动作
-print:默认的处理动作,显示至屏幕
-ls:类似于对查找到的文件执行“ls -l”命令
-fls file:查找到的所有文件的长格式信息保存至指定文件中,相当于 -ls > file
-delete:删除查找到的文件,慎用!
-ok COMMAND {} \; 对查找到的每个文件执行由COMMAND指定的命令,对于每个文件执行命令之前,都会交互式要求用户确认
-exec COMMAND {} \; 对查找到的每个文件执行由COMMAND指定的命令{}: 用于引用查找到的文件名称自身
特别说明:find命令根据时间查找详解
find命令认为24小时之内是第0天,下面是帮助理解的图形。
例子:比如我现在时间是2020-02-22 12:00想要查找/etc目录下3天内被修改过的文件。
find /etc -mtime -2
例子:那想查找前面第3天的文件呢?
第0天就是(2020-02-22 12:00——2020-02-21 12:00),
第1天就是(2020-02-21 12:00——2020-02-20 12:00)
第2天就是(2020-02-20 12:00——2020-02-19 12:00)
第3天就是(2020-02-19 12:00——2020-02-18 12:00)
结果就是 find /etc -mtime 3 (其实找到的就是2020-02-19 12:00——2020-02-18 12:00之间被修改过的文件)虽然这是写的第三天,但其实真正的是第4天的数据文件。
例子:如果我想查找3天前的文件呢?
find /etc -mtime +3
例子:查找/etc目录下大于1M且类型为普通文件的所有文件
[root@iz2zed4h7bg5dihzfqhmn9z ~]# find /etc/ -size +1M -a -type f -exec ls -lh {} \;
-r--r--r-- 1 root root 8.0M Jul 5 14:33 /etc/udev/hwdb.bin
-rw-r--r-- 1 root root 1.4M Jul 1 13:02 /etc/selinux/targeted/contexts/files/file_contexts.bin
-rw-r--r-- 1 root root 3.8M Jul 1 13:02 /etc/selinux/targeted/policy/policy.31
-rw------- 1 root root 3.8M Jul 1 13:02 /etc/selinux/targeted/active/policy.linked
-rw------- 1 root root 3.8M Jul 1 13:02 /etc/selinux/targeted/active/policy.kern
上面我们说了,find是遍历整个目录,那隐藏文件是否依然会被查找出来呢?试试看。
[root@iz2zed4h7bg5dihzfqhmn9z etc]# dd if=/dev/zero of=/etc/.f2.img bs=1M count=10
10+0 records in
10+0 records out
10485760 bytes (10 MB) copied, 0.00742828 s, 1.4 GB/s
[root@iz2zed4h7bg5dihzfqhmn9z etc]# ls -l | grep f2
[root@iz2zed4h7bg5dihzfqhmn9z etc]# find /etc/ -size +1M -a -type f -exec ls -lh {} \;
-rw-r--r-- 1 root root 10M Dec 25 10:39 /etc/.f2.img
-r--r--r-- 1 root root 8.0M Jul 5 14:33 /etc/udev/hwdb.bin
-rw-r--r-- 1 root root 1.4M Jul 1 13:02 /etc/selinux/targeted/contexts/files/file_contexts.bin
-rw-r--r-- 1 root root 3.8M Jul 1 13:02 /etc/selinux/targeted/policy/policy.31
-rw------- 1 root root 3.8M Jul 1 13:02 /etc/selinux/targeted/active/policy.linked
-rw------- 1 root root 3.8M Jul 1 13:02 /etc/selinux/targeted/active/policy.kern
上面的例子证明find命令适用于查找出隐藏文件。
例子2 :查找当前系统上没有属主或属组,且最近一个周内曾被访问过的文件或目录
[root@iz2zed4h7bg5dihzfqhmn9z ~]# find / -nouser -o -nogroup -mtime -6
例子3:查找/etc目录下至少有一类用户没有执行权限的文件
[root@iz2zed4h7bg5dihzfqhmn9z ~]# find /etc/ ! -perm /111
在接下来说说文件打包命令tar
在linux中时长也需要进行压缩,解压缩等操作,那在linux中有哪些常用的压缩和解压缩的工具呢?
gzip和gunzip
格式: gzip [options] file
options -k keep, 保留原文件,CentOS 8 新特性
-d 解压缩,相当于gunzip
-c 结果输出至标准输出,保留原文件不改变
-# 指定压缩比,#取值为1-9,值越大压缩比越大
gunzip 解压缩
bzip2和bunzip2
格式:bizp2 [options] file
-k keep, 保留原文件
-d 解压缩
-c 结果输出至标准输出,保留原文件不改变
-# 1-9,压缩比,默认为9
bunzip2 解压缩
xz和unxz
格式:xz [options] file
-k keep, 保留原文件
-d 解压缩
-c 结果输出至标准输出,保留原文件不改变
-# 压缩比,取值1-9,默认为6
unxz 解压缩
zip和unzip
格式:zip [options] file
zip 可以实现打包目录和多个文件成一个文件并压缩,但可能会丢失文件属性信息,如:所有者和组信
息,一般建议使用 tar 代替
-r 打包压缩文件夹
-p 通过管道传给文件
tar
格式:tar [options] ....filename.tar. 目标文件
-c 建立新的打包文件
-C <目录> 切换工作目录,先进入指定目录再执行压缩/解压缩操作,可用于仅压缩特定目录里的内容或解压缩到特定目录
-x 从归档文件中提取文件
-f<备份文件> 指定打包文件
-v 显示指令执行过程
-p 保留原来的文件权限与属性
-- -remove-files 归档/压缩之后删除源文件
-z 相当于gzip压缩工具
-j 相当于bzip2压缩工具
-J 相当于xz压缩工具
例子4:打包/etc/目录下面所有conf结尾的文件,压缩包名称为当天的时间,并拷贝到/usr/local/src目录备份。
[root@iz2zed4h7bg5dihzfqhmn9z src]# find /etc/ -name "*.conf" | xargs tar -cpvf `date +%F`.tar && cp -a `date +%F`.tar /usr/local/src/
[root@iz2zed4h7bg5dihzfqhmn9z ~]# ll /usr/local/src/
total 480
-rw-r--r-- 1 root root 491520 Dec 25 14:55 2020-12-25.tar
上面用到了一个参数xargs,xargs用于产生某个命令的参数,xargs 可以读入 stdin 的数据,并且以空格符或回车符将 stdin 的数据分隔成为参数