Android实现多次闪退清除数据

 更新时间:2016年04月11日 16:03:46   投稿:lijiao  
这篇文章主要介绍了Android实现多次闪退清除数据的相关资料,感兴趣的小伙伴们可以参考一下

很多时候由于后台返回的数据异常,可能会导致App闪退。而如果这些异常数据被App本地缓存下来,那么即使杀掉进程重新进入还是会发生闪退。唯一的解决方法就是清除App数据,但是用户可能没有这个意识或者嫌麻烦就直接不再使用了,这是我们无法接受的。在使用淘宝、追书神器等App时我发现有时候它们也会连续闪退,但是往往闪退三次后就恢复正常了,所以一般成熟的App都会做连续闪退三次后清除缓存数据的工作。而目前笔者搜不到有哪篇blog来讲这方面的事情,所以就姑且由我来讲讲此事,为希望提高App用户体验的朋友提供些许参考。

ACRA
为了能够在闪退的时候做一些事情,我们可以使用ACRA,这是Github上的一个开源项目,允许使用者设置一些Sender在App闪退的时候做一些事情。具体使用可以直接参考Github。如果不希望使用ACRA,那么也可以自己实现一个UncachedExceptionHandler并替换系统默认的Handler,并在这个Handler里面对数据进行处理。

实现清除数据
ACRA提供了自己的一些Sender,如使用系统邮件客户端向指定邮箱发送邮件的EmailIntentSender。而我们希望记录闪退次数和清除数据则需要implements ReportSender接口。

public class CrashHandler implements ReportSender {
  @Override
  public void send(Context context, CrashReportData errorContent) throws ReportSenderException {
    Timber.i("闪退,检查是否需要清空数据");
    new CrashModel().checkAndClearData();
  }
}

这里我们写了一个CrashModel用来记录闪退次数和时间决定是否需要清空数据,具体代码如下。 由于在ReportSender的时候无法打开其它线程,所以我们无法使用SharedPerferences来清理数据(打开SP的时候其实打开了一个新线程)。为此需要找到数据缓存的位置并将文件删除。同样道理,记录闪退时间也只能通过文件记录。当然,你可以选择一些文件不进行删除,如用户信息等不太容易出问题的数据。

public class CrashModel {

  private static final String KEY_CRASH_TIMES = "crash_times";
  private static final String CRASH_TIME_FILE_NAME = "crash_time";
  //不能通过App.getPackageName来获取包名,否则会有问题,只能默认为cn.campusapp.campus。所以对于debug或者运营版本,清数据会把release的清掉
  private static final String FILE_DIR = String.format("/data/data/%s/", BuildConfig.APPLICATION_ID);
  private static final String ACCOUNT_FILE_NAME = String.format("%s%s", FILE_DIR, "shared_prefs/account_pref.xml");
  private static ArrayList<String> FILES_DONTNEED_DELETE = new ArrayList<>(); //该目录中的文件不会被删除

  static {
    FILES_DONTNEED_DELETE.add(ACCOUNT_FILE_NAME); //目前账号信息文件不会被删除,但是会手动改变数据,只保留userId accessToken 和school
  }

  protected ArrayList<Long> mCrashTimes;
  Gson gson = new Gson();
  private File mFileDir;

  public CrashModel() {
    mFileDir = new File(FILE_DIR);
    mCrashTimes = readCrashTimes();
    if (mCrashTimes == null) {
      mCrashTimes = new ArrayList<>();
      storeCrashTimes(mCrashTimes);
    }
  }


  public void checkAndClearData() {
    long timeNow = System.currentTimeMillis();

    if (checkClearData(timeNow, new ArrayList<>(mCrashTimes))) {
      Timber.i("已经在5分钟之内有三次闪退,需要清理数据");
      try {
        clearData();
      } catch (Exception e) {
        Timber.e(e, "清空所有数据失败");
      }
    } else {
      mCrashTimes.add(timeNow);
      storeCrashTimes(mCrashTimes);
      Timber.i("此次不需要清空数据, %s", gson.toJson(mCrashTimes));
    }
  }

  private void storeCrashTimes(ArrayList<Long> crashTimes) {
    try {
      String str = gson.toJson(crashTimes);
      Files.writeToFile(mFileDir, CRASH_TIME_FILE_NAME, str);
    } catch (Exception e) {
      Timber.e(e, "保存闪退时间失败");
    }

  }

  private ArrayList<Long> readCrashTimes() {
    try {
      String timeStr = Files.readFileContent(mFileDir, CRASH_TIME_FILE_NAME);
      return gson.fromJson(timeStr, new TypeToken<ArrayList<Long>>() {
      }.getType());
    } catch (Exception e) {
      Timber.e(e, "读取闪退时间失败");
    }
    return null;
  }

