https://blog.csdn.net/doiido/article/details/43966819
1、基本语法:
if [ command ]; then
符合该条件执行的语句
fi
2、扩展语法:
if [ command ];then
符合该条件执行的语句
elif [ command ];then
符合该条件执行的语句
else
符合该条件执行的语句
fi
3、语法说明:
bash shell会按顺序执行if语句,如果command执行后且它的返回状态是0,则会执行符合该条件执行的语句,否则后面的命令不执行,跳到下一条命令。
当有多个嵌套时,只有第一个返回0退出状态的命令会导致符合该条件执行的语句部分被执行,如果所有的语句的执行状态都不为0,则执行else中语句。
返回状态:最后一个命令的退出状态,或者当没有条件是真的话为0。
注意:
1、[ ]表示条件测试。注意这里的空格很重要。要注意在'['后面和']'前面都必须要有空格
2、在shell中,then和fi是分开的语句。如果要在同一行里面输入,则需要用分号将他们隔开。
3、注意if判断中对于变量的处理,需要加引号,以免一些不必要的错误。没有加双引号会在一些含空格等的字符串变量判断的时候产生错误。比如[ -n "$var" ]如果var为空会出错
4、判断是不支持浮点值的
5、如果只单独使用>或者<号,系统会认为是输出或者输入重定向,虽然结果显示正确,但是其实是错误的,因此要对这些符号进行转意
6、在默认中,运行if语句中的命令所产生的错误信息仍然出现在脚本的输出结果中
7、使用-z或者-n来检查长度的时候,没有定义的变量也为0
8、空变量和没有初始化的变量可能会对shell脚本测试产生灾难性的影响,因此在不确定变量的内容的时候,在测试号前使用-n或者-z测试一下
9、? 变量包含了之前执行命令的退出状态(最近完成的前台进程)(可以用于检测退出状态)
常用参数:
文件/目录判断:
[ -a FILE ] 如果 FILE 存在则为真。
[ -b FILE ] 如果 FILE 存在且是一个块文件则返回为真。
[ -c FILE ] 如果 FILE 存在且是一个字符文件则返回为真。
[ -d FILE ] 如果 FILE 存在且是一个目录则返回为真。
[ -e FILE ] 如果 指定的文件或目录存在时返回为真。
[ -f FILE ] 如果 FILE 存在且是一个普通文件则返回为真。
[ -g FILE ] 如果 FILE 存在且设置了SGID则返回为真。
[ -h FILE ] 如果 FILE 存在且是一个符号符号链接文件则返回为真。(该选项在一些老系统上无效)
[ -k FILE ] 如果 FILE 存在且已经设置了冒险位则返回为真。
[ -p FILE ] 如果 FILE 存并且是命令管道时返回为真。
[ -r FILE ] 如果 FILE 存在且是可读的则返回为真。
[ -s FILE ] 如果 FILE 存在且大小非0时为真则返回为真。
[ -u FILE ] 如果 FILE 存在且设置了SUID位时返回为真。
[ -w FILE ] 如果 FILE 存在且是可写的则返回为真。(一个目录为了它的内容被访问必然是可执行的)
[ -x FILE ] 如果 FILE 存在且是可执行的则返回为真。
[ -O FILE ] 如果 FILE 存在且属有效用户ID则返回为真。
[ -G FILE ] 如果 FILE 存在且默认组为当前组则返回为真。(只检查系统默认组)
[ -L FILE ] 如果 FILE 存在且是一个符号连接则返回为真。
[ -N FILE ] 如果 FILE 存在 and has been mod如果ied since it was last read则返回为真。
[ -S FILE ] 如果 FILE 存在且是一个套接字则返回为真。
[ FILE1 -nt FILE2 ] 如果 FILE1 比 FILE2 新, 或者 FILE1 存在但是 FILE2 不存在则返回为真。
[ FILE1 -ot FILE2 ] 如果 FILE1 比 FILE2 老, 或者 FILE2 存在但是 FILE1 不存在则返回为真。
[ FILE1 -ef FILE2 ] 如果 FILE1 和 FILE2 指向相同的设备和节点号则返回为真。
字符串判断
[ -z STRING ] 如果STRING的长度为零则返回为真,即空是真
[ -n STRING ] 如果STRING的长度非零则返回为真,即非空是真
[ STRING1 ] 如果字符串不为空则返回为真,与-n类似
[ STRING1 == STRING2 ] 如果两个字符串相同则返回为真
[ STRING1 != STRING2 ] 如果字符串不相同则返回为真
[ STRING1 < STRING2 ] 如果 “STRING1”字典排序在“STRING2”前面则返回为真。
[ STRING1 > STRING2 ] 如果 “STRING1”字典排序在“STRING2”后面则返回为真。
数值判断
[ INT1 -eq INT2 ] INT1和INT2两数相等返回为真 ,=
[ INT1 -ne INT2 ] INT1和INT2两数不等返回为真 ,<>
[ INT1 -gt INT2 ] INT1大于INT2返回为真 ,>
[ INT1 -ge INT2 ] INT1大于等于INT2返回为真,>=
[ INT1 -lt INT2 ] INT1小于INT2返回为真 ,<
[ INT1 -le INT2 ] INT1小于等于INT2返回为真,<=
逻辑判断
[ ! EXPR ] 逻辑非,如果 EXPR 是false则返回为真。
[ EXPR1 -a EXPR2 ] 逻辑与,如果 EXPR1 and EXPR2 全真则返回为真。
[ EXPR1 -o EXPR2 ] 逻辑或,如果 EXPR1 或者 EXPR2 为真则返回为真。
[ ] || [ ] 用OR来合并两个条件
[ ] && [ ] 用AND来合并两个条件
其他判断
[ -t FD ] 如果文件描述符 FD (默认值为1)打开且指向一个终端则返回为真
[ -o optionname ] 如果shell选项optionname开启则返回为真
IF高级特性:
双圆括号(( )):表示数学表达式
在判断命令中只允许在比较中进行简单的算术操作,而双圆括号提供更多的数学符号,而且在双圆括号里面的'>','<'号不需要转意。
双方括号[[ ]]:表示高级字符串处理函数
双方括号中判断命令使用标准的字符串比较,还可以使用匹配模式,从而定义与字符串相匹配的正则表达式。
双括号的作用:
在shell中,[ $a != 1 || $b = 2 ]是不允许出,要用[ $a != 1 ] || [ $b = 2 ],而双括号就可以解决这个问题的,[[ $a != 1 || b=2]]。又比如这个["b = 2 ]]。又比如这个[ "b=2]]。又比如这个["a" -lt "b"],也可以改成双括号的形式(("b" ],也可以改成双括号的形式(("b"],也可以改成双括号的形式(("a" < "$b"))
实例
1:判断目录$doiido是否存在,若不存在,则新建一个
if [ ! -d "$doiido"]; then
mkdir "$doiido"
fi
2:判断普通文件$doiido是否存,若不存在,则新建一个
if [ ! -f "$doiido" ]; then
touch "$doiido"
fi
3:判断$doiido是否存在并且是否具有可执行权限
if [ ! -x "$doiido"]; then
mkdir "$doiido"
chmod +x "$doiido"
fi
4:是判断变量$doiido是否有值
if [ ! -n "$doiido" ]; then
echo "$doiido is empty"
exit 0
fi
5:两个变量判断是否相等
if [ "$var1" = "$var2" ]; then
echo '$var1 eq $var2'
else
echo '$var1 not eq $var2'
fi
6:测试退出状态:
if [ $? -eq 0 ];then
echo 'That is ok'
fi
7:数值的比较:
if [ "$num" -gt "150" ]
echo "$num is biger than 150"
fi
8:a>b且a<c
(( a > b )) && (( a < c ))
[[ $a > $b ]] && [[ $a < $c ]]
[ $a -gt $b -a $a -lt $c ]
9:a>b或a<c
(( a > b )) || (( a < c ))
[[ $a > $b ]] || [[ $a < $c ]]
[ $a -gt $b -o $a -lt $c ]
10:检测执行脚本的用户
if [ "$(whoami)" != 'root' ]; then
echo "You have no permission to run $0 as non-root user."
exit 1;
fi```
上面的语句也可以使用以下的精简语句
```shell
[ "$(whoami)" != 'root' ] && ( echo "You have no permission to run $0 as non-root user."; exit 1 )
11:正则表达式
doiido="hero"
if [[ "$doiido" == h* ]];then
echo "hello,hero"
fi
其他例子
1、查看当前操作系统类型
#!/bin/sh
SYSTEM=`uname -s`
if [ $SYSTEM = "Linux" ] ; then
echo "Linux"
elif [ $SYSTEM = "FreeBSD" ] ; then
echo "FreeBSD"
elif [ $SYSTEM = "Solaris" ] ; then
echo "Solaris"
else
echo "What?"
fi
2、if利用read传参判断
#!/bin/bash
read -p "please input a score:" score
echo -e "your score [$score] is judging by sys now"
if [ "$score" -ge "0" ]&&[ "$score" -lt "60" ];then
echo "sorry,you are lost!"
elif [ "$score" -ge "60" ]&&[ "$score" -lt "85" ];then
echo "just soso!"
elif [ "$score" -le "100" ]&&[ "$score" -ge "85" ];then
echo "good job!"
else
echo "input score is wrong , the range is [0-100]!"
fi
3、判断文件是否存在
#!/bin/sh
today=`date -d yesterday +%y%m%d`
file="apache_$today.tar.gz"
cd /home/chenshuo/shell
if [ -f "$file" ];then
echo "OK"
else
echo "error $file" >error.log
mail -s "fail backup from test" loveyasxn924@126.com <error.log
fi
4、这个脚本在每个星期天由cron来执行。如果星期的数是偶数,他就提醒你把垃圾箱清理:
#!/bin/bash
WEEKOFFSET=$[ $(date +"%V") % 2 ]
if [ $WEEKOFFSET -eq "0" ]; then
echo "Sunday evening, put out the garbage cans." | mail -s "Garbage cans out" your@your_domain.org
fi
5、挂载硬盘脚本(windows下的ntfs格式硬盘)
#! /bin/sh
dir_d=/media/disk_d
dir_e=/media/disk_e
dir_f=/media/disk_f
a=`ls $dir_d | wc -l`
b=`ls $dir_e | wc -l`
c=`ls $dir_f | wc -l`
echo "checking disk_d..."
if [ $a -eq 0 ]; then
echo "disk_d is not exsit,now creating..."
sudo mount -t ntfs /dev/disk/by-label/software /media/disk_d
else
echo "disk_d exits"
fi
echo "checking disk_e..."
if [ $b -eq 0 ]; then
echo "disk_e is not exsit,now creating..."
sudo mount -t ntfs /dev/disk/by-label/elitor /media/disk_e
else
echo "disk_e exits"
fi
echo "checking disk_f..."
if [ $c -eq 0 ]; then
echo "disk_f is not exsit,now creating..."
sudo mount -t ntfs /dev/disk/by-label/work /media/disk_f
else
echo "disk_f exits"
fi
Read More ~
什么是UAC和UAS
https://community.cisco.com/t5/%E5%8D%8F%E4%BD%9C%E6%96%87%E6%A1%A3/%E4%BB%80%E4%B9%88%E6%98%AFuac%E5%92%8Cuas/ta-p/4336761
SIP协议采用Client/Server模型。每一个请求(Request) 触发服务器的一个操作;每个操作被称为方法(Method);每个请求除了指明具体方法外,还携带了一系列的 头域(Header field),这些头域携带了各种业务信息;除了头域外,消息中还可以携带任意类型的消息体(Body),呼叫中最常使用的SDP信息就是通过消息体在Client和Server之间传递的。Server从接收到请求到处理完毕,要回复多个临时响应(Response),和一个终结响应(Response),终结响应有且只有一个。
请求和他的所有响应构成一个事务(Transaction),一个完整的呼叫过程包含多个事务,比如呼叫建立和呼叫释放就是
两个相互独立的事务。
用户代理(User Agent) 是发起或者接收呼叫的逻辑实体。
用户代理客户端-UAC(User Agent Client),用于发起请求;
用户代理服务器-UAS(User Agent Server),用于接收请求。
UAC/UAS的划分是针对一个事务的。在一个呼叫中的多个事务里,UAC和UAS的角色是可以互换的。例如在A和B的呼叫中,A向B发起呼叫,在呼叫建立的事务中,A是UAC,B是UAS;呼叫结束时,B先挂机,在呼叫释放的事务中,B是UAC,A是UAS。换句话说,每个一般的UA都是UAS和UAC的结合体。
UA的实际物理形态有:IP Phone,SoftPhone,GateWay......
Proxy Server作为一个网络逻辑实体代理客户端转发请求或者响应;同Proxy Server类似的还有一种设备是B2BUA,
顾名思义,就是背背的两个UA组成的一个逻辑实体,它作为UAS终结一个事务,同时作为UAC发起另外一个事务。Proxy Server同B2BUA相比,Proxy Server是一个事务传递过程中的中间节点,而B2BUA将一个事务转变成另一个事务。
在SIP组网中还包括Location Server、Registrar、Redirect Server,分别负责维护地址映射表,注册管理,呼叫重定向。他们和Proxy Server 可以在同一台设备上也可以运行于不同的设备上。SIP Server是Proxy Server、Location Server、Registrar、Redirect Server的总称。
SIP Server采用B2BUA模型。接受请求端为UAS端,代理转发或主动发起请求端为UAC端,整个SIPServr为UAC/UAS的组合体。通过UAC/UAS之间消息交互完成会话的建立、改变、结束的阶段。SIP Server协助网关交换媒体信息,
但不参与会话建立后媒体流传输。
Read More ~
Linux下删除乱码文件及文件夹方法
转载自:https://blog.csdn.net/momoringmok/article/details/105013079
1.用ls -i 命令找到文件或目录的 inode
2.删除文件:
find -inum 1182336 -delete
find -inum 1182338 -delete
find -inum 1182337 -delete
find -inum 1182340 -delete
3.删除目录(会提示 No such file or directory):
find -inum 345678 -exec rm -rf {} ;
Read More ~
mysql 切割(以表分区)
查询建表sql
SHOW CREATE TABLE table_name;
其中原建表语句中的LESS THAN (738917) ,其中738917可以通过
SELECT FROM_DAYS(738917); 查询具体的时间。
添加分区
ALTER TABLE new_table PARTITION BY RANGE (TO_DAYS(date_column)) (
PARTITION p1 VALUES LESS THAN (TO_DAYS('2021-01-01')),
PARTITION p2 VALUES LESS THAN (TO_DAYS('2022-01-01'))
);
导出数据到新表
INSERT INTO new_table PARTITION (p2019)
SELECT * FROM source_table PARTITION (p2019);
统计分区数据数
SELECT COUNT(*) FROM new_table PARTITION (p2019);
SELECT COUNT(*) FROM source_table PARTITION (p2019);
对比备份前后分区数据
SELECT (SELECT COUNT(*) FROM table1 PARTITION (p2022)) AS source_table_row_count, (SELECT COUNT(*) FROM table2 PARTITION (p2022)) AS new_table_row_count;
删除源表数据
ALTER TABLE source_table DROP PARTITION p201901;
Read More ~
号码归属地信息从dat文件导出成列表
广电开始有电信业务了,号码归属地表又是一顿更新,如果跟不上步伐可能会引发一些事故。目前暂时本人暂时无法做到自己去爬数据,所以就只能借助网上大佬的数据库来使用了。
找了一圈,找到了lovedboy大佬的github库。以及相对应的脚本。原本的使用方法应该是数据文件+脚本形成可以即时调用的方法,但无奈自家是写进mysql的,只能无奈将其导出。
修改了大佬的原调用脚本,测试失败。😅
索性摆烂,将大佬脚本和需求丢进chatgpt跑一圈,出来以下脚本,试运行,丝滑。😲
放出来给大家看看。
# -*- coding: utf-8 -*-
import os
import struct
import sys
__author__ = 'lovedboy'
if sys.version_info > (3, 0):
def get_record_content(buf, start_offset):
end_offset = buf.find(b'\x00', start_offset)
return buf[start_offset:end_offset].decode()
else:
def get_record_content(buf, start_offset):
end_offset = buf.find('\x00', start_offset)
return buf[start_offset:end_offset]
class Phone(object):
def __init__(self, dat_file=None):
if dat_file is None:
dat_file = os.path.join(os.path.dirname(__file__), "phone.dat")
with open(dat_file, 'rb') as f:
self.buf = f.read()
self.head_fmt = "<4si"
self.phone_fmt = "<iiB"
self.head_fmt_length = struct.calcsize(self.head_fmt)
self.phone_fmt_length = struct.calcsize(self.phone_fmt)
self.version, self.first_phone_record_offset = struct.unpack(
self.head_fmt, self.buf[:self.head_fmt_length])
self.phone_record_count = (len(self.buf) - self.first_phone_record_offset) // self.phone_fmt_length
def get_phone_dat_msg(self):
print("版本号:{}".format(self.version))
print("总记录条数:{}".format(self.phone_record_count))
@staticmethod
def get_phone_no_type(no):
if no == 4:
return "电信虚拟运营商"
if no == 5:
return "联通虚拟运营商"
if no == 6:
return "移动虚拟运营商"
if no == 3:
return "电信"
if no == 2:
return "联通"
if no == 1:
return "移动"
@staticmethod
def _format_phone_content(phone_num, record_content, phone_type):
province, city, zip_code, area_code = record_content.split('|')
return {
"phone": phone_num,
"province": province,
"city": city,
"zip_code": zip_code,
"area_code": area_code,
"phone_type": Phone.get_phone_no_type(phone_type)
}
def _parse_record(self, record):
phone_num, record_offset, phone_type = struct.unpack(self.phone_fmt, record)
record_content = get_record_content(self.buf, record_offset)
return Phone._format_phone_content(phone_num, record_content, phone_type)
def all(self):
records = []
buflen = len(self.buf)
for i in range(self.first_phone_record_offset, buflen, self.phone_fmt_length):
record = self.buf[i: i + self.phone_fmt_length]
records.append(self._parse_record(record))
return records
def test(self):
self.get_phone_dat_msg()
records = self.all()
for record in records:
print(self.human_phone_info(record))
@staticmethod
def human_phone_info(phone_info):
if not phone_info:
return ''
return "{}|{}|{}|{}|{}|{}".format(phone_info['phone'],
phone_info['province'],
phone_info['city'],
phone_info['zip_code'],
phone_info['area_code'],
phone_info['phone_type'])
if __name__ == "__main__":
phone = Phone()
phone.test()
Read More ~