在我们的日常开发过程中,做一些系统初始化操作那是司空见惯的,那怎么实现这样的功能则需要看实际的应用场景。最常用的操作是就是在系统模块起来时执行初始化,创建实例等等,可以在 WMS,AMS这些地方加入,或是 SystemReady 的时候加上,至于具体加在哪里,当然得看你的实际应用场景,保持模块的干净整洁和解耦也是一大任务和要素。
但这次我们却收到了一个特殊的需求,需要第一次开机时做一些数据的拷贝。单纯的拷贝而不带任何业务逻辑。于是我们一开始便选择将拷贝的操作放在一个预制数据的模块当中,但后来的测试中发现,该模块和实际始终的模块其实是两个进程。有可能会导致数据库未拷贝完全就被拿去使用并且刷新,导致数据不正常,从而引发开机后无法运行的问题。而实际使用的模块又紧紧只是 system 权限,没有 mount 和 mkdir 的权力。GG。无奈之下,只能尝试在原来的模块做一个 md5 的校验,不对则重新拷贝。但问题又来了,在 Cpp 代码中做 MD5 校验没有现成的方法,还要导入一堆函数,而且执行时间也相对较晚,不够符合我们的要求,最终喜欢偷懒的我们想到了在 init.rc 中执行脚本的方法来完成,重点是用命令行执行 MD5 校验实在是方便了很多很多。
init.rc
中可以启动一个脚本,来完成特定的操作。
service cp_xxx_database /system/etc/init.xxx.database.sh
class main
user root
group root
oneshot
上面的代码说明了一下几点:
- 启动的
section
是一个service
,该服务名字为cp_xxx_database
。 - 启动之后执行的操作是执行一个位于
/system/etc/
目录下的init.xxx.database.sh
脚本。 -
class main
标注了启动方式,通过在init.rc
中的class_start main
指令来启动该服务。 -
user root
和group root
说明了使用的是root
权限。 -
oneshot
说明的是该操作只会执行一次,并不像其他带有restart
指令的service
一样当被kill
调之后会重新调起。
而该脚本则只是一个普通的 shell 脚本。并不复杂,只是跟在服务器上的环境不一样,有些命令需要标明路径。例如脚本开头是 #!/system/bin/sh
。而不是我们平时普通的 #!/bin/sh
,这一点需要注意一下。
#!/system/bin/sh
XxxDatabase="/data/xxx/data/DATABASE"
PreXxxDatabase="/system/etc/XXXXX_XXX_DATABASE"
dbMd5=""
preDbMd5=""
echo "check /data/xxx/data/TAG" > /dev/ttyAMA0
if [ -f "/data/xxx/data/TAG" ];then
echo "xxxxx xxx database is exit" > /dev/ttyAMA0
exit 0;
fi
echo "xxxxx xxx database is not exit, it will cp" > /dev/ttyAMA0
while :
do
rm -rf $XxxDatabase
mkdir -p /data/xxx/data/
chmod 777 /data/xxx/data/
cp -f $PreXxxDatabase $XxxDatabase
sync
preDbMd5=$(md5 $PreXxxDatabase | busybox awk '{print $1}')
dbMd5=$(md5 $XxxDatabase | busybox awk '{print $1}')
echo "cping" > /dev/ttyAMA0
if [ "$dbMd5" == "$preDbMd5" ];then
touch /data/xxx/data/TAG
chmod 777 /data/xxx/data/TAG
break
fi
sleep 0.3
done
echo "copy done" > /dev/ttyAMA0