Android Loader 异步加载详解二:探寻Loader内部机制

本文涉及的产品
日志服务 SLS,月写入数据量 50GB 1个月
简介: Android Loader 异步加载详解二:探寻Loader内部机制 转载请标明出处:http://blog.csdn.net/zhaoyanjun6/article/details/70259914 本文出自【赵彦军的博客】Android Loader 异步加载详解一:基础概念 Android Loader 异步加载详解二:探寻Loader内部机制在上一篇文章中,讲解了 Loader 的基本概念。

Android Loader 异步加载详解二:探寻Loader内部机制

转载请标明出处:http://blog.csdn.net/zhaoyanjun6/article/details/70259914
本文出自【赵彦军的博客】

Android Loader 异步加载详解一:基础概念
Android Loader 异步加载详解二:探寻Loader内部机制

在上一篇文章中,讲解了 Loader 的基本概念。这一篇将会用实战的方式来探寻 Android Loader的内部机制。我们准备做一个 读取手机短信的例子。废话不多说,直接上效果图:

这里写图片描述

所有的代码都在:https://github.com/zyj1609wz/Loader

实例源码

  • 首先 SmsActivity 的源码
package com.app.loader.sms;

import android.app.LoaderManager;
import android.content.Loader;
import android.database.Cursor;
import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.util.Log;
import android.view.View;
import android.widget.EditText;
import android.widget.ListView;
import android.widget.SimpleCursorAdapter;
import com.app.loader.R;

public class SmsActivity extends AppCompatActivity {

    private int loaderId = 0 ;
    private ListView lv;
    private SimpleCursorAdapter adapter;
    private LoaderManager.LoaderCallbacks loaderCallbacks ;

    private EditText editText ;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_sms );
        Log.e( "loader", "activity  onCreate: ");
        lv = (ListView) findViewById( R.id.listview );

        adapter = new SimpleCursorAdapter(this,
                R.layout.sms_listview_item ,
                null,
                new String[]{"address","body"},
                new int[]{R.id.address, R.id.body},
                SimpleCursorAdapter.FLAG_REGISTER_CONTENT_OBSERVER);

        lv.setAdapter(adapter);

        loaderCallbacks = new MyCallback() ;

        //初始化,并且创建Loader 实例,并且开始执行
        getLoaderManager().initLoader( loaderId , null,  loaderCallbacks );

        editText = (EditText) findViewById( R.id.editText );
        findViewById( R.id.start).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                //开始查询
                String tag = editText.getText().toString() ;
                Bundle bundle = new Bundle();
                bundle.putString("key", tag );
                getLoaderManager().restartLoader( loaderId , bundle, loaderCallbacks  );
            }
        });

        findViewById( R.id.init).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                //初始化,并且创建Loader 实例,并且开始执行
                getLoaderManager().initLoader( loaderId , null,  loaderCallbacks );
            }
        });

    }

    class MyCallback implements LoaderManager.LoaderCallbacks<Cursor> {

        @Override
        public Loader onCreateLoader(int id, Bundle args) {
            SmsLoader loader = new SmsLoader( SmsActivity.this , args);
            Log.e( "loader", "onCreateLoader: ");
            return loader ;
        }

        @Override
        public void onLoadFinished(Loader<Cursor> loader, Cursor data) {
            Log.e( "loader", "onLoadFinished: ");
            adapter.changeCursor( data );
        }

        @Override
        public void onLoaderReset(Loader loader) {
            Log.e( "loader", "onLoaderReset: ");
            //当 Activity OnDestory() , 系统会回调这个方法
            adapter.swapCursor(null);
        }
    }

    @Override
    protected void onResume() {
        Log.e( "loader", "activity onResume: ");
        super.onResume();
    }

    @Override
    protected void onPause() {
        Log.e( "loader", "activity onPause: ");
        super.onPause();
    }

    @Override
    protected void onDestroy() {
        Log.e( "loader", "activity onDestroy: ");
        super.onDestroy();
    }
}

  • activity_sms 布局
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    tools:context="com.app.loader.MainActivity">

    <EditText
        android:id="@+id/editText"
        android:layout_width="match_parent"
        android:layout_height="50dp"
        android:hint="请你输入过滤关键词"
        />

    <Button
        android:id="@+id/start"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="开始查询"
        />

    <Button
        android:id="@+id/init"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:text="开始执行 init方法"
        />

    <ListView
        android:id="@+id/listview"
        android:layout_width="match_parent"
        android:layout_height="match_parent">
    </ListView>

</LinearLayout>
  • SmsLoader 类继承 AsyncTaskLoader
package com.app.loader.sms;

import android.content.AsyncTaskLoader;
import android.content.Context;
import android.database.Cursor;
import android.net.Uri;
import android.os.Bundle;
import android.util.Log;

