软件编程
位置:首页>> 软件编程>> Android编程>> Android Loader的使用以及手机通讯录的获取方法

Android Loader的使用以及手机通讯录的获取方法

作者:大鹏1987  发布时间:2023-12-22 02:10:05 

标签:Android,Loader,手机通讯录

Android的主线程中执行长时间操作,导致界面无响应,会引起ANR。如果需要执行较长时间的操作,一般会在另一个线程处理,然后将数据转交给主线程进行显示,Android本身为我们提供了一些机制处理这种情况,今天就来看看Loader。Loader主要用来在Activity和Fragment中异步加载数据,使用也非常简单。

Loader的初始化非常简单,Activity提供了获取LoaderManager的接口,之后调用LoaderManager的initLoader即可。initLoader接受三个参数:

Loader的唯一标识符ID,用来区分多个Loader;

传递给Loader的参数,可选;

Loader的回调。

销毁Loader,只需要使用LoaderManager的destoryLoader即可,参数只传递一个Loader的ID。

Loader的回调接口LoaderCallbacks有三个方法:


public interface LoaderCallbacks<D> {
Loader<D> onCreateLoader(int var1, Bundle var2);

void onLoadFinished(Loader<D> var1, D var2);

void onLoaderReset(Loader<D> var1);
}

onCreateLoader在Loader创建时被调用;

onLoadFinished在Loader加载数据完成时调用;

onLoaderReset在Loader被reset时被调用。

Loader接口就这些,我们通过一个实际的应用案例看看怎么使用Loader,这里选择了手机通信录的获取,获取手机的通信录信息,需要使用Phone的ContentProvider,下面例子中详细来看。

由于要访问通信录,需要在manifest文件添加权限:


<uses-permission android:name="android.permission.READ_CONTACTS"/>
<uses-permission android:name="android.permission.CALL_PHONE"/>

activity_main.xml非常的简单,就包括一个ListView,用来显示通讯录列表。


<?xml version="1.0" encoding="utf-8"?>
<FrameLayout 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"
 tools:context="com.example.yjp.contractgetter.MainActivity">

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

</FrameLayout>

MainActivity类的代码比较长,我们分解来看,MainActivity要实现LoaderCallbacks接口


class MainActivity : AppCompatActivity(), LoaderManager.LoaderCallbacks<Cursor> {

使用Kotlin的伙伴对象,定义了静态成员变量,LOADER_ID是Loader的ID,PHONE_PROJECTION是查询通讯录时希望查找的列集合


companion object {
 private val LOADER_ID = 0
 private val PHONE_PROJECTION = arrayOf(Phone._ID, Phone.DISPLAY_NAME, Phone.NUMBER)
}

onCreate使用SimpleCursorAdapter作为ListView的Adapter,然后调用initLoader初始化了Loader


private var mAdapter:SimpleCursorAdapter? = null

override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
setContentView(R.layout.activity_main)

mAdapter = SimpleCursorAdapter(this,
android.R.layout.simple_list_item_2,
null,
arrayOf(Phone.DISPLAY_NAME, Phone.NUMBER),
intArrayOf(android.R.id.text1, android.R.id.text2),
0)
listView.adapter = mAdapter
listView.onItemClickListener = AdapterView.OnItemClickListener {
_, _, position, _ ->
val cursor = listView.getItemAtPosition(position) as Cursor
val displayNameIndex = cursor.getColumnIndex(Phone.DISPLAY_NAME)
Toast.makeText(this, cursor.getString(displayNameIndex), Toast.LENGTH_SHORT).show()
}

loaderManager.initLoader(LOADER_ID, null, this)
}

onDestory销毁Loader


override fun onDestroy() {
 super.onDestroy()
 loaderManager.destroyLoader(LOADER_ID)
}

三个回调方法:


override fun onCreateLoader(id: Int, bundle: Bundle?): Loader<Cursor> {
return CursorLoader(this,
Phone.CONTENT_URI,
PHONE_PROJECTION,
null,
null,
Phone.DISPLAY_NAME)
}

override fun onLoaderReset(cursor: Loader<Cursor>?) {
mAdapter?.swapCursor(null)
}

override fun onLoadFinished(loader: Loader<Cursor>?, cursor: Cursor?) {
mAdapter?.swapCursor(cursor)
}

onCreateLoader创建了一个CursorLoader,该Loader回执行ContentProvider的操作,然后返回一个cursor;

onLoaderReset时,将Adapter中的cursor置null;

onLoadFinished时,由于CursorLoader查询完数据,会返回新的cursor,我们使用新的Cursor去替换之前Adapter中的cursor。

这样,我们就可以在Activity启动时自动异步加载数据了,手机上试试,可以发现,加载非常顺畅。github已上传代码。

来源:http://blog.csdn.net/yjp19871013/article/details/78940657

0
投稿

猜你喜欢

手机版 软件编程 asp之家 www.aspxhome.com