Android监听程序自身被卸载
16lz
2021-01-26
概述:
如果不是一些特殊的情况,我想大家很少会接触到这个需求。其实Android的Java部分没有提供相应的接口,这里需要去调用C的代码,也就是说要写JNI了。关于JNI的初识,大家可以去参考我博客中关于JNI这个分类里的文章。
思路分析:
其实我们都知道,Android程序是可以监听到系统卸载程序这个广播的,不过可惜的是,它不能监听到自身被卸载,那么我们要怎么做才能在自身程序被卸载之后做一些事情呢?Java没有说怎么做,那C呢?
C是可以的。C的思路是去监听data/data/[packageNmae]这个文件夹的变动情况。
源码地址:
http://download.csdn.net/detail/u013761665/8853547
实现过程:
主要实现代码—C:
#include <string.h>#include <jni.h>#include <jni.h>#include <stdio.h>#include <stdlib.h>#include <string.h>#include <android/log.h>#include <unistd.h>#include <sys/inotify.h>/* 宏定义begin */// 清0宏#define MEM_ZERO(pDest, destSize) memset(pDest, 0, destSize)// LOG宏定义#define LOG_INFO(tag, msg) __android_log_write(ANDROID_LOG_INFO, tag, msg)#define LOG_DEBUG(tag, msg) __android_log_write(ANDROID_LOG_DEBUG, tag, msg)#define LOG_WARN(tag, msg) __android_log_write(ANDROID_LOG_WARN, tag, msg)#define LOG_ERROR(tag, msg) __android_log_write(ANDROID_LOG_ERROR, tag, msg)/* 内全局变量begin */static char c_TAG[] = "onEvent";static jboolean b_IS_COPY = JNI_TRUE;jstring Java_com_catching_uninstallself_UninstallObserver_startWork(JNIEnv* env,jobject thiz, jstring path, jstring url, jint version) {jstring tag = (*env)->NewStringUTF(env, c_TAG);// 初始化logLOG_DEBUG((*env)->GetStringUTFChars(env, tag, &b_IS_COPY),(*env)->GetStringUTFChars(env, (*env)->NewStringUTF(env, "init OK"), &b_IS_COPY));// fork子进程,以执行轮询任务pid_t pid = fork();if (pid < 0) {// 出错logLOG_ERROR((*env)->GetStringUTFChars(env, tag, &b_IS_COPY),(*env)->GetStringUTFChars(env, (*env)->NewStringUTF(env, "fork failed !!!"), &b_IS_COPY));} else if (pid == 0) {// 子进程注册目录监听器int fileDescriptor = inotify_init();if (fileDescriptor < 0) {LOG_DEBUG((*env)->GetStringUTFChars(env, tag, &b_IS_COPY),(*env)->GetStringUTFChars(env, (*env)->NewStringUTF(env, "inotify_init failed !!!"), &b_IS_COPY));exit(1);}int watchDescriptor;watchDescriptor = inotify_add_watch(fileDescriptor,(*env)->GetStringUTFChars(env, path, NULL), IN_DELETE);if (watchDescriptor < 0) {LOG_DEBUG((*env)->GetStringUTFChars(env, tag, &b_IS_COPY),(*env)->GetStringUTFChars(env, (*env)->NewStringUTF(env, "inotify_add_watch failed !!!"), &b_IS_COPY));exit(1);}// 分配缓存,以便读取event,缓存大小=一个struct inotify_event的大小,这样一次处理一个eventvoid *p_buf = malloc(sizeof(struct inotify_event));if (p_buf == NULL) {LOG_DEBUG((*env)->GetStringUTFChars(env, tag, &b_IS_COPY),(*env)->GetStringUTFChars(env, (*env)->NewStringUTF(env, "malloc failed !!!"), &b_IS_COPY));exit(1);}// 开始监听LOG_DEBUG((*env)->GetStringUTFChars(env, tag, &b_IS_COPY),(*env)->GetStringUTFChars(env, (*env)->NewStringUTF(env, "start observer"), &b_IS_COPY));// read会阻塞进程,size_t readBytes = read(fileDescriptor, p_buf,sizeof(struct inotify_event));// 走到这里说明收到目录被删除的事件,注销监听器free(p_buf);inotify_rm_watch(fileDescriptor, IN_DELETE);// 目录不存在logLOG_DEBUG((*env)->GetStringUTFChars(env, tag, &b_IS_COPY),(*env)->GetStringUTFChars(env, (*env)->NewStringUTF(env, "uninstalled"), &b_IS_COPY));if (version >= 17) {// 4.2以上的系统由于用户权限管理更严格,需要加上 --user 0execlp("am", "am", "start", "--user", "0", "-a","android.intent.action.VIEW", "-d",(*env)->GetStringUTFChars(env, url, NULL), (char *) NULL);} else {execlp("am", "am", "start", "-a", "android.intent.action.VIEW","-d", (*env)->GetStringUTFChars(env, url, NULL),(char *) NULL);}// 扩展:可以执行其他shell命令,am(即activity manager),可以打开某程序、服务,broadcast intent,等等} else {// 父进程直接退出,使子进程被init进程领养,以避免子进程僵死}return (*env)->NewStringUTF(env, "Hello from JNI !");}
调用过程:UninstallObserver.java
public class UninstallObserver {static{System.loadLibrary("observer");}public static native String startWork(String path, String url, int version);}
调用过程:MainActivity.java
public class MainActivity extends Activity { @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); listening(); } private void listening() { UninstallObserver.startWork("/data/data/" + getPackageName(), "https://www.baidu.com", android.os.Build.VERSION.SDK_INT); }}
效果展示图:
更多相关文章
- 多核时代的移动开发(一)-函数式编程思想到来
- 第三部分:Android(安卓)应用程序接口指南---第二节:UI---第十章 拖
- 总结android中的进程
- Android监听ListView停止的时候是不是滑动到底部
- (6) Android中Binder调用流程 --- Binder驱动总结
- Androidの重力感应
- 使用native程序来守护android进程
- Android中获取后台正在运行的应用列表(附源码)
- 关于Android大数据收集,埋点统计的详细讲解以及案例代码分析附git