  /**
   * 检查是否需要清空数据,目前的清空策略是在5分钟之内有三次闪退的就清空数据,也就是从后往前遍历,只要前两次闪退发生在5分钟之内,就清空数据
   *
   * @return
   */
  private boolean checkClearData(long time, ArrayList<Long> crashTimes) {
    Timber.i(gson.toJson(crashTimes));
    int count = 0;
    for (int i = crashTimes.size() - 1; i >= 0; i--) {
      long crashTime = crashTimes.get(i);
      if (time - crashTime <= 5 * 60 * 1000) {
        count++;
        if (count >= 2) {
          break;
        }
      }
    }
    if (count >= 2) {
      //在5分钟之内有三次闪退,这时候需要清空数据
      return true;
    } else {
      return false;
    }
  }

  /**
   * 清空数据,包括数据库中的和SharedPreferences中的
   *
   * @throws Exception
   */
  private void clearData() throws Exception {
    Timber.i("开始清理数据");
    Files.deleteFilesExceptSomeInDirectory(mFileDir, FILES_DONTNEED_DELETE);
  }


}

然后我们需要将CrashHandler 添加到ACRA的异常处理Sender列表中。在你的Application类中添加如下代码。

@ReportsCrashes(
  //一些ACRA的设置,具体参考ACRA文档,因为我们使用自定义Sender,所以这里完全可以不用设置
    //mailTo = "bugs@treeholeapp.cn",
    //mode = ReportingInteractionMode.TOAST,
    //resToastText = R.string.crash_toast_text
)
public class App extends Application {

  @Override
  public void onCreate() {
   if (!BuildConfig.DEBUG) { //这里我判断只有在非DEBUG下才清除数据,主要是为了在开发过程中能够保留线程。
        ACRA.init(APPLICATION_CONTEXT);
        CrashHandler handler = new CrashHandler();
        ACRA.getErrorReporter().setReportSender(handler); //在闪退时检查是否要清空数据
    }

  }
}

总结
以上即为实现多次闪退后清除数据的实现,希望大家开发的App Bug越来越少。

相关文章

  • Android网络编程之简易新闻客户端

    Android网络编程之简易新闻客户端

    这篇文章主要为大家详细介绍了Android网络编程之简易新闻客户端的相关资料,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-05-05
  • 一行代码教你解决Scrollview和TextInput焦点获取问题

    一行代码教你解决Scrollview和TextInput焦点获取问题

    这篇文章主要为大家介绍了一行代码教你解决Scrollview和TextInput焦点获取问题,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-12-12
  • android TextView加下划线的方法

    android TextView加下划线的方法

    这篇文章主要介绍了android TextView加下划线的方法,大家可以在项目代码里试试
    2013-11-11
  • 获取Android系统唯一识别码的方法

    获取Android系统唯一识别码的方法

    这篇文章主要介绍了获取Android系统唯一识别码的方法,涉及通过编程获取Android系统硬件设备标识的相关技巧,具有一定参考借鉴价值,需要的朋友可以参考下
    2015-10-10
  • Android UI自定义ListView实现下拉刷新和加载更多效果

    Android UI自定义ListView实现下拉刷新和加载更多效果

    这篇文章主要介绍了Android UI自定义ListView实现下拉刷新和加载更多效果,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2016-11-11
  • Android入门教程之RecyclerView的具体使用详解

    Android入门教程之RecyclerView的具体使用详解

    RecyclerView是Android一个更强大的控件,其不仅可以实现和ListView同样的效果,还有优化了ListView中的各种不足。其可以实现数据纵向滚动,也可以实现横向滚动(ListView做不到横向滚动)。接下来讲解RecyclerView的用法
    2021-10-10
  • android中创建通知栏Notification代码实例

    android中创建通知栏Notification代码实例

    这篇文章主要介绍了android中创建通知栏Notification代码实例,本文直接给出实现代码,需要的朋友可以参考下
    2015-05-05
  • Android使用Notification实现通知功能

    Android使用Notification实现通知功能

    这篇文章主要为大家详细介绍了Android使用Notification实现通知功能,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-11-11
  • Android简单实现无限滚动自动滚动的ViewPager

    Android简单实现无限滚动自动滚动的ViewPager

    这篇文章主要介绍了Android简单实现无限滚动自动滚动的ViewPager,百度谷歌上面也有很多关于这方面的教程,但是感觉都略显麻烦,而且封装的都不是很彻底。所以试着封装一个比较好用的ViewPager,实现思路一起通过本文学习吧
    2016-12-12
  • Kotlin try catch异常处理i详解

    Kotlin try catch异常处理i详解

    在 Kotlin 协程当中,我们通常把异常分为两大类,一类是取消异常(CancellationException),另一类是其他异常。在 Kotlin 协程当中,这两种异常的处理方式是不一样的
    2022-11-11

最新评论