标签归档:shell

【shell】使用shc工具加密Shell脚本

shc是一个脚本编译工具, 使用RC4加密算法, 它能够把shell程序转换成二进制可执行文件(支持静态链接和动态链接)。

wget http://www.datsi.fi.upm.es/%7Efrosal/sources/shc-3.8.7.tgz
tar vxf shc-3.8.7.tgz 
cd shc-3.8.7
make test

创建一个测试脚本:

[root@server120 tmp]# cat 1.sh
#!/bin/bash
echo "xxoo"
[root@server120 tmp]# shc-3.8.7/shc -f 1.sh
[root@server120 tmp]# ls 1.sh*
1.sh 1.sh.x 1.sh.x.c

可以看到生成了两个文件1.sh.x 1.sh.x.c,其中1.sh.x为二进制文件,赋予执行权限后,可直接执行。

[root@server120 tmp]# file 1.sh.x
1.sh.x: ELF 64-bit LSB executable, x86-64, version 1 (SYSV), dynamically linked (uses shared libs), for GNU/Linux 2.6.18, stripped

1.sh.x.c 是c源文件。可以删除。

[root@server120 tmp]# mv 1.sh.x xxoo ;./xxoo
xxoo

【shell】获取函数返回值

函数可以让我们将一个复杂功能划分成若干模块,让程序结构更加清晰,代码重复利用率更高。像其他编程语言一样,Shell 也支持函数。Shell 函数必须先定义后使用。
Shell 函数的定义格式如下:
xxxx() {
list of commands
[ return value ]
}
xxxx
调用函数只需要给出函数名,不需要加括号。
1)函数返回值,可以显式增加return语句;如果不加,会将最后一条命令运行结果作为返回值。其中0表示成功,1表示失败,127表示命令未找到。
例如:

#!/bin/sh  
test()
{
    echo "arg1 = $1"  
    xxxx
}
test 1
echo $?         # print return result 
[root@vincent tmp]# ./test.sh 
arg1 = 1
./test.sh: line 5: xxxxx: command not found
127

可以看到返回值为127。
Shell 函数返回值只能是整数,如果返回字符串,往往会得到错误提示:“numeric argument required”。如果一定要让函数返回字符串,那么可以先定义一个变量,用来接收函数的计算结果,脚本在需要的时候访问这个变量来获得函数返回值。
2)echo 返回值,把shell函数作为子程序调用,将其结果写到子程序的标准输出。

testFunc()  
{
    local_result='local value'
    echo $local_result  
}

result=`testFunc`
echo $result

 

【shell】数组的使用

1)定义数组
在Shell中,用括号来表示数组,数组元素用“空格”符号分割开。定义数组的一般形式为:
    array_name=(value1 … valuen)
例如:
array_name=(value0 value1 value2 value3)

还可以单独定义数组的各个分量:
array_name[0]=value0
array_name[1]=value1
array_name[2]=value2
可以不使用连续的下标,而且下标的范围没有限制。

2)读取遍历数组
读取数组元素值的一般格式是:
    ${array_name[index]}
例如:
valuen=${array_name[2]}
使用@ 或 * 可以获取数组中的所有元素,例如:
${array_name[*]}
${array_name[@]}
示例:
array=(1 2 3 )
for i in ${array[*]}
do
        echo $i
done
[root@vincent tmp]# ./test.sh 
1
2
3
3)获取数组的长度
获取数组长度的方法与获取字符串长度的方法相同,例如:
# 取得数组元素的个数
length=${#array_name[@]}
# 或者
length=${#array_name[*]}
 

【shell】EOF的使用

在shell编程中,”EOF“通常与”<<“结合使用,“<<EOF“表示后续的输入作为子命令或子shell的输入,直到遇到”EOF“,再次返回到主调shell,可将其理解为分界符(delimiter)。既然是分界符,那么形式自然不是固定的,这里可以将”EOF“可以进行自定义,但是前后的”EOF“必须成对出现且不能和shell命令冲突。其使用形式如下:
交互式程序(命令)<<EOF
command1
command2

EOF

1)例如登录Mysql并查询数据库脚本test.sh
mysql -u root -phehe123 << EOF
use mysql;
select user,host from user;
EOF
[root@vincent tmp]# ./test.sh 
user    host
root    %
root    127.0.0.1
    localhost
ids    localhost
root    localhost
    vincent
root    vincent

2)特殊用法:
: << COMMENTBLOCK
   shell脚本代码段
