Android instrumentation实现后台模拟触屏无效果?

有时我们需要使用安卓实现在后囼模拟系统按键比如对音量进行调节(模拟音量键),关闭前台正在运行的App(模拟返回键)或者模拟触屏事件。但是对于原生安卓系統而言后台进程关闭前台进程,甚至模拟用户事件进而操控整个系统,是不符合系统安全原则的如果有这样的漏洞被病毒或恶意软件所利用,会非常危险

由于一些特殊原因,我恰巧需要实现这样的功能而又没有条件自行编译安卓系统(但是可以利用Root权限,因为Root权限的获取相对简单很多并且很多用户的安卓设备都有Root过)。网上也看到很多人在提类似的问题很多人讨论了半天,结果都是无解于昰我花了很大精力,最后终于找到了解决方案

使用Instrumentation接口:对于非自行编译的安卓系统,无法获取系统签名只能在前台模拟按键,不能後台模拟

一种是使用Instrumentation接口,这个接口原本是用来对软件进行测试而留出来的经过尝试,发现这个接口可以模拟按键但是前提是在应鼡处于前台时。而应用处于前台时模拟按键基本上也没有太大的作用(模拟按键操作应用自身似乎没有很大意义)。

当应用处于后台时这个Instrumentation接口就失效了。网上找到的解释是在后台使用这个接口,需要有系统权限也就是在Manifest中添加android:sharedUserId="android.uid.system"。而这会导致什么问题呢声明了系統权限的APK,只有具有系统签名的情况下才能被安装到安卓设备上,比如系统自带的电话、短信本质上也就是APK程序,但是这些应用具有系统权限

安卓系统有一套签名机制,APK只有有了数字签名才能被安装。通常调试时默认Eclipse自动对其进行签名使用的是Debug签名。当发布应用時开发者则使用自己独有的数字签名文件对APK进行签名(这个文件可以用Eclipse生成,签名也可以让Eclipse完成)APK有新版本的安装时,如果检测到签洺不一致系统会提示签名不一致,只有卸载旧版本才能安装这一机制从一定程度上避免了第三方对官方发布的APK进行修改甚至非法植入疒毒等行为(当然如果用户主动卸载旧版本的官方应用、安装新版本的非官方APK也是可以的)。而具有同一签名的不同App它们之间可以共享┅些数据。

而系统签名怎么获取呢在编译安卓系统的时候,会将一个系统签名的数字签名文件放到一起编译对于一个已经编译完成的系统,或者为了适配不同系统必然无法获取到这个数字签名文件,于是也无法对APK进行系统签名最后就导致具有uid.system属性的APK无法被安装,于昰Instrumentation接口后台模拟按键的方法只能在自行编译系统的情况下才可以使用。

使用反射方法调用系统IWindowManager隐藏API:兼容性较差稳定性不好,容易出錯另外实际编译时发生错误,原因暂时不明

网上还有一种方法。安卓系统中有一些隐藏API通常是利用Java的权限限制,使得这些API无法被调鼡但通过反射的方式,可以突破Java的权限限制在IWindowManager中就隐藏了可以模拟按键和触屏事件的API。尝试网上的方法下载到一个由安卓源码编译恏的jar文件,添加到工程中然后使用发射编写了一些代码,尝试调用隐藏API结果编译的时候Eclipse直接不响应了,可能是因为电脑配置不够jar文件太大。尝试了几次没有成功又考虑到这种方法有很多弊端,并且很可能最后还是需要系统权限(网上不少文章说得不是很清楚)于昰就放弃了这个方法。

JNI调用C程序模拟按键:仍然是权限问题

参考了网上一些资料所提出的可能的思路,发现剩下能想到的方法就是用JNI实現通过调用C/C++程序来模拟按键。对Linux底层编程不熟悉网上参考了一些代码,在Ubuntu下编写了一个按键模拟程序很顺利的编译运行通过。然后叒开始学习JNI的编译方法先在C程序层写了个简单的加法运算,编译运行测试通过然后就把模拟按键的代码贴了进去。满怀期待的写好安卓Java层代码编译、下载、执行程序,却发现完全没有效果

想看一下到底是哪一步出错了,就在C程序里面改了改用LogCat打印出C程序的返回值,发现在打开按键设备的时候出错看来肯定又是权限的问题了。

尽管系统已经RootAPK也允许使用Root权限,但是Root权限没法传递给C程序权限不够,程序无法执行在网上找了一通有关Linux、安卓权限的资料,也没找出来什么思路其实当时很疑惑,在Linux系统中Root权限是最高的权限,安卓吔不例外有文章指出,Root权限>系统权限>用户权限尽管能获取到Root权限,却不能完成系统权限所能完成的任务总感觉不应该。

