详解Android App卸载后跳转到指定的反馈页面的方法

很多人也许会问:360被卸载之后会跳转到指定的反馈页面,是怎么弄的?

其实这个问题的核心就在于:应用被卸载了,如果能够做到后续的代码逻辑继续执行

我们再来仔细分析一下场景和流程
一个应用被用户卸载肯定是有理由的,而开发者却未必能得知这一重要的理由,毕竟用户很少会主动反馈建议,多半就是用得不爽就卸,如果能在被卸载后获取到用户的一些反馈,那对开发者进一步改进应用是非常有利的。目前据我所知,国内的Android应用中实现这一功能的只有360手机卫士、360平板卫士,那么如何实现这一功能的?

我们可以把实现卸载反馈的问题转化为监听自己是否被卸载,只有得知自己被卸载,才可以设计相应的反馈处理流程。以下的列表是我在研究这一问题的思路:

1、注册BroadcastReceiver,监听"android.intent.action.PACKAGE_REMOVED"系统广播
结果:NO。未写代码,直接分析,卸载的第一步就是退出当前应用的主进程,而此广播是在已经卸载完成后才发出的,此时主进程都没有了,去哪onReceive()呢?

2、若能收到"将要卸载XX包"的系统广播,在主进程被退出之前就抢先进行反馈处理就好了,可惜没有这样的系统广播,不过经过调研,倒是发现了一个办法,读取系统log,当日志中包含"android.intent.action.DELETE"和自己的包名时,意味着自己将要被卸载。
结果:NO。调试时发现此方法有两个缺陷,(1)点击设置中的卸载按钮即发出此Intent,此时用户尚未在弹框中确认卸载;(2)pm命令卸载不出发此Intent,意味着被诸如手机安全管家,豌豆荚等软件卸载时,无法提前得知卸载意图。

3、由于时间点不容易把控,所以干脆不依赖系统广播或log,考虑到卸载过程会删除"/data/data/包名"目录,我们可以用线程直接轮询这个目录是否存在,以此为依据判断自己是否被卸载。
结果:NO。同方法1,主进程退出,相应的线程必定退出,线程还没等到判断目录是否存在就已经被销毁了。

4、改用C端进程轮询"/data/data/包名"目录是否存在
结果:YES。借助Java端进程fork出来的C端进程在应用被卸载后不会被销毁。

解决的方案确定了,下面来看一下代码吧:

#include <jni.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <android/log.h>
#include <unistd.h>
#include <sys/inotify.h> 

#include "com_example_uninstalldemos_NativeClass.h" 

/* 宏定义begin */
//清0宏
#define MEM_ZERO(pDest, destSize) memset(pDest, 0, destSize) 

#define LOG_TAG "onEvent" 

//LOG宏定义
#define LOGD(fmt, args...) __android_log_print(ANDROID_LOG_INFO, LOG_TAG, fmt, ##args) 

JNIEXPORT jstring JNICALL Java_com_example_uninstalldemos_NativeClass_init(JNIEnv* env, jobject thiz) { 

  //初始化log
  LOGD("init start..."); 

  //fork子进程,以执行轮询任务
  pid_t pid = fork();
  if (pid < 0) {
    //出错log
    LOGD("fork failed...");
  } else if (pid == 0) {
    //子进程注册"/data/data/pym.test.uninstalledobserver"目录监听器
    int fileDescriptor = inotify_init();
    if (fileDescriptor < 0) {
      LOGD("inotify_init failed...");
      exit(1);
    } 

    int watchDescriptor;
    watchDescriptor = inotify_add_watch(fileDescriptor,"/data/data/com.example.uninstalldemos", IN_DELETE);
    LOGD("watchDescriptor=%d",watchDescriptor);
    if (watchDescriptor < 0) {
      LOGD("inotify_add_watch failed...");
      exit(1);
    } 

    //分配缓存,以便读取event,缓存大小=一个struct inotify_event的大小,这样一次处理一个event
    void *p_buf = malloc(sizeof(struct inotify_event));
    if (p_buf == NULL) {
      LOGD("malloc failed...");
      exit(1);
    }
    //开始监听
    LOGD("start observer...");
    size_t readBytes = read(fileDescriptor, p_buf,sizeof(struct inotify_event)); 

    //read会阻塞进程,走到这里说明收到目录被删除的事件,注销监听器
    free(p_buf);
    inotify_rm_watch(fileDescriptor, IN_DELETE); 

    //目录不存在log
    LOGD("uninstall"); 

    //执行命令am start -a android.intent.action.VIEW -d http://shouji.360.cn/web/uninstall/uninstall.html
    execlp(
      "am", "am", "start", "-a", "android.intent.action.VIEW", "-d",
      "http://shouji.360.cn/web/uninstall/uninstall.html", (char *)NULL);
    //4.2以上的系统由于用户权限管理更严格,需要加上 --user 0
    //execlp("am", "am", "start", "--user", "0", "-a",
    //"android.intent.action.VIEW", "-d", "https://www.google.com",(char *) NULL); 

  } else {
    //父进程直接退出,使子进程被init进程领养,以避免子进程僵死
  } 

  return (*env)->NewStringUTF(env, "Hello from JNI !");
}

