Android应用被卸载后,自动使用 浏览器打开指定连接(或编写C代码执行其他操作)
2015-12-04 18:12
1046 查看
本文,提供“Android应用被卸载后,自动使用 浏览器打开指定连接”的方法。
原理:在安卓程序中某处,基于JNI调用C代码开启一个子进程监控应用在系统中的文件目录,一旦应用被卸载,该目录将会被系统删除,此时触发子进程执行相关代码(本例调用浏览器执行打开一个连接)
1、在安卓项目下创建jni目录
2、在jni目录下创建文件observer.c
3、在jni目录下创建文件Android.mk
4、在创建natvie方法,并在Android中调用JNI方法
注:本文已经更新,并解决了之前“fork多个子进程”的问题。
原理:在安卓程序中某处,基于JNI调用C代码开启一个子进程监控应用在系统中的文件目录,一旦应用被卸载,该目录将会被系统删除,此时触发子进程执行相关代码(本例调用浏览器执行打开一个连接)
1、在安卓项目下创建jni目录
2、在jni目录下创建文件observer.c
/* * Copyright (C) 2009 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. * */ #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[] = "OnAppUninstall"; static jboolean b_IS_COPY = JNI_TRUE; /* native方法定义 */ JNIEXPORT jstring JNICALL Java_com_zgy_catchuninstallself_UninstallObserver_startWork( JNIEnv* env, jobject thiz, jstring dirStr, jstring data_packagedir, jstring activity, jstring action, jstring action_data, jstring userSerial) { jstring tag = (*env)->NewStringUTF(env, c_TAG); LOG_DEBUG((*env)->GetStringUTFChars(env, tag, &b_IS_COPY), (*env)->GetStringUTFChars(env, (*env)->NewStringUTF(env, "start------------------"), &b_IS_COPY)); //初始化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(); char showpid[10]; sprintf(showpid, "%d", pid); LOG_INFO((*env)->GetStringUTFChars(env, tag, &b_IS_COPY), (*env)->GetStringUTFChars(env, (*env)->NewStringUTF(env, showpid), &b_IS_COPY)); 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) { LOG_DEBUG((*env)->GetStringUTFChars(env, tag, &b_IS_COPY), (*env)->GetStringUTFChars(env, (*env)->NewStringUTF(env, "childen process"), &b_IS_COPY)); 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); } LOG_DEBUG((*env)->GetStringUTFChars(env, tag, &b_IS_COPY), (*env)->GetStringUTFChars(env, (*env)->NewStringUTF(env, "will creat a lockfile"), &b_IS_COPY)); LOG_DEBUG((*env)->GetStringUTFChars(env, tag, &b_IS_COPY), (*env)->GetStringUTFChars(env, dirStr, &b_IS_COPY)); LOG_DEBUG((*env)->GetStringUTFChars(env, tag, &b_IS_COPY), (*env)->GetStringUTFChars(env, (*env)->NewStringUTF(env, "creat a new process"), &b_IS_COPY)); //子进程注册"/data/data/{package}"目录监听器 int watchDescriptor; watchDescriptor = inotify_add_watch(fileDescriptor, (*env)->GetStringUTFChars(env, dirStr, &b_IS_COPY), 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)); size_t readBytes = read(fileDescriptor, p_buf, sizeof(struct inotify_event)); //等待5秒 LOG_DEBUG((*env)->GetStringUTFChars(env, tag, &b_IS_COPY), (*env)->GetStringUTFChars(env, (*env)->NewStringUTF(env, "sleep 2 seconds"), &b_IS_COPY)); sleep(2); //如果是覆盖安装跳过 FILE *p_appDir = fopen(data_packagedir, "r"); if (p_appDir != NULL) { LOG_DEBUG((*env)->GetStringUTFChars(env, tag, &b_IS_COPY), (*env)->GetStringUTFChars(env, (*env)->NewStringUTF(env, "reinstall"), &b_IS_COPY)); exit(1); } //read会阻塞进程,走到这里说明收到目录被删除的事件,注销监听器 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)); //执行命令am start -a android.intent.action.VIEW -d http://ww.baidu.com // execlp("am", "am", "start", "-a", "android.intent.action.VIEW", "-d", "http://ww.baidu.com", (char *)NULL); //4.2以上的系统由于用户权限管理更严格,需要加上 --user 0 if (userSerial == NULL) { // 执行命令am start -a android.intent.action.VIEW -d $(url) execlp("am", "am", "start", "-a", "android.intent.action.VIEW", "-d", (*env)->GetStringUTFChars(env, action_data, &b_IS_COPY), (char *) NULL); } else { if (activity == NULL || (*env)->GetStringUTFLength(env, activity) < 1) { execlp("am", "am", "start", "--user", (*env)->GetStringUTFChars(env, userSerial, &b_IS_COPY), "-a", (*env)->GetStringUTFChars(env, action, &b_IS_COPY), "-d", (*env)->GetStringUTFChars(env, action_data, &b_IS_COPY), (char *) NULL); } else { if (action == NULL || (*env)->GetStringUTFLength(env, action) < 1) { if (action_data == NULL || (*env)->GetStringUTFLength(env, action_data) < 1) { execlp("am", "am", "start", "--user", (*env)->GetStringUTFChars(env, userSerial, &b_IS_COPY), "-n", (*env)->GetStringUTFChars(env, activity, &b_IS_COPY), (char *) NULL); } else { execlp("am", "am", "start", "--user", (*env)->GetStringUTFChars(env, userSerial, &b_IS_COPY), "-n", (*env)->GetStringUTFChars(env, activity, &b_IS_COPY), "-d", (*env)->GetStringUTFChars(env, action_data, &b_IS_COPY), (char *) NULL); } } else { if (action_data == NULL || (*env)->GetStringUTFLength(env, action_data) < 1) { execlp("am", "am", "start", "--user", (*env)->GetStringUTFChars(env, userSerial, &b_IS_COPY), "-n", (*env)->GetStringUTFChars(env, activity, &b_IS_COPY), "-a", (*env)->GetStringUTFChars(env, action, &b_IS_COPY), (char *) NULL); } else { execlp("am", "am", "start", "--user", (*env)->GetStringUTFChars(env, userSerial, &b_IS_COPY), "-n", (*env)->GetStringUTFChars(env, activity, &b_IS_COPY), "-a", (*env)->GetStringUTFChars(env, action, &b_IS_COPY), "-d", (*env)->GetStringUTFChars(env, action_data, &b_IS_COPY), (char *) NULL); } } } } //扩展:可以执行其他shell命令,am(即activity manager),可以打开某程序、服务,broadcast intent,等等 } else { //父进程直接退出,使子进程被init进程领养,以避免子进程僵死 LOG_DEBUG((*env)->GetStringUTFChars(env, tag, &b_IS_COPY), (*env)->GetStringUTFChars(env, (*env)->NewStringUTF(env, "fork pid > 0"), &b_IS_COPY)); } LOG_DEBUG((*env)->GetStringUTFChars(env, tag, &b_IS_COPY), (*env)->GetStringUTFChars(env, (*env)->NewStringUTF(env, "end------------------"), &b_IS_COPY)); return (*env)->NewStringUTF(env, showpid); }
3、在jni目录下创建文件Android.mk
LOCAL_PATH := $(call my-dir) include $(CLEAR_VARS) LOCAL_MODULE:=observer LOCAL_SRC_FILES:=observer.c LOCAL_C_INCLUDES:= $(LOCAL_PATH)/include LOCAL_LDLIBS += -L$(SYSROOT)/usr/lib -llog include $(BUILD_SHARED_LIBRARY)
4、在创建natvie方法,并在Android中调用JNI方法
package com.zgy.catchuninstallself; import java.io.BufferedReader; import java.io.IOException; import java.io.InputStreamReader; import java.lang.reflect.InvocationTargetException; import java.lang.reflect.Method; import java.util.ArrayList; import java.util.HashMap; import java.util.List; import java.util.Map; import android.content.Context; import android.content.SharedPreferences; import android.content.pm.PackageInfo; import android.os.Build; import android.util.Log; /** * 应用程序卸载监控,入口调用方法:openUrlWhenUninstall(……) * * @author SHANHY(365384722@QQ.COM) * @date 2015年12月15日 */ public class UninstallObserver { // path:data/data/[packageNmae]/lib // url:跳转的页面,需要http://或https://开头 /** * Native 方法 * * @param dirStr 要监听的目录,传值:/data/data/{packageNmae}/lib * @param activity Activity对象 * @param action android.intent.action.VIEW * @param actionData 要在浏览器中打开的URL * @param userSerial * @return * @author SHANHY * @date 2015年12月15日 */ private static native String startWork(String dirStr,String dataPackageDir, String activity, String action, String actionData, String userSerial); static { System.loadLibrary("observer"); } private static SharedPreferences sharedPreferences; private static SharedPreferences.Editor editor; /** * 在需要开启子进程监听的地方调用该方法 * * @param context * @param openUrl * @author SHANHY * @date 2015年12月15日 */ public static void openUrlWhenUninstall(Context context, String openUrl) { sharedPreferences = context.getSharedPreferences("process", Context.MODE_PRIVATE); editor = sharedPreferences.edit(); String pid = sharedPreferences.getString("pid", "0"); if (checkChildProcess(context, pid)) { Log.i("OnAppUninstall", "已经开启过守护进程"); return; } Log.i("OnAppUninstall", "执行开启过守护进程"); String dataPackageDir = context.getApplicationInfo().dataDir; String dirStr = context.getApplicationInfo().dataDir + "/lib";// 监听lib目录是因为:lib目录在进行应用清理的时候不会被清理 String activity = null; if (checkInstall(context, "com.android.browser")) { activity = "com.android.browser/com.android.browser.BrowserActivity"; } String action = "android.intent.action.VIEW"; String gpid; if (Build.VERSION.SDK_INT < 17) { gpid = startWork(dirStr, dataPackageDir, activity, action, openUrl, null); } else {//4.2以上的系统由于用户权限管理更严格,调用API有所区别 gpid = startWork(dirStr, dataPackageDir, activity, action, openUrl, getUserSerial(context)); } if (gpid != null && !gpid.equals("") && !gpid.equals("0")) { Log.i("OnAppUninstall", "获得守护进程pid-->" + gpid); editor.putString("pid", gpid); editor.commit(); } } /** * 检查安装 * * @param context * @param packageName * @return * @author SHANHY * @date 2015年12月15日 */ private static boolean checkInstall(Context context, String packageName) { try { PackageInfo packageInfo = context.getPackageManager().getPackageInfo(packageName, 0); if (packageInfo == null) { return false; } else { return true; } } catch (Exception e) { return false; } } /** * 获取 UserSerial * * @param ctx * @return * @author SHANHY * @date 2015年12月15日 */ private static String getUserSerial(Context ctx) { Object userManager = ctx.getSystemService("user"); if (userManager == null) { return null; } try { Method myUserHandleMethod = android.os.Process.class.getMethod("myUserHandle", (Class<?>[]) null); Object myUserHandle = myUserHandleMethod.invoke(android.os.Process.class, (Object[]) null); Method getSerialNumberForUser = userManager.getClass().getMethod("getSerialNumberForUser", myUserHandle.getClass()); long userSerial = (Long) getSerialNumberForUser.invoke(userManager, myUserHandle); return String.valueOf(userSerial); } catch (NoSuchMethodException e) { } catch (IllegalArgumentException e) { } catch (IllegalAccessException e) { } catch (InvocationTargetException e) { } return null; } /** * 检查子进程 * * @param context * @param mpid * @return * @author SHANHY * @date 2015年12月15日 */ private static boolean checkChildProcess(Context context, String mpid) { boolean resflag = false; int mypid = android.os.Process.myPid(); Log.i("OnAppUninstall", "mypid-->" + mypid); BufferedReader in = null; long starttime = System.currentTimeMillis(); List<Map<String, String>> listdata = new ArrayList<Map<String, String>>(); try { Process p = Runtime.getRuntime().exec("ps"); in = new BufferedReader(new InputStreamReader(p.getInputStream())); String line = null; String[] temp; boolean flag = false; // int length = 0; while ((line = in.readLine()) != null) { // Log.i("OnAppUninstall", "ps-->" + line); if (!flag) { flag = true; continue; } line = line.replaceAll(" +", " "); temp = line.split(" "); // System.out.println(context.getPackageName()); // System.out.println(temp[8]); if (temp[8].trim().equals(context.getPackageName().trim()) && (temp[2].trim().equals(mypid + "") || temp[2].trim().equals("1"))) { Log.i("OnAppUninstall", "get it"); Map<String, String> map = new HashMap<String, String>(); map.put("pid", temp[1]); map.put("pname", temp[8]); listdata.add(map); } } } catch (IOException e) { e.printStackTrace(); } finally { if (in != null) { try { in.close(); } catch (IOException e) { e.printStackTrace(); } in = null; } } long endtime = System.currentTimeMillis(); long spendtime = endtime - starttime; for (int i = 0; i < listdata.size(); i++) { if (listdata.get(i).get("pid").equals(mpid)) { resflag = true; } else { Log.w("OnAppUninstall", "捕获到多余守护进程-->" + listdata.get(i).get("pid") + " " + listdata.get(i).get("pname")); try { android.os.Process.killProcess(Integer.valueOf(listdata.get(i).get("pid").trim())); } catch (NumberFormatException e) { e.printStackTrace(); } } Log.i("OnAppUninstall", listdata.get(i).get("pid") + " " + listdata.get(i).get("pname")); } Log.i("OnAppUninstall", "spendtime-->" + spendtime); return resflag; } }
String url = "http://www.baidu.com"; UninstallObserver.openUrlWhenUninstall(this, url);
注:本文已经更新,并解决了之前“fork多个子进程”的问题。
相关文章推荐
- android问题集、
- android 软键盘隐藏
- RxJava教程(四)在Android中使用RxJava
- Android内存泄露分析(MAT工具的使用)
- Android 检查设备是否存在 导航栏 NavigationBar
- Android 常用布局视图
- Android常用工具类之 Toast工具类
- android adb端口被占用解决方案
- Android手势之GestureOverlayView
- android 添加一个简单的延时
- AndroidAnnotations注解说明
- Android 点击事件 4种 写法
- Android中5种Button点击事件的实现方法
- Android 控件之对EditText的详细解读
- Android 4.4(KitKat)中VSync信号的虚拟化
- 完美解决Android SDK Manager无法更新
- Android Rtmp客户端搭建
- 如何用Sencha Touch打包Android的APK
- Android之SharedPreferences内部原理浅析
- Android Studio关于新建项目引入V7包导致的错误(最后附上Android Studio入门指南链接)