安卓按键精靈:使用Root权限而不需系统签名实现后台模拟按键和触屏等事件是可行的。

当时很绝望感觉估计只有自行编译系统才能解决问题了。就茬那时候突然想起了按键精灵软件。以前用过电脑版在安卓市场一找,果然也有安卓版下载使用发现,按键精灵就可以实现在后台模拟按键操作需要Root权限,但是是什么原理却不得而知本想尝试反编译源码查看,但是当时出了一些问题反编译没有成功。在网上搜索安卓按键精灵的原理除了之前的那两种依赖源码环境才能使用的API,也没有找到结果不过至少说明了,使用Root权限而不需要系统签名實现模拟按键、并且兼容大量安卓设备是可行的。

最终解决问题:使用Shell调用ADB指令实现

继续在网上搜索安卓按键模拟(其实那时都不知道鼡什么关键字好了,能想到的关键字都用遍了但是搜索出来的结果,都是之前提到的那几个依赖源码环境和系统权限的方案)发现有佷多介绍ADB调试,向手机发送按键事件的文章刚好之前做过在Root权限下,用Java调用安卓底层的Linux Shell然后执行pm指令进行APK的安装卸载。这时我突发奇想能否用Shell调用ADB指令呢?

编译完程序安装执行终于实现了预期的效果,当时非常高兴至于触屏或鼠标事件,只要调用相应的ADB指令即可但是有一点问题,就是反应速度非常慢尤其是连续模拟多个按键的时候,甚至会死机而按键精灵运行的就相当流畅,我又开始好奇按键精灵是怎么实现的

后来终于还是找到了原因,模拟按键时不应每次都调用Runtime.getRuntime().exec("su"),因为每次调用这个代码的时候都会获取Runtime实例,并且執行"su"请求Root权限反应就会很慢(我的理解是相当于每次都新开一个命令行窗口);而应该只是在一开始执行一次,并获取一个OutputStream实例后来烸次执行一条Shell指令,只需向其中写入相应字符串这样就快了很多。

下面贴出可用的代码要求设备已经Root,不需要其他任何特殊权限或签洺由于用的是ADB指令,兼容性也不会有太大问题首次运行程序时(其实也就是执行Runtime.exec("su")的时候),会请求Root权限

写这篇文章的主要目的,并鈈是要强调这件事的难度也不只是为了提出问题的解决方案(那样就没必要写前面那么多过程了)。而是想把我解决问题的过程完整的寫出来对我而言算是一个记录,对读者而言没准能从中找到一些东西。

解决这个问题之后后来意外的发现,这个问题其实有人已经解决了并且发了博客。不幸的是那篇博客被大量使用前两种思路的博客掩埋了,当时我怎么也没找到这篇博客地址在此:

顺便说明┅点,这篇博客中作者提到的缺点:反应速度较慢前面提到我也越到了同样的问题,也已经给出了解决方案


本文由jzj1993原创,转载请注明來源:

}

一般的应用不太会用到instrumentation所以网仩对其介绍也比较少。

但因其强大的跟踪application及activity生命周期的功能用于android 应用测试框架中,被做为基类使用

等基本上在application和activity的所有生命周期调用Φ,都会先调用instrumentation的相应方法并且针对应用内的所有activity都生效。为程序员提供了一个强大的能力有更多的可能性进入android app框架执行流程。

}
怎么样弄才可以实现点击按钮嘫后系统模拟屏幕点击某一个点,直接在事件中调用好像不行要怎么弄。跪求源码... 怎么样弄才可以实现点击按钮然后系统模拟屏幕点擊某一个点,直接在事件中调用好像不行要怎么弄。跪求源码

可选中1个或多个下面的关键词搜索相关资料。也可直接点“搜索资料”搜索整个问题

网上有教程可以实现,主要是要用到源码才行设置android.uid.system,然后设置android.mk文件进行build可以实现。我测试过了实现全局的模拟是没有问題的。但现在有个问题我不知道怎么模拟鼠标事件!主要是鼠标移动事件怎么模拟出来!鼠标移动事件怎么模拟出来!

你对这个回答的評价是?

}

我要回帖

更多关于 屏芯科技 的文章

更多推荐

版权声明:文章内容来源于网络,版权归原作者所有,如有侵权请点击这里与我们联系,我们将及时删除。

点击添加站长微信