COMMENTBLOCK
用来注释整段脚本代码。 : 是shell中的空语句。
echo start
:<<COMMENTBLOCK
echo
echo "this is a test"
echo
COMMENTBLOCK
echo end
这段脚本执行时,中间部分不会被执行:
[root@newserver shell]# sh eof.sh
start
end

3)cat和eof结合使用具有追加功能
cat >> file.txt << EOF
123
456
EOF
[root@vincent tmp]# cat file.txt 
123
456

【shell】运算符

先说下if语句的简化,下面示例中会用到。

shell中的运算符有test、[]、[[]],新手很容易就搞混了,这其中是存在一些区别的。
test
和[]用法相同。示例如下:
[root@vincent ~]# test 10 -lt 20 -o 200 -gt 100 && echo 1
1
[]和[[]]:
使用[]和[[]]的时候每一项两边都要有空格。
[[]]比[]要强大,比如说:

1)[[]]支持字符串的模糊匹配
[[ $a == z* ]]    # 如果$a以"z"开头(模式匹配)那么将为true
[[ $a == "z*" ]] # 如果$a等于z*(字符匹配),那么结果为true
2)字符串的比较可以使用[],但是比较的变量需要用双引号引起来,或者使用[[]],不需要使用双引号引起来。
3)[]中如果整形的比较如果使用> <需要转义,[[]]中不需要转义。
[root@server120 tmp]# [ 1 > 0 ] && echo 1
1
[root@server120 tmp]# [ 1 > 2 ] && echo 1
1
[root@server120 tmp]# [ 1 \> 2 ] && echo 1
[root@server120 tmp]# [ 2 \> 1 ] && echo 1
1
4)[]中的逻辑运算可以使用-o -a,[[]]中可以使用&& ||
[root@server120 tmp]# [[ 1 = 1 && 2 > 1 ]] && echo 1
1
[root@server120 tmp]# [[ 1 = 1 && 1 > 1 ]] && echo 1
5)[[]]中可以使用数学运算
[root@server120 tmp]# [[ 1+1 > 2 ]] && echo 1
[root@server120 tmp]# [[ 1+1 > 1 ]] && echo 1
1

算数运算符
使用let
[root@Hehe tmp]# let result=1+2
[root@Hehe tmp]# echo $result
3
使用expr
a=10
b=20
val=`expr $a + $b`
echo “a + b : $val”

整数比较
关系运算符只支持数字,不支持字符串,除非字符串的值是数字。

-eq    检测两个数是否相等,相等返回 true。    [ $a -eq $b ] 返回 true。
-ne    检测两个数是否相等,不相等返回 true。    [ $a -ne $b ] 返回 true。
-gt    检测左边的数是否大于右边的,如果是,则返回 true。    [ $a -gt $b ] 返回 false。
-lt    检测左边的数是否小于右边的,如果是,则返回 true。    [ $a -lt $b ] 返回 true。
-ge    检测左边的数是否大等于右边的,如果是,则返回 true。    [ $a -ge $b ] 返回 false。
-le    检测左边的数是否小于等于右边的,如果是,则返回 true。    [ $a -le $b ] 返回 true。
<       小于(需要双括号),如:(("$a" < "$b"))
<=       小于等于(需要双括号),如:(("$a" <= "$b"))
>       大于(需要双括号),如:(("$a" > "$b"))
>=       大于等于(需要双括号),如:(("$a" >= "$b"))
if ( 2 > 1 );then
 echo 1
fi
执行结果为:
[root@localhost tmp]# ./1.sh
./1.sh: line 1: 2: command not found
if (( 2 > 1 ));then
 echo 1
fi
执行结果为:
[root@localhost tmp]# ./1.sh
1

if语句中常用的简化就是&&和||
[ 1 -eq 1 ] && echo “equal”
[ 1 -eq 2 ] || echo “not equal”
返回结果如下:
[root@server120 tmp]# ./test.sh
equal
not equal

