美文网首页安卓开发博客
线程和线程池 -- Loader

线程和线程池 -- Loader

作者: TomyZhang | 来源:发表于2018-07-22 17:48 被阅读2次

一、概念

在Android 3.0 中引入了 Loader 机制,让开发者能轻松在 Activity 和 Fragment 中异步加载数据,Loader 机制具有以下特征:
● 可用于每个 Activity 或 Fragment。
● 支持异步加载数据。
● 监控数据源并在内容变化时传递新结果。
● 在某一配置更改后重建加载器时,会自动重新连接上一个加载器的游标。因此,它们无需重新查询其数据。

二、常用类

Loader:
一种执行异步加载数据的抽象类,这是加载器的基类。我们通常会使用 CursorLoader,但也可以实现自己的子类。当加载器处于活动状态时,应监控其数据源并在内容变化时传递新结果。

AsyncTaskLoader:
提供 AsyncTask 来执行工作的抽象加载器。

CursorLoader:
AsyncTaskLoader 的子类,它将查询 ContentResolver 并返回一个 Cursor。使用此加载器是从 ContentProvider 异步加载数据的最佳方式,而不用通过 Activity 或 Fragment 的 API 来执行托管查询。

LoaderManager:
一种与 Activity 或 Fragment 相关联的抽象类,用于管理一个或多个 Loader 实例。这有助于应用管理与 Activity 或 Fragment 生命周期相关的、运行时间较长的操作。它常见的用法是 与 CursorLoader 一起使用,不过应用也可以自由写入自己的加载器,用于加载其他类型的数据。

LoaderManager.LoaderCallbacks:
回调接口,用于客户端与 LoaderManager 进行交互,主要有如下三个方法:
onCreateLoader():初始化并返回一个新的Loader实例。
onLoadFinished():当一个加载器完成加载过程之后会回调这个方法。
onLoaderReset():当一个加载器被重置并且数据无效时会回调这个方法。

三、使用

public class LoaderActivity extends AppCompatActivity {

   @BindView(R.id.listView)
   ListView listView;

   private LoaderManager loaderManager;
   private ArrayAdapter<String> arrayAdapter ;
   private List<String> contactList = new ArrayList<>();

   @Override
   protected void onCreate(@Nullable Bundle savedInstanceState) {
       super.onCreate(savedInstanceState);
       setContentView(R.layout.activity_loader);
       ButterKnife.bind(this);

       arrayAdapter = new ArrayAdapter<String>(this,android.R.layout.simple_list_item_1,contactList);
       listView.setAdapter(arrayAdapter);
   
       loaderManager = getSupportLoaderManager();

       //动态申请权限
       if (ContextCompat.checkSelfPermission(this,
               Manifest.permission.READ_CONTACTS)
               != PackageManager.PERMISSION_GRANTED) {
           ActivityCompat.requestPermissions(this,
                   new String[]{Manifest.permission.READ_CONTACTS},
                   0);
       } else {
           loaderManager.initLoader(0, null, mLoaderCallback);
       }

   }

   //callBack 实现
   private LoaderManager.LoaderCallbacks<Cursor> mLoaderCallback = new LoaderManager.LoaderCallbacks<Cursor>() {

       private final String[] Contact_PROJECTION = {
               ContactsContract.CommonDataKinds.Phone.DISPLAY_NAME,
               ContactsContract.CommonDataKinds.Phone.NUMBER};

       @Override
       public Loader<Cursor> onCreateLoader(int id, Bundle args) {
           CursorLoader cursorLoader =
                   new CursorLoader(LoaderActivity.this,
                           ContactsContract.CommonDataKinds.Phone.CONTENT_URI, Contact_PROJECTION,
                           null, null, null);
           return cursorLoader;
       }

       @Override
       public void onLoadFinished(Loader<Cursor> loader, Cursor data) {

           //首先要清空数据源,避免重复数据
           contactList.clear();
           while (data.moveToNext()) {
               String displayName = data.getString(data.getColumnIndex(ContactsContract.CommonDataKinds.Phone.DISPLAY_NAME));
               String number = data.getString(data.getColumnIndex(ContactsContract.CommonDataKinds.Phone.NUMBER));
               //分两行展示
               contactList.add(displayName + "\n" + number);
           }
           arrayAdapter.notifyDataSetChanged();
       }

       @Override
       public void onLoaderReset(Loader<Cursor> loader) {
           contactList.clear();
       }
   };

   @Override
   protected void onResume() {
       super.onResume();
       //loaderManager.initLoader(0,null,mLoaderCallback);
   }

   @Override
   protected void onDestroy() {
       super.onDestroy();
       loaderManager.destroyLoader(0);
   }

   @Override
   protected void onSaveInstanceState(Bundle outState) {
       super.onSaveInstanceState(outState);
   }
   
   @Override
   public void onRequestPermissionsResult(int requestCode, @NonNull String[] permissions, @NonNull int[] grantResults) {
       super.onRequestPermissionsResult(requestCode, permissions, grantResults);
       if(requestCode == 0){
           if(grantResults[0] == PackageManager.PERMISSION_GRANTED){
               loaderManager.initLoader(0, null, mLoaderCallback);
           }
       }
   }
}

相关文章

  • 线程和线程池 -- Loader

    一、概念 在Android 3.0 中引入了 Loader 机制,让开发者能轻松在 Activity 和 Frag...

  • Springboot | 线程池的学习,多线程池配置示例

    一、线程和进程,线程的生命周期二、单线程和多线程三、线程池的概念四、线程池的使用五、多线程池配置示例 一、线程和进...

  • 线程池查漏补缺

    tomcat线程池和jdk线程池区别 概述 线程池是什么,为什么要线程池 jdk有哪些线程池和原理 第三方中间件的...

  • 线程池概述

    为什么要使用线程池? 线程池核心参数 线程池的几种拒绝策略 execute()和submit()的区别 线程池工作...

  • java线程池

    线程VS线程池 普通线程使用 创建线程池 执行任务 执行完毕,释放线程对象 线程池 创建线程池 拿线程池线程去执行...

  • 第九章 线程池

    线程池的实现原理和使用建议。 当提交一个新任务到线程池时,线程池的处理流程如下。1)线程池判断核心线程池里的线程是...

  • 我们的线程被饿死了

    我们的线程被饿死了 我们在构建线程池的时候可以构建单个线程的线程池和多个线程的线程池。 那么线程池使用不当可不可能...

  • Android线程池的使用

    一、线程与线程池,为什么要使用线程池 1、Android中的线程 在Android中有主线程和子线程的区分。主线程...

  • Java线程池的使用

    线程类型: 固定线程 cached线程 定时线程 固定线程池使用 cache线程池使用 定时调度线程池使用

  • Android面试Java基础篇(二)

    (一)问:Java线程池的实现原理和使用 线程池即存放和管理线程的一个池子 (1)复用线程池中的线程,避免因为线程...

网友评论

    本文标题:线程和线程池 -- Loader

    本文链接:https://www.haomeiwen.com/subject/vmfrmftx.html