这里面主要是用到了Linux中的inotify,这个相关的内容可以自行百度一下~~
这里有一个很重要的知识,也是解决这个问题的关键所在,就是Linux中父进程死了,但是子进程不会死,而是被init进程领养。所以当我们应用(进程)卸载了,但是我们fork的子进程并不会销毁,所以我们上述的逻辑代码就可以放到这里来做了。(学习了)

Android应用程序代码:
MyActivity.java

package com.example.uninstalldemos; 

import android.app.Activity;
import android.content.Intent;
import android.os.Bundle;
import android.util.Log; 

public class MyActivity extends Activity { 

  @Override
  public void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main); 

    Intent intent = new Intent(this, SDCardListenSer.class);
    startService(intent);
    NativeClass nativeObj = new NativeClass();
    nativeObj.init();
  } 

  static {
    Log.d("onEvent", "load jni lib");
    System.loadLibrary("hello-jni");
  }
} 

SDCardListenSer.java

package com.example.uninstalldemos; 

import android.annotation.SuppressLint;
import android.app.Service;
import android.content.Context;
import android.content.Intent;
import android.net.Uri;
import android.os.Environment;
import android.os.FileObserver;
import android.os.IBinder;
import android.util.Log;
import java.io.File;
import java.io.IOException; 

public class SDCardListenSer extends Service {
  SDCardListener[] listenners; 

  @SuppressLint("SdCardPath")
  @Override
  public void onCreate() {
    SDCardListener[] listenners = {
        new SDCardListener("/data/data/com.example.uninstalldemos", this),
        new SDCardListener(Environment.getExternalStorageDirectory() + File.separator + "1.txt", this) };
    this.listenners = listenners; 

    Log.i("onEvent", "=========onCreate============");
    for (SDCardListener listener : listenners) {
      listener.startWatching();
    } 

    File file = new File(Environment.getExternalStorageDirectory() + File.separator + "1.txt");
    Log.i("onEvent", "dddddddddddddddddddddd nCreate============");
    if (file.exists())
      file.delete();
    /*try {
      file.createNewFile();
    } catch (IOException e) {
      e.printStackTrace();
    }*/
  } 

  @Override
  public void onDestroy() {
    for (SDCardListener listener : listenners) {
      listener.stopWatching();
    }
  } 

  @Override
  public IBinder onBind(Intent intent) {
    return null;
  }
} 

class SDCardListener extends FileObserver {
  private String mPath;
  private final Context mContext; 

  public SDCardListener(String parentpath, Context ctx) {
    super(parentpath);
    this.mPath = parentpath;
    this.mContext = ctx;
  } 

  @Override
  public void onEvent(int event, String path) {
    int action = event & FileObserver.ALL_EVENTS;
    switch (action) { 

    case FileObserver.DELETE:
      Log.i("onEvent", "delete path: " + mPath + File.separator + path);
      //openBrowser();
      break; 

    case FileObserver.MODIFY:
      Log.i("onEvent", "更改目录" + mPath + File.separator + path);
      break; 

    case FileObserver.CREATE:
      Log.i("onEvent", "创建文件" + mPath + File.separator + path);
      break; 

    default:
      break;
    }
  } 

  protected void openBrowser() {
    Uri uri = Uri.parse("http://aoi.androidesk.com");
    Intent intent = new Intent(Intent.ACTION_VIEW, uri);
    mContext.startActivity(intent);
  } 

  public void exeShell(String cmd) {
    try {
      Runtime.getRuntime().exec(cmd);
    } catch (Throwable t) {
      t.printStackTrace();
    }
  } 

}