布尔运算符
10 != 20 : a is not equal to b
10 -lt 100 -a 20 -gt 15 : returns true
10 -lt 100 -o 20 -gt 100 : returns true
10 -lt 5 -o 20 -gt 100 : returns false

布尔运算符列表

运算符    说明    举例
!    非运算,表达式为 true 则返回 false,否则返回 true。    [ ! false ] 返回 true。
-o    或运算,有一个表达式为 true 则返回 true。    [ $a -lt 20 -o $b -gt 100 ] 返回 true。
-a    与运算,两个表达式都为 true 才返回 true。    [ $a -lt 20 -a $b -gt 100 ] 返回 false

字符串运算符

运算符    说明    举例
=    检测两个字符串是否相等,相等返回 true。    [ $a = $b ] 返回 false。
==       同=
!=    检测两个字符串是否相等,不相等返回 true。    [ $a != $b ] 返回 true。
-z    检测字符串长度是否为0,为0返回 true。    [ -z $a ] 返回 false。
-n    检测字符串长度是否为0,不为0返回 true。    [ -z $a ] 返回 true。
str    检测字符串是否为空,不为空返回 true。    [ $a ] 返回 true。

测试脚本:
xx=”xx”
yy=””
[ “$xx” ] && echo “xx:True”
[ “$yy” ] || echo “yy:False”
返回结果如下:
xx:True
yy:False

文件测试运算符

-b file    检测文件是否是块设备文件,如果是,则返回 true。    [ -b $file ] 返回 false。
-c file    检测文件是否是字符设备文件,如果是,则返回 true。    [ -b $file ] 返回 false。
-d file    检测文件是否是目录,如果是,则返回 true。    [ -d $file ] 返回 false。
-f file    检测文件是否是普通文件(既不是目录,也不是设备文件),如果是,则返回 true。    [ -f $file ] 返回 true。
-g file    检测文件是否设置了 SGID 位,如果是,则返回 true。    [ -g $file ] 返回 false。
-k file    检测文件是否设置了粘着位(Sticky Bit),如果是,则返回 true。    [ -k $file ] 返回 false。
-p file    检测文件是否是具名管道,如果是,则返回 true。    [ -p $file ] 返回 false。
-u file    检测文件是否设置了 SUID 位,如果是,则返回 true。    [ -u $file ] 返回 false。
-r file    检测文件是否可读,如果是,则返回 true。    [ -r $file ] 返回 true。
-w file    检测文件是否可写,如果是,则返回 true。    [ -w $file ] 返回 true。
-x file    检测文件是否可执行,如果是,则返回 true。    [ -x $file ] 返回 true。
-s file    检测文件是否为空(文件大小是否大于0),不为空返回 true。    [ -s $file ] 返回 true。

创建一个空文件file

[root@server120 tmp]# touch file
[root@server120 tmp]# [ -s file ] && echo "NOT NULL" || echo "NULL"
NULL
[root@server120 tmp]# echo 1 > file
[root@server120 tmp]# [ -s file ] && echo "NOT NULL" || echo "NULL"
NOT NULL

-e file    检测文件(包括目录)是否存在,如果是,则返回 true。    [ -e $file ] 返回 true。

[root@vincent tmp]# [ -e file.txt ] && echo 1
1
[root@vincent tmp]# rm -f file.txt
[root@vincent tmp]# [ -e file.txt ] && echo 1
[root@vincent tmp]#

【shell】while循环中的变量无法保存问题

问题SHELL举例:

#!/bin/sh
Counter=0
ls /opt/src/ | while read LINE
do
let "Counter=Counter+1"
done
echo $Counter

该SHELL用于计算/opt/src/中有多少文件,但实际运行中无论/opt/src/下有多少文件,最后输出都是0。
经查,造成这个问题的原因是while循环位于管道中,这意味着在运行过程中,while循环实际是位于一个新的SHELL中的,while循环中的Counter变量和文件开头定义的Counter变量是两个不同的变量,所以while循环中所改变的值在while循环结束后无法保存下来。

解决办法:

#!/bin/sh
Counter=0
ls /opt/src/ > /tmp/t1.txt
while read LINE
do
let "Counter=Counter+1"
done < t1.txt
rm /tmp/t1.txt -f
echo $Counter