由于项目开发需要用到手机联系人和通话记录的详细信息,于是对联系人和通话记录的表及各字段以及具体实现做了详细分析。
首先,通过模拟器或者Root过的手机,在/data/data/com.android.providers.contacts/database文件夹下得到contacts2.db数据库文件,然后用SQLite Expert软件打开,可以看到有如下表:
其中,与通话记录有关的表为:calls;与联系人有关的表主要有:contacts,data,mimetypes,raw_contacts,下面做详细分析。
1.通话记录
与通话记录有关的信息全部存储在独立的calls表中,没有其它关联的表,打开calls表,可以看到它一共有24个字段。
表对应的类为android.provider.CallLog.Calls,在Calls类中有查询所有通话记录的Uri(Uri CONTENT_URI =Uri.parse("content://call_log/calls"))和根据电话号码查询通讯记录的Uri(Uri CONTENT_FILTER_URI =Uri.parse("content://call_log/calls/filter"));虽然表中有24个字段,但是在Calls类中并没有表现出24个字段,下面为Calls类中的字段解释:
可以看到,这些字段就是可以从数据库中读取到的通话记录的数据。
那么查询这些信息的具体实现是怎样的呢?首先,通过内容提供者,得到Cursor:
ContentResolver cr = getContentResolver(); Cursor cursor = cr.query(CallLog.Calls.CONTENT_URI, null, null, null,CallLog.Calls.DEFAULT_SORT_ORDER);其中CallLog.Calls.DEFAULT_SORT_ORDER="date DESC",表示默认的时间排序方式,也就是从最近的一次通话记录开始读。
cursor.getInt(cursor.getColumnIndex(CallLog.Calls.TYPE))//获取通话类型:1.呼入2.呼出3.未接 cursor.getString(cursor.getColumnIndex(CallLog.Calls.CACHED_NAME))//获取联系人的名字 cursor.getString(cursor.getColumnIndex(CallLog.Calls.NUMBER))//获取联系人的电话号码 cursor.getString(cursor.getColumnIndex(CallLog.Calls.DATE))//获取通话日期 cursor.getString(cursor.getColumnIndex(CallLog.Calls.DURATION))//获取通话时长,值为多少秒 ......得到这些数据,还需要做一定的处理,比如通话时长的值为秒,但显示的时候通常以“00:00:00”显示,另外要想显示通话记录次数,需要获取到所有的数据后再做统计等等。下面提供一种时间显示处理方式:
private String formatDuration(long duration) { StringBuilder sb = new StringBuilder(); if (duration == 0) { sb.append("00:00"); } else if (duration > 0 && duration < 60) { sb.append("00:"); if (duration < 10) { sb.append("0"); } sb.append(duration); } else if (duration > 60 && duration < 3600) { long min = duration / 60; long sec = duration % 60; if (min < 10) { sb.append("0"); } sb.append(min); sb.append(":"); if (sec < 10) { sb.append("0"); } sb.append(sec); } else if (duration > 3600) { long hour = duration / 3600; long min = duration % 3600 / 60; long sec = duration % 3600 % 60; if (hour < 10) { sb.append("0"); } sb.append(hour); sb.append(":"); if (min < 10) { sb.append("0"); } sb.append(min); sb.append(":"); if (sec < 10) { sb.append("0"); } sb.append(sec); } return sb.toString(); }最后,这些数据都得到了,但是可能由于数据量特别大(可能有上千条数据)的时候,如果处理的很慢,会给用户带来不好的体验,在这儿只提供几种思路,不提供具体代码实现了。
1.通过AsyncQueryHandler(A helper class to help make handling asynchronousContentResovle
queries easier.)类,专门提供给数据库查询的类,只需要实现它的onQueryComplete方法,然后调用startQuery方法就开始执行,经过测试,效果还好,速度有一定的提升。
2.通过ResourceCursorAdapter( An easy adapter that creates views defined in an XML file. You can specify the XML file that defines the appearance of the views.)类,该类直接将Cursor联系起来,自动做一些数据处理,经过测试,效果很好,速度比用AsyncQueryHandler还快。在SDK的android-sdk-windows\samples\android-14\ApiDemos\src\com\example\android\apis\app中,有一个QuickContactsDemo.java Demo可以参考。
到此,通话记录的分析结束。下一篇,将对获取联系人信息进行剖析。