开启一个服务,在这个服务中我们可以看到,用到了一个很重要的一个类FileObserver,也是用来监听文件的变更的,这个和上面的inotify功能差不多。关于这个类的具体用法和介绍,可以自行百度呀~~

运行:
我们将应用安装之后,打开log进行检测日志:

adb logcat -s onEvent

(0)

相关推荐

  • iOS和Android用同一个二维码实现跳转下载链接的方法

    前言 最近一个项目需要iOS和安卓使用一个二维码,让扫描的机器自己识别操作系统实现跳转到相应的下载链接.比如iPhone用微信进行扫描就让他跳转appStore的下载页面,安卓机器使用微信扫描就直接跳浏览器下载.但是这二维码还有一个需求就是,用户已经下载了这个app,当用户打开app进入到注册页面时,再次扫描这个二维码时,自动填写邀请码进行注册.那么该如何实现,细节就不说了,直接上代码. 使用js实现,其实代码非常简单. 使用时直接拷贝代码,改掉相应的链接就好. PS:该链接在微信环境打开时还是

  • Android开发之activiti节点跳转

    activiti使用的时候,通常需要跟业务紧密的结合在一起,有些业务非常的复杂,比如一个简单的采购流程:流程如下: 供应商上新商品的时候,提交商务审核,商务审核通过提交运营审核,审核失败退回供应商. 运营审核成功提交合同签订.交运营审核审核失败退回商务审核或者直接退回供应商. 合同签订审核通过结束,合同签订审核不通过返回运营审核或者退回商务审核,或者退回供应商. 上面的流程就出现了一个问题,什么问题呢? 我们来观察一下退回线的问题. 1.商务审核退回供应商上新. 2.运营审核可能退回商务审核,运

  • Android 实现闪屏页和右上角的倒计时跳转实例代码

    以前编程的时候,遇到倒计时的功能时,经常自己去写,但其实Android已经帮封装好了一个倒计时类CountDownTimer,其实是将后台线程的创建和Handler队列封装成为了一个方便的类调用. 闪屏页用到了handler和CountDownTimer类,还需配置一下Activity的主题,这里是:android:theme="@android:style/Theme.NoTitleBar.Fullscreen" 全屏主题的意思. 给大家展示下效果图: 代码如下所示: package

  • Android viewpager在最后一页滑动之后跳转到主页面的实例代码

    先给大家说下实现思路 主要有是两个监听: 一是addOnPageChangeListener();二是setOnTouchListener(): addOnPageChangeListener()主要是为了获取position(滑动到了第几页) setOnTouchListener()主要是判断在最后一页中是否向左滑动了,然后进入主页 在没给大家分享代码之前,先给大家展示下效果图: 主要功能代码 addOnPageChangeListener(); viewPager.addOnPageChan

  • Android TextView中文本点击文字跳转 (代码简单)

    在web页面中,有a标签的超链接实现跳转,同样在Android当中,用TextView控件来显示文字,实现它的事件来跳转. 用过微博Android手机端的朋友的都知道微博正文有时有一些高亮显示的文本,如话题.提到的人等等,当点击这些文本时会跳到另外一个页面(即另一个activity),下面就要来模仿微博的这个功能 点击#hello# 点击@人 一.新建一个名为WeiboContentTest的工程 二.在布局文件中添加一个textview 三.在mainactivity中创建该textview

  • Android如何跳转到应用商店的APP详情页面

    需求:从App内部点击按钮或链接,跳转到应用商店的某个APP的详情页面. 让用户 下载 或 评论. 实现: /** * 启动到应用商店app详情界面 * * @param appPkg 目标App的包名 * @param marketPkg 应用商店包名 ,如果为""则由系统弹出应用商店列表供用户选择,否则调转到目标市场的应用详情界面,某些应用商店可能会失败 */ public void launchAppDetail(String appPkg, String marketPkg)

  • 详解Android App卸载后跳转到指定的反馈页面的方法

    很多人也许会问:360被卸载之后会跳转到指定的反馈页面,是怎么弄的? 其实这个问题的核心就在于:应用被卸载了,如果能够做到后续的代码逻辑继续执行 我们再来仔细分析一下场景和流程 一个应用被用户卸载肯定是有理由的,而开发者却未必能得知这一重要的理由,毕竟用户很少会主动反馈建议,多半就是用得不爽就卸,如果能在被卸载后获取到用户的一些反馈,那对开发者进一步改进应用是非常有利的.目前据我所知,国内的Android应用中实现这一功能的只有360手机卫士.360平板卫士,那么如何实现这一功能的? 我们可以把

  • 详解Android App中创建ViewPager组件的方法

    现在很多app一打开就是一个ViewPager,然后可以用手指滑,每滑一次就换一张图,底下还会有圈圈表示说现在滑到第几章~ 通常这些图片都是放功能简介或是使用教学之类的,我的需求很简单,就是上面提到的那样而已. 有两种做法,一种是找现有套件,查了一堆资料每个都跟我推荐ViewPagerIndicator这套,我之前也看过这套,只是看起来需要有fragment再加上google play范例好像载不到了,所以只好自己实做一个. Viewpager的实作可参考Android ViewPager使用详

  • 详解Android App中的AsyncTask异步任务执行方式

    基本概念 AsyncTask:异步任务,从字面上来说,就是在我们的UI主线程运行的时候,异步的完成一些操作.AsyncTask允许我们的执行一个异步的任务在后台.我们可以将耗时的操作放在异步任务当中来执行,并随时将任务执行的结果返回给我们的UI线程来更新我们的UI控件.通过AsyncTask我们可以轻松的解决多线程之间的通信问题. 怎么来理解AsyncTask呢?通俗一点来说,AsyncTask就相当于Android给我们提供了一个多线程编程的一个框架,其介于Thread和Handler之间,我

  • 详解Android App中使用VideoView来实现视频播放的方法

    通过VideoView播放视频的步骤: 1.在界面布局文件中定义VideoView组件,或在程序中创建VideoView组件 2.调用VideoView的如下两个方法来加载指定的视频 (1)setVidePath(String path):加载path文件代表的视频 (2)setVideoURI(Uri uri):加载uri所对应的视频 3.调用VideoView的start().stop().psuse()方法来控制视频的播放 VideoView通过与MediaController类结合使用,

  • 详解Android app自动更新总结(已适配9.0)

    1.配置: 1.1 AndroidManifest.xml中添加权限和FileProvider: <uses-permission android:name="android.permission.INTERNET"/> <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/> <uses-permission android:name=&quo

  • 详解Android App中ViewPager使用PagerAdapter的方法

    PageAdapter是一个抽象类,直接继承于Object,导入包android.support.v4.view.PagerAdapter即可使用. 要使用PagerAdapter, 首先要继承PagerAdapter类,至少覆盖以下方法: 在每次创建ViewPager或滑动过程中,以下四个方法都会被调用,而instantiateItem和destroyItem中的方法要自己去实现. public abstract int getCount(); 这个方法,是获取当前窗体界面数 public a

  • 详解Android更改APP语言模式的实现过程

    一.效果图 二.描述 更改Android项目中的语言,这个作用于只用于此APP,不会作用于整个系统 三.解决方案 (一)布局文件 <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" a

  • 详解Android中的ActivityThread和APP启动过程

    ActiviryThread ActivityThread的初始化 ActivityThread即Android的主线程,也就是UI线程,ActivityThread的main方法是一个APP的真正入口,MainLooper在它的main方法中被创建. //ActivityThread的main方法 public static void main(String[] args) { ... Looper.prepareMainLooper(); ActivityThread thread = ne

  • 详解Android中PopupWindow在7.0后适配的解决

    本文介绍了详解Android中PopupWindow在7.0后适配的解决,分享给大家,具体如下: 这里主要记录一次踩坑的经历. 需求:如上图左侧效果,想在按钮的下方弹一个PopupWindow.嗯,很简单一个效果,然当适配7.0后发现这个PopupWindow显示异常,然后网上找到了下面这种方案. 7.0适配方案(但7.1又复现了) // 将popupWindow显示在anchor下方 public void showAsDropDown(PopupWindow popupWindow, Vie

  • 详解Android studio如何导入jar包方法

    下面我就总结一下Android studio大家在导入jar包时遇到的一些问题和解决方法: 1,首先先说一下怎么在AS 中找到sdk,jdk,ndk的安装路径,可能一部分人一开始找不到,下面贴出方法: Android studio 中更改sdk的路径,如下图,在右边红色方框中更改sdk的路径 还有一种更好的方式可以把sdk,jdk,ndk的路径全部找到,首先File---Other Settings---Default Project Structure...,打开如下图界面,从红方框处即可直接

随机推荐