如何提高arraylist的提取速度?

时间:2019-02-13 05:50:21

标签: java android listview arraylist android-contacts

我正在使用Arraylist来获取应用程序中所有可用的联系人。这是无效的,因为Arraylist需要花费很长时间来获取和填充Listview,因为几乎有600+ contacts

我正在寻找一种性能更好的替代方法。

尽管我搜索了其他相关问题,但找不到方便的问题。

这是我的Java代码:

private List<String> getContactList() {
      List<String> stringList=new ArrayList<>();
      ContentResolver cr = context.getContentResolver();
      Cursor cur = cr.query(ContactsContract.Contacts.CONTENT_URI,
        null, null, null, null);

      if ((cur != null ? cur.getCount() : 0) > 0) {
        while (cur != null && cur.moveToNext()) {
          String id = cur.getString(
            cur.getColumnIndex(ContactsContract.Contacts._ID));
            String name = cur.getString(cur.getColumnIndex(
            ContactsContract.Contacts.DISPLAY_NAME)
          );

          if (cur.getInt(cur.getColumnIndex(
            ContactsContract.Contacts.HAS_PHONE_NUMBER)) > 0) {
              Cursor pCur = cr.query(                  
                ContactsContract.CommonDataKinds.Phone.CONTENT_URI,
                null,
                ContactsContract.CommonDataKinds.Phone.CONTACT_ID + " = ?",
                  new String[]{id}, null
               );

               while (pCur.moveToNext()) {
                 String phoneNo = pCur.getString(pCur.getColumnIndex(
                 ContactsContract.CommonDataKinds.Phone.NUMBER));                  
                 Log.v("Data : ",""+id+" "+name+" "+phoneNo);
                 stringList.add(id);
                 stringList.add(name);
                 stringList.add(phoneNo);
               }
               pCur.close();
             }
            }
          }
          if(cur!=null){
            cur.close();
          }
          return stringList;
        }   

3 个答案:

答案 0 :(得分:1)

您可以考虑使用Paging库:https://developer.android.com/topic/libraries/architecture/paging/

它的设计思想是,列表仅显示一定数量的项目,因此,实际上不需要以超出其可能显示的方式加载方式。例如,一个ListView可能只显示10个联系人,因此不需要提取600个联系人。

相反,分页库将在用户滚动时获取较小的数量,从而消除了600个联系人的加载时间,600个联系人的内存等,从而提高了效率。

答案 1 :(得分:0)

如果您担心速度,我会尝试使用Set,尽管ArrayList中有600多个联系人应该没问题。当数据集数以百万计甚至更多时,这将成为一个问题。我会尝试查看您代码中的其他效率低下的情况。

就Set而言,两个最常见的Java数据结构是HashSet和TreeSet。如果要订购集合,请使用TreeSet。 HashSet快一点,但是您会失去订购的机会。两者都有O(1)访问时间。

答案 2 :(得分:0)

您的查询效率低下,您当前正在对每个联系人进行查询,这非常慢,您可以通过一个大型查询(很快)来获得全部信息:

String[] projection = new String[] { Phone.CONTACT_ID, Phone.DISPLAY_NAME, Phone.NUMBER };
Cursor c = cr.query(Phone.CONTENT_URI, projection, null, null, null);
while (c.moveToNext()) {
   long contactId = c.getLong(0);
   String name = c.getString(1);
   String phone = c.getString(2);
   Log.i("Phones", "got contact phone: " + contactId + " - " + name + " - " + phone);
}
c.close();