/**
 * Created by ${zhaoyanjun} on 2017/4/19.
 */

public class SmsLoader extends AsyncTaskLoader<Cursor> {

    private Bundle bundle;
    private Uri uri = Uri.parse("content://sms");
    private String []colums = {"_id","address","body"};

    public SmsLoader(Context context , Bundle bundle ) {
        super(context);
        this.bundle = bundle;
    }

    @Override
    public Cursor loadInBackground() {
        String selection = null;
        String[] selectionArgs = null;
        if (bundle!=null) {
            selection = "body like ? ";
            selectionArgs = new String[]{"%"+bundle.getString("key")+"%"};
        }

        Cursor cursor = getContext().getContentResolver().query(uri, colums, selection  , selectionArgs, null);

        Log.e( "loader", "loadInBackground: ");
        return cursor;
    }

    @Override
    protected void onStartLoading() {
        super.onStartLoading();
        forceLoad();
        Log.e( "loader", "onStartLoading: ");
        //这里一定要执行 forceLoad(); 否则 Loader 不会正常工作
        // onStartLoading()  --> forceLoad --> 调用 AsyncTaskLoader 里面的 forceLoad()
        // --> 开始创建 AsyncTask 对象实例,并且运行 AsyncTask 的 doInBackground --> SmsLoader 类中 loadInBackground()
        // --> 开始把结果 回调给主线程 AsyncTask onPostExecute()  -->  AsyncTaskLoader  dispatchOnLoadComplete()
        // -- AsyncTaskLoader deliverResult() 把最后的结果回调给 LoaderManager.LoaderCallbacks

    }
}

测试

  • 第一次 Activity 启动 Log 日志分析 :
com.app.loader E/loader: activity  onCreate: 
com.app.loader E/loader: onCreateLoader: 
com.app.loader E/loader: onStartLoading: 
com.app.loader E/loader: loadInBackground: 
com.app.loader E/loader: onLoadFinished: 

可以看到 第一次调用 initLoader() 方法后

 //初始化,并且创建Loader 实例,并且开始执行
 getLoaderManager().initLoader( loaderId , null,  loaderCallbacks );

首先 调用 LoaderCallbacks 中的 onCreateLoader 来创建一个 Loader 对象,然后调用 SmsLoader 中的 onStartLoading方法。然后调用 SmsLoader 中的 loadInBackground 开始执行 异步任务。最后在 LoaderCallbacksonLoadFinished 方法中回调。

  • 点击 “开始执行 Init 方法” 按钮,Log 分析
    点击 “开始执行 Init 方法”按钮后,开始执行
getLoaderManager().initLoader( loaderId , null,  loaderCallbacks );

Log 日志为:

com.app.loader E/loader: onLoadFinished: 

这里只回调了 onLoadFinished 的方法,把异步操作产生的数据给传递出来。请注意,这里没有走 loadInBackground 方法,说明此时 onLoadFinished 回传的数据,是旧数据,也就是上一次异步产生的数据。

但是有时我们想丢弃旧数据然后重新开始创建一个新Loader,这可怎么办呢?别担心,要丢弃旧数据调用restartLoader()即可。

  • restartLoader 方法探究。

点击 开始查询按钮 ,会执行下面的代码

//开始查询
String tag = editText.getText().toString() ;
Bundle bundle = new Bundle();
bundle.putString("key", tag );
getLoaderManager().restartLoader( loaderId , bundle, loaderCallbacks  );

Log 日志:

com.app.loader E/loader: onCreateLoader: 
com.app.loader E/loader: onStartLoading: 
com.app.loader E/loader: loadInBackground: 
com.app.loader E/loader: onLoadFinished: 

通过日志可以看出 restartLoader重新执行了 onCreateLoader 创建了一个新的 Loader 对象; loadInBackground 丢弃了旧数据,重新加载了新数据 , 并且回调 onLoadFinished

  • 当前 Activity 从后台到前台

在测试的时候,我发现当前 Activity 从后台到前台的时候,调用顺序如下:

com.app.loader E/loader: onStartLoading: 
com.app.loader E/loader: activity onResume:  
com.app.loader E/loader: loadInBackground: 
com.app.loader E/loader: onLoadFinished: 

可以看到当前界面从后台到前台的过程中,Loader 会自动调异步任务,并且回调新的数据。

  • 当前 Activity 销毁

当前 Activity 销毁的时候,调用顺序如下:

com.app.loader E/loader: activity onPause: 
com.app.loader E/loader: activity onDestroy: 
com.app.loader E/loader: onLoaderReset: 

可以看到 LoaderCallbacksonLoaderReset 方法会回调。

