Android 解决不同进程发送KeyEvent 的问题

本文介绍了一种通过修改Android应用的用户ID为系统级别,并通过JNI方法调用内核发送KeyEvent,实现在不同进程间模拟按键的技术方案。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

最近在做有关于Remote Controller 的功能,该功能把手机做成TV的遥控器来处理。在手机的客户端发送消息到TV的android 服务端,服务端接收到客户端的请求消息,模拟KeyEvent命令,发送Key值。

最简单的发送命令为如下代码:

复制代码
public static voidsimulateKeystroke( final intKeyCode){

newThread( newRunnable(){

public voidrun(){
// TODOAuto-generatedmethodstub
try{

Instrumentationinst= newInstrumentation();
inst.sendKeyDownUpSync(KeyCode);
} catch(Exceptione){
// TODO:handleexception
}
}
}).start();
}
复制代码

这种方法在当前的界面和相同的进程上是没有问题的,可以实现的基本的需求。但当我还是把服务开启着,按HOME将服务或者界面退出到后台时,再通过客户端向服务端服务发送消息使其模拟按键时,不幸的事情发送了:

Injecting to another application requires INJECT_EVENTS permission

提示没有INJECT_EVENTS这个权限。没则加之,在AndroidManifest.xml文件里面添加该权限,再运行,问题还是没有解决,原因是上面代码最终还是调用的WindowsManagerService 里面的injectKeyEvent方法,该方法会去验证你当前的程序的pid和uid,如果两者在分发key 键时返回-1则会提示上面的error.

好了,废话一大堆,下面到了真正解决这一问题的方法了。

网上各种google 各种百度,找不到自己需要的答案。

想过一个方法是(尚未验证):

通过jni的方法将kernel 的发送keyevent的方法用NDK封装成方法,做成库给java调用,从而绕过Android WindowsManagerService 的验证,这是我初期想到的解决思路,但尚未验证。

另外一个通过验证的方法为:

将你的服务的userId改成系统级别的,在manifest加如下代码:

<manifestxmlns:android="http://schemas.android.com/apk/res/android"

package ="com.xuzhitech.remote.server"
android:versionCode
="1"
android:versionName
="1.0"
android:sharedUserId
="android.uid.system" >

加上这一代码,需要在源码里面编译才能生效,添加Android.mk文件:

复制代码
LOCAL_PATH:=$(callmy-dir)
include$(CLEAR_VARS)

LOCAL_MODULE_TAGS:=optional

LOCAL_SRC_FILES:=$(callall-subdir-java-files)

LOCAL_PACKAGE_NAME:=RemoteAndroidServer
LOCAL_CERTIFICATE:=platform
#LOCAL_CERTIFICATE:=share

LOCAL_OVERRIDES_PACKAGES:=Home
include$(BUILD_PACKAGE)
复制代码

这里的LOCAL_CERTIFICATE 要使用platform编译,而不是share编译。

到了这里,你就可以跨进程模拟按键了。

Best Regards.

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值