(1)它必须以如下行开始(必须放在文件的第一行):
#!/bin/sh
符号#!用来告诉系统执行该脚本的程序,本例使用/bin/sh。编辑结束并保存后,如果要执行该脚本,必须先使其可执行: chmod +x filename。此后在该脚本所在目录下,输入 ./filename 即可执行该脚本。
#!/bin/sh & #!/bin/bash的差别,在ubuntu中,可以知道
两者其实是同一个东西,使用起来没有差别。
如果是在嵌入式系统中比如安卓里面,通过文件系统可以知道,运行在它里面的脚本只能加#!/system/bin/sh,且链接内容是这样的
如果安卓中运行脚本,提示No such file or directory,其实是脚本格式的原因,只需要将windows下写的shell脚本转成unix格式即可解决。
(2)变量赋值和引用。Shell编程中,使用变量无需事先声明,需要给变量赋值时,可以这么写: 变量名=值 。要取用一个变量的值,只需在变量名前面加一个$ ( 注意: 给变量赋值的时候,不能在”=”两边留空格 )
a,对字符串变量赋值并输出:
a=”hello world”
# 打印字符串变量a的值:
echo “A is:” $a
执行结果:A is: hello world
b,内部执行shell命令,用单引号,如:
now_date_1=`date`
echo “now_date_1 is ” $now_date_1
结果是:now_date_1 is 2014年 12月 29日 星期一 23:51:46 CST
c,字符串连接,多个字符串并在一起即可,如:
FILES=” boota64.bin”
now_date_1=”wahaha”
echo “now_date_1 is ” $now_date_1$FILES
输出是:now_date_1 is wahaha boota64.bin
(3)if 语句。”if”表达式如果条件为真,则执行then后的部分:
if ….; then
….
elif ….; then
….
else
….
fi
通常用” [ ] “来表示条件测试,注意这里的空格很重要,要确保方括号前后的空格。例如:
[ -f “somefile” ] :判断是否是一个文件
[ -x “/bin/ls” ] :判断/bin/ls是否存在并有可执行权限
[ -n “$var” ] :判断$var变量是否有值,长度非零则为真
[ -e “somefile” ] :判断文件是否存在
[ -d “$cust_dir” ] :判断$cust_dir所代指的目录是否存在
[ “$a” = “$b” ] :判断$a和$b是否相等
[ -z “$DIRNAME” ]:判断是否有值,长度为零则为真
判断条件前加上!,则是判断非条件,比如if [ ! -d mediatek/config/$our_proj ] ; 如果目录不存在。
(4)循环
while …
do
….
done
只要测试表达式条件为真,则while循环将一直运行。关键字”break”用来跳出循环,而关键字”continue”则可以跳过一个循环的余下部分,直接跳到下一次循环中。
或者用for循环,注意是双括号
#!/bin/bash
for ((i =1 ;i < 227;i ++))
do
cp logo.png Nissan_$i.png
done
另:特殊的循环,用来编译字符串拷贝文件
FILES="lk.bin logo.bin boot.img secro.img userdata.img cache.img "
for FILE in $FILES
do
cp out/target/product/$PLATFORM/$FILE $DIR
done
(5)函数功能
在sh中可以定义某段类似于函数的集成语句段,在名字前使用function申明。方法如:
function real6410_prebuild()
{
mkdir -p out/target/product/$PRODUCT/obj/lib
mkdir -p out/target/product/$PRODUCT/system/lib
cp vendor/realarm/real6410/*.so out/target/product/$PRODUCT/obj/lib
cp vendor/realarm/real6410/*.so out/target/product/$PRODUCT/system/lib
}
之后,在其他地方直接使用real6410_prebuild就行了。
(6)sync
#!/bin/bash
./prebuild.sh $1 $2 $3 $4 $5 $6 && ./mk $1_$2 n && ./Target $1 $2 $3 $4 $5 $6
(8)参数$所带的特殊意义
$# 传给脚本的参数个数
$0 脚本本身的文件名字
$1传递给该shell脚本的第一个参数
$2传递给该shell脚本的第二个参数
$?最后运行的命令的结束代码(返回值)
$@所有参数列表
$*和 $@ 都表示传递给函数或脚本的所有参数,不被双引号(” “)包含时,都以”$1” “$2” … “$n” 的形式输出所有参数。但是当它们被双引号(” “)包含时,”$*” 会将所有的参数作为一个整体,以”$1 $2 … $n”的形式输出所有参数;”$@” 会将各个参数分开,以”$1″ “$2” … “$n” 的形式输出所有参数。
可用如下shell脚本验证
#!/bin/sh
echo "number:$#"
echo "scname:$0"
echo "first:$1"
echo "second:$2"
echo "argume:$@"
./variable aa bb,执行结果是:
number:2
scname:./variable
first:aa
second:bb
argume:aa bb
(9)比较操作
整数比较
-eq 等于,如:if [ “$a” -eq “$b” ]
-ne 不等于,如:if [ “$a” -ne “$b” ]
-gt 大于,如:if [ “$a” -gt “$b” ]
-ge 大于等于,如:if [ “$a” -ge “$b” ]
-lt 小于,如:if [ “$a” -lt “$b” ]
-le 小于等于,如:if [ “$a” -le “$b” ]
< 小于(需要双括号),如:((“$a” < “$b”))
<= 小于等于(需要双括号),如:((“$a” <= “$b”))
> 大于(需要双括号),如:((“$a” > “$b”))
>= 大于等于(需要双括号),如:((“$a” >= “$b”))
字符串比较
= 等于,如:if [ “$a” = “$b” ]
== 等于,如:if [ “$a” == “$b” ],与=等价
!=不等于
如:
if [ “$2” = “lca” ] ; then
cust_vendor=lenovo72_we_lca
else
cust_vendor=lenovo72_we_jb3
fi
(10)获取shell中的当前日期和时间
#!/bin/sh
now_date_1=`date +%Y-%m-%d-%H-%M-%S`
echo "now_date_1 is " $now_date_1
结果是now_date_1 is 2014-12-29-23-46-08
如果是获取时间差:
start=$(date +%s)
xxxxxxx
end=$(date +%s)
time=$(( $end - $start ))
echo "total build cost time is $time s"
(11)执行脚本出现bad interpreter:No such file or directory
是文件格式的问题。这个文件是在Windows下编写的。换行的方式与Unix不一样,但是在VI下面如果不Set一下又完全看不出来。判断方法是:
vi filename
然后用命令
:set ff
可以看到dos或unix的字样。如果的确是dos格式的,那么你可以用set ff=unix把它强制为unix格式的,然后存盘退出运行。或者,用UE的文件-》转换-》DOS到UNIX也可以转换
(12)sed命令
sed是一个很好的文件处理工具,本身是一个管道命令,主要是以行为单位进行处理,可以将数据行进行替换、删除、新增、选取等特定工作,sed命令行格式为:
sed [-nefri] ‘command’ 输入文本名。
常用选项:
-n∶使用安静模式。在一般的用法中,所有来自 STDIN的资料一般都会被列出到萤幕。但如果加上 -n 参数后,则只有经过sed 特殊处理的那一行(或者动作)才会被列出来。
-e∶直接在指令列模式上进行 sed 的动作编辑;
-f∶直接将 sed 的动作写在一个档案内, -f filename 则可以执行 filename 内的sed 动作;
-r∶sed 的动作支援的是延伸型正规表示法的语法。(预设是基础正规表示法语法)
-i∶直接修改读取的档案内容,而不是由萤幕输出。
常用命令:
a ∶新增, a 的后面可以接字串,而这些字串会在新的一行出现(目前的下一行)~
c ∶取代, c 的后面可以接字串,这些字串可以取代 n1,n2 之间的行!
d ∶删除,因为是删除啊,所以 d 后面通常不接任何咚咚;sed ‘2,$d’ ab #删除ab文件的第二行到最后一行
i ∶插入, i 的后面可以接字串,而这些字串会在新的一行出现(目前的上一行);
p ∶列印,亦即将某个选择的资料印出,通常 p 会与参数 sed -n 一起运作。sed -n ‘2,$p’ ab #显示ab文件第二行到最后一行
s ∶取代,可以直接进行取代的工作哩!通常这个 s 的动作可以搭配正规表示法!例如 1,20s/old/new/g 就是啦!
特殊用法:替换一行中的某部分,格式:sed ‘s/要替换的字符串/新的字符串/g’,如下:
sed -i “s/ = /=/g” yyy
sed -i “s/ =/=/g” yyy
(13)实例,可执行脚本update的内容如下。
#!/bin/bash
if [ "$1" = "" ]; then
echo "Please input resolution,"
echo "Such as: qvga, wqvga, wvga, hvga"
exit
fi
p=$1
./tool/bmp_to_raw ./temp0.raw ./$p/"${p}_uboot".bmp
./tool/bmp_to_raw ./temp1.raw ./$p/"${p}_battery00".bmp
./tool/bmp_to_raw ./temp2.raw ./$p/"${p}_battery02".bmp
./tool/bmp_to_raw ./temp3.raw ./$p/"${p}_battery04".bmp
./tool/bmp_to_raw ./temp4.raw ./$p/"${p}_battery06".bmp
./tool/bmp_to_raw ./temp5.raw ./$p/"${p}_battery08".bmp
./tool/bmp_to_raw ./temp6.raw ./$p/"${p}_low_battery1".bmp
./tool/bmp_to_raw ./temp7.raw ./$p/"${p}_low_battery2".bmp
./tool/bmp_to_raw ./temp8.raw ./$p/"${p}_batteryfull".bmp
./tool/bmp_to_raw ./temp9.raw ./$p/"${p}_charger_ov".bmp
./tool/bmp_to_raw ./boot_logo ./$p/"${p}_kernel".bmp
./tool/zpipe -l 9 ./"${p}.raw" temp0.raw temp1.raw temp2.raw temp3.raw temp4.raw temp5.raw temp6.raw temp7.raw temp8.raw temp9.raw
rm -rf ./temp0.raw ./temp1.raw ./temp2.raw ./temp3.raw ./temp4.raw ./temp5.raw ./temp6.raw ./temp7.raw ./temp8.raw ./temp9.raw ./bootlogo.raw
echo "conversion finished"
说明:执行时用./update xxx就可以执行。在””中使用参数变量名字,使用${p},如果使用$p不识别
(14)unset USERNAME,unset就是把USERNAME变量弄成空的
(15)一个sh脚本中,运行另一个sh脚本,在这个sh脚本中用语句sh xxxx.sh即可,或者./xxx.sh
(16)read命令,从键盘读取字符输入
echo "Enter you name:"
read name
echo "hello $name,welcome to here"
exit 0
提示回显输入内容,可以用在很多场合。
(17)条件相与或者相或
用&&和||,比如if [ “$1” = “nissan” ] || [ “$1” = “beryllium” ] ; then
(18)win平台上的bat中跑shell,su跑
adb wait-for-device
adb devices
echo "push tools/adb-ec /data/local/tmp/adb"
adb push tools/adb-ec /data/local/tmp/adb
echo "push tools/usb_modeswitch /data/local/tmp/usb_modeswitch"
adb push tools/usb_modeswitch /data/local/tmp/usb_modeswitch
echo "push easyconn_4.2.sh /data/local/tmp/easyconn_4.2.sh"
adb push tools/easyconn_4.2.sh /data/local/tmp/easyconn_4.2.sh
echo "push tools/busybox /data/local/tmp"
adb push tools/busybox /data/local/tmp/busybox
echo "chmod 777 /data/local/tmp/busybox"
adb shell "chmod 777 /data/local/tmp/busybox"
echo "chmod 777 /data/local/tmp/easyconn_4.2.sh"
adb shell "chmod 777 /data/local/tmp/easyconn_4.2.sh"
echo "su < /data/local/tmp/easyconn_4.2.sh"
adb shell "su < /data/local/tmp/easyconn_4.2.sh"
pause
sh的内容
#!/bin/bash
mount -o remount rw /system
/data/local/tmp/busybox cp /data/local/tmp/adb-ec /system/xbin/adb
/data/local/tmp/busybox cp /data/local/tmp/usb_modeswitch /system/xbin/usb_modeswitch
chown root:root /system/xbin/adb
chmod 6755 /system/xbin/adb
chown root:root /system/xbin/usb_modeswitch
chmod 6755 /system/xbin/usb_modeswitch
另一种跑法:
adb shell < startserver.txt
startserver.txt的内容
./system/bin/adb-ec-server.sh &
exit
(19)case语句,实例
# vi test.sh
:
echo “input : ”
read num
echo “the input data is $num”
case $num in //case行尾必须为单词“in”
1) //每一个值路径必须以右括号“)”结束
echo “January”;; //语句双分号结束
2)
echo “Feburary”;;
5)
echo “may” 每个case可以有多条命令
echo “sdfd”
echo “sdf”;; 但最后一条命令一定是双分号结束
*)
echo “not correct input”;; //*)是default的意思,当使用前面的各种模式均无法匹配该变量时,将执行“*)”后 的命令序列。
esac
(20)单独的#,比如:
local FORCE=”” # -B
$SERVER mmm $FORCE kernel-3.18:kernel -j$CPUS
#表示黏贴符号
(21)shift命令。这个脚本你运行一下就知道了
看见作用了吧。每次运行shift(不带参数的),销毁一个参数,后面的参数前移。输出如下
YY-QS:~# ./test.sh 1 2 3 4 5 6 7 8 9 10
1 2 3 4 5 6 7 8 9 10
2 3 4 5 6 7 8 9 10
3 4 5 6 7 8 9 10
4 5 6 7 8 9 10
5 6 7 8 9 10
6 7 8 9 10
7 8 9 10
8 9 10
9 10
10
Shift命令一次移动参数的个数由其所带的参数指定。例如当shell程序处理完前九个命令行参数后,可以使用shift 9命令把$10移到$1
(22)if[X$1 = X];then 什么意思?X有啥用
如果上述等式成立,那就是变量$1 没有内容,是空变量,主要是用来测试命令列上有没有参数。
参考原文:http://bbs.chinaunix.net/thread-2231835-1-1.html
参考原文:http://nisoze.blog.163.com/blog/static/12484200820110128509637/
参考原文:http://blog.sina.com.cn/s/blog_464f6dba0100psy9.html
参考原文:
linux shell bash 比较操作 – Linux操作系统:Ubuntu_Centos_Debian – 红黑联盟