zoukankan      html  css  js  c++  java
  • Android监听程序自身被卸载

    概述:

      假设不是一些特殊的情况,我想大家非常少会接触到这个需求。事实上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);
    
    	// 初始化log
    	LOG_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) {
    		// 出错log
    		LOG_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的大小,这样一次处理一个event
    		void *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);
    
    		// 文件夹不存在log
    		LOG_DEBUG((*env)->GetStringUTFChars(env, tag, &b_IS_COPY),
    				(*env)->GetStringUTFChars(env, (*env)->NewStringUTF(env, "uninstalled"), &b_IS_COPY));
    
    		if (version >= 17) {
    			// 4.2以上的系统因为用户权限管理更严格,须要加上 --user 0
    			execlp("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);
        }
    }


    效果展示图:




  • 相关阅读:
    【Quartz】常用方法的使用方式(三)
    【Quartz】实现接口封装化(二)
    【Quartz】定时器初步实验(一)
    数据库和ado连接语句的使用总结
    PMP 德尔菲技术
    java 事务解释。
    防盗链--解释
    java月利率计算(等额本息贷款)
    java 日志的数据脱敏
    MDC 输出线程信息帮助定位问题
  • 原文地址:https://www.cnblogs.com/wzzkaifa/p/7017426.html
Copyright © 2011-2022 走看看