onLoaderReset 方法被调用的时候,代表 这个 Loader 正在被重置,此时的数据不可用。应用程序应该在这一点上删除对Loader数据的任何引用。

比如:

  @Override
   public void onLoaderReset(Loader loader) {
      Log.e( "loader", "onLoaderReset: ");
      //当 Activity OnDestory() , 系统会回调这个方法
      adapter.swapCursor(null);
  }

参考文档:

官方文档

使用CursorLoader执行查询任务

Android应用Loaders全面详解及源码浅析

Android之Loader介绍

相关实践学习
日志服务之使用Nginx模式采集日志
本文介绍如何通过日志服务控制台创建Nginx模式的Logtail配置快速采集Nginx日志并进行多维度分析。
相关文章
|
2月前
|
存储 安全 Android开发
探索Android与iOS的隐私保护机制
在数字化时代,移动设备已成为我们生活的一部分,而隐私安全是用户最为关注的问题之一。本文将深入探讨Android和iOS两大主流操作系统在隐私保护方面的策略和实现方式,分析它们各自的优势和不足,以及如何更好地保护用户的隐私。
|
2月前
|
Linux Android开发 iOS开发
深入探索Android与iOS的多任务处理机制
在移动操作系统领域,Android和iOS各有千秋,尤其在多任务处理上展现出不同的设计理念和技术实现。本文将深入剖析两大平台在后台管理、资源分配及用户体验方面的策略差异,揭示它们如何平衡性能与电池寿命,为用户带来流畅而高效的操作体验。通过对比分析,我们不仅能够更好地理解各自系统的工作机制,还能为开发者优化应用提供参考。
|
2月前
|
算法 Linux 调度
深入探索安卓系统的多任务处理机制
【10月更文挑战第21天】 本文旨在为读者提供一个关于Android系统多任务处理机制的全面解析。我们将从Android操作系统的核心架构出发,探讨其如何管理多个应用程序的同时运行,包括进程调度、内存管理和电量优化等方面。通过深入分析,本文揭示了Android在处理多任务时所面临的挑战以及它如何通过创新的解决方案来提高用户体验和设备性能。
65 1
|
3月前
|
消息中间件 存储 Java
Android面试高频知识点(2) 详解Android消息处理机制(Handler)
Android面试高频知识点(2) 详解Android消息处理机制(Handler)
|
3月前
|
消息中间件 存储 Java
Android面试高频知识点(2) 详解Android消息处理机制(Handler)
Android面试高频知识点(2) 详解Android消息处理机制(Handler)
64 1
|
3月前
|
存储 安全 数据安全/隐私保护
探索安卓与iOS的隐私保护机制####
【10月更文挑战第15天】 本文深入剖析了安卓和iOS两大操作系统在隐私保护方面的策略与技术实现,旨在揭示两者如何通过不同的技术手段来保障用户数据的安全与隐私。文章将逐一探讨各自的隐私控制功能、加密措施以及用户权限管理,为读者提供一个全面而深入的理解。 ####
114 1
|
3月前
|
消息中间件 存储 Java
Android消息处理机制(Handler+Looper+Message+MessageQueue)
Android消息处理机制(Handler+Looper+Message+MessageQueue)
84 2
|
缓存 Java 调度
【Android 异步操作】线程池 ( 线程池作用 | 线程池种类 | 线程池工作机制 | 线程池任务调度源码解析 )
【Android 异步操作】线程池 ( 线程池作用 | 线程池种类 | 线程池工作机制 | 线程池任务调度源码解析 )
144 0
|
1月前
|
搜索推荐 前端开发 API
探索安卓开发中的自定义视图:打造个性化用户界面
在安卓应用开发的广阔天地中,自定义视图是一块神奇的画布,让开发者能够突破标准控件的限制,绘制出独一无二的用户界面。本文将带你走进自定义视图的世界,从基础概念到实战技巧,逐步揭示如何在安卓平台上创建和运用自定义视图来提升用户体验。无论你是初学者还是有一定经验的开发者,这篇文章都将为你打开新的视野,让你的应用在众多同质化产品中脱颖而出。
57 19
|
1月前
|
JSON Java API
探索安卓开发:打造你的首个天气应用
在这篇技术指南中,我们将一起潜入安卓开发的海洋,学习如何从零开始构建一个简单的天气应用。通过这个实践项目,你将掌握安卓开发的核心概念、界面设计、网络编程以及数据解析等技能。无论你是初学者还是有一定基础的开发者,这篇文章都将为你提供一个清晰的路线图和实用的代码示例,帮助你在安卓开发的道路上迈出坚实的一步。让我们一起开始这段旅程,打造属于你自己的第一个安卓应用吧!
64 14