android安全问题(二) 程序锁
博客分类: android
导读:本文介绍如何实现对应用加锁的功能,无须root权限
某些人有时候会有这样一种需求,小A下载了个软件,只是软件中的美女过于诱惑与暴露,所以他不想让别人知道这是个什么软件,起码不想让别人打开浏览。而这款软件又没有锁,任何人都可以打开,肿么办呢?如果打开它的时候需要输入密码,那该多好阿!于是,程序锁这种应用就产生了
程序锁不是最近才有的,很久之前android就有这种apk了
这一期我们来苛刻如何实现程序加锁功能
首先,我们先明确一下我们要做的程序具有什么功能
1可以选择需要加锁的程序
2可以设置密码
3可以关闭程序锁
这里作为演示,我们就尽量简化代码
我们先说最关键的部分
最关键的地方在于:当用户打开一个应用的时候,怎么弹出密码页面?
这里没有什么太好的办法,需要扫描task中的topActivity
首先,我们先获得运行的task
- mActivityManager=(ActivityManager)context.getSystemService("activity");
- //mActivityManager.getRunningTasks(1);//List<RunningTaskInfo>
getRunningTasks方法返回一个List,我们来看看这个List是什么
……
返回的List是有序的,第一个是最近的,所以我们取出第一个即可,然后得到此task中的最上层的Activity
- ComponentNametopActivity=mActivityManager.getRunningTasks(1).get(0).topActivity;
topActivity居然是ComponentName类型,下面的事情就好办了,获得包名和类名
- ComponentNametopActivity=mActivityManager.getRunningTasks(1).get(0).topActivity;
- StringpackageName=topActivity.getPackageName();
- StringclassName=topActivity.getClassName();
- Log.v(TAG,"packageName"+packageName);
- Log.v(TAG,"className"+className);
- if(testPackageName.equals(packageName)
- &&testClassName.equals(className)){
- Intentintent=newIntent();
- intent.setClassName("com.example.locktest","com.example.locktest.PasswordActivity");
- intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
- mContext.startActivity(intent);
- }
由于我没有选择程序这一步,所以我就固定一个应用做测试,这里选择的是htc的note应用
- StringtestPackageName="com.htc.notes";
- StringtestClassName="com.htc.notes.collection.NotesGridViewActivity";
下面我们该想,这段代码何时执行了
打开一个应用程序,系统不会发送广播,我们无法直接监听,所以这里我们采取定时扫描的策略
这里只是一个简单的实现,之后我们再讨论优化
我们采取每秒中检查一次task的方式,这里使用Timer吧,用Handler也一样可以实现
- privateTimermTimer;
- privatevoidstartTimer(){
- if(mTimer==null){
- mTimer=newTimer();
- LockTasklockTask=newLockTask(this);
- mTimer.schedule(lockTask,0L,1000L);
- }
- }
到这里,其实我们的关键代码就已经完成了
下面贴出完整带代码,注意:我们只关注弹出锁界面这部分,其他部分自行实现(比如文章末尾提到的)
Task,负责检查task,并在适当的时候弹出密码页面
- publicclassLockTaskextendsTimerTask{
- publicstaticfinalStringTAG="LockTask";
- privateContextmContext;
- StringtestPackageName="com.htc.notes";
- StringtestClassName="com.htc.notes.collection.NotesGridViewActivity";
- privateActivityManagermActivityManager;
- publicLockTask(Contextcontext){
- mContext=context;
- mActivityManager=(ActivityManager)context.getSystemService("activity");
- }
- @Override
- publicvoidrun(){
- ComponentNametopActivity=mActivityManager.getRunningTasks(1).get(0).topActivity;
- StringpackageName=topActivity.getPackageName();
- StringclassName=topActivity.getClassName();
- Log.v(TAG,"packageName"+packageName);
- Log.v(TAG,"className"+className);
- if(testPackageName.equals(packageName)
- &&testClassName.equals(className)){
- Intentintent=newIntent();
- intent.setClassName("com.example.locktest","com.example.locktest.PasswordActivity");
- intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
- mContext.startActivity(intent);
- }
- }
- }
LockService,负责执行定时任务,取消任务等
- publicclassLockServiceextendsService{
- privateTimermTimer;
- publicstaticfinalintFOREGROUND_ID=0;
- privatevoidstartTimer(){
- if(mTimer==null){
- mTimer=newTimer();
- LockTasklockTask=newLockTask(this);
- mTimer.schedule(lockTask,0L,1000L);
- }
- }
- publicIBinderonBind(Intentintent){
- returnnull;
- }
- publicvoidonCreate(){
- super.onCreate();
- startForeground(FOREGROUND_ID,newNotification());
- }
- publicintonStartCommand(Intentintent,intflags,intstartId){
- startTimer();
- returnsuper.onStartCommand(intent,flags,startId);
- }
- publicvoidonDestroy(){
- stopForeground(true);
- mTimer.cancel();
- mTimer.purge();
- mTimer=null;
- super.onDestroy();
- }
- }
MainActivity,测试用,作为应用入口,启动service(产品中,我们可以在receiver中启动service)。
- publicclassMainActivityextendsActivity{
- publicvoidonCreate(BundlesavedInstanceState){
- super.onCreate(savedInstanceState);
- startService(newIntent(this,LockService.class));
- }
- }
PasswordActivity,密码页面,很粗糙,没有核对密码逻辑,自行实现
记得重写onBackPressed函数,不然按返回键的时候……你懂的
- publicclassPasswordActivityextendsActivity{
- privatestaticfinalStringTAG="PasswordActivity";
- ButtonokButton;
- EditTextpasswordEditText;
- privatebooleanmFinish=false;
- @Override
- protectedvoidonCreate(BundlesavedInstanceState){
- super.onCreate(savedInstanceState);
- setContentView(R.layout.password);
- passwordEditText=(EditText)findViewById(R.id.password);
- okButton=(Button)findViewById(R.id.ok);
- okButton.setOnClickListener(newView.OnClickListener(){
- publicvoidonClick(Viewv){
- Stringpassword=passwordEditText.getText().toString();
- Log.v(TAG,"password"+password);
- mFinish=true;
- finish();
- }
- });
- }
- publicvoidonBackPressed(){}
- publicvoidonPause(){
- super.onPause();
- if(!mFinish){
- finish();
- }
- }
- }
xml这里就不贴了,记得添加权限
- <uses-permissionandroid:name="android.permission.GET_TASKS"/>
关于程序的其他部分,这里只做简要说明
选择应用对其进行加锁部分
1列出系统中所有程序(你也可以自由发挥,比如过滤掉原始应用)
2选择,然后存入数据库(当然,最好也有取消功能,记得从数据库中删除数据)
程序锁总开关
可以使用sharedPreference,设置一个boolean开关
现在,当我想要打开htc的note应用的时候,就会弹出密码页面当我解锁,按home会回到桌面,长按home,点击note,还是会弹出密码框
因为是每秒检查一次,所以可能会有一点点延迟,你可以设置为500毫秒,但是越频繁,占用资源就越多
上面的代码我取得topActivity后检查了其包名行和类名,所以只有当打开指定的页面的时候,才会弹出密码锁
比如我对Gallery应用加密了,但是用户正在编辑短信,这时候它想发彩信,于是他通过短信进入到了Gallery……
对于某些用户的某些需求来说,这是不能容忍的,这时,我们只需简单修改下判断逻辑即可:只检查包名,包名一致就弹出密码锁,这样就完美了
程序锁我就分析到这里
最后一句
当使用程序锁的时候,你长按home,发现程序锁也出现在“最近的任务”中,肿么办……给此activity设置android:excludeFromRecents="true"即可