Android软件开发之获取通讯录联系人信息 + android联系人信息的存储结构 + Android联系人读取操作笔记


图中选中的数据库 contacts2.db就是系统储存联系人的数据库,我们将它打开看看里面储存了些什么东东? 如果对数据库不太清楚的请查看我的博文Android游戏开发之数据库SQLite 详细介绍(十七)


       

(snwrking: DB的路径是 :/data/data/com.android.providers.contacts/databases/contacts2.db


打开contacts.db后 发面里面有一堆表,同学们先别慌张。今天我们主要讨论红框内的4个比较常用的表,后期我在介绍其它表的使用。这里说一下如果你想在真机上查看数据库的话必需要先获得root权限,否则无法查看。(snwrking : 我就是在模拟器上用的。模拟器上直接就有ROOT权限,可以打开data/data目录。 至于真机上,请用SuperOneClick)







1.contacts 表

_id :表的ID,主要用于其它表通过contacts 表中的ID可以查到相应的数据。
display_name:  联系人名称
photo_id:头像的ID,如果没有设置联系人头像,这个字段就为空
times_contacted:通话记录的次数
last_time_contacted: 最后的通话时间 

lookup :是一个持久化的储存 因为用户可能会改名子 但是它改不了lookup



(snwrking : 我在模拟器上的DB上,contacts表中没有display_name字段,这个字段反而是在raw_contacts里。不知道是不是版本的问题。我的模拟器版本是 Android2.3.3)

2.data表

raw_contact_id:通过raw_contact_id可以找到 raw_contact表中相对的数据。

data1 到 data15  这里保存着联系人的信息 联系人名称 联系人电话号码  电子邮件  备注 等等。

   





3.phone_look_up表


data_id  : 通过data_id可以找到 datat表中相对的数据。
raw_contact_id : 通过raw_contact_id 可以找到 raw_contact_表中相对的数据。
normalized_number: 这个字段就比较有意思了,它是将每个电话号码逆序排列。






4.raw_contact表


version :版本号,用于监听变化
deleted :删除标志, 0为默认 1 表示这行数据已经删除
display_name : 联系人名称
last_time_contacts : 最后联系的时间







有关这些的源码都在android.provider.ContactsContract这个类里面,如果想深入了解的话 可以去看看,数据库相关的操作 联查啊 啥的  都在里面。

下面说说代码是怎么用的

         先说说 Phone.CONTENT_URI,获取联系人的时候需要去这个url中去找数据 。它所指向的其实是“content:// com.android.contacts/data/phones”。这个url 对应着contacts表 和   raw_contacts表 以及 data表 所以说我们的数据都是从这三个表中获取的。


这里强调一下query第二个参数 


[java]  view plain copy
  1. private static final String[] PHONES_PROJECTION = new String[] {  
  2.        Phone.DISPLAY_NAME, Phone.NUMBER, Photo.PHOTO_ID,Phone.CONTACT_ID };  


它的意思是只去表中找 显示名称  电话号码 头像ID  联系人ID 这4个数据 ,如果你须要其它数据 的话 就须要修改这里。


获得手机通讯录联系人信息

[java]  view plain copy
  1.    /**得到手机通讯录联系人信息**/  
  2.    private void getPhoneContacts() {  
  3. ContentResolver resolver = mContext.getContentResolver();  
  4.   
  5. // 获取手机联系人  
  6. Cursor phoneCursor = resolver.query(Phone.CONTENT_URI,PHONES_PROJECTION, nullnullnull);  
  7.   
  8.   
  9. if (phoneCursor != null) {  
  10.     while (phoneCursor.moveToNext()) {  
  11.   
  12.     //得到手机号码  
  13.     String phoneNumber = phoneCursor.getString(PHONES_NUMBER_INDEX);  
  14.     //当手机号码为空的或者为空字段 跳过当前循环  
  15.     if (TextUtils.isEmpty(phoneNumber))  
  16.         continue;  
  17.       
  18.     //得到联系人名称  
  19.     String contactName = phoneCursor.getString(PHONES_DISPLAY_NAME_INDEX);  
  20.       
  21.     //得到联系人ID  
  22.     Long contactid = phoneCursor.getLong(PHONES_CONTACT_ID_INDEX);  
  23.   
  24.     //得到联系人头像ID  
  25.     Long photoid = phoneCursor.getLong(PHONES_PHOTO_ID_INDEX);  
  26.       
  27.     //得到联系人头像Bitamp  
  28.     Bitmap contactPhoto = null;  
  29.   
  30.     //photoid 大于0 表示联系人有头像 如果没有给此人设置头像则给他一个默认的  
  31.     if(photoid > 0 ) {  
  32.         Uri uri =ContentUris.withAppendedId(ContactsContract.Contacts.CONTENT_URI,contactid);  
  33.         InputStream input = ContactsContract.Contacts.openContactPhotoInputStream(resolver, uri);  
  34.         contactPhoto = BitmapFactory.decodeStream(input);  
  35.     }else {  
  36.         contactPhoto = BitmapFactory.decodeResource(getResources(), R.drawable.contact_photo);  
  37.     }  
  38.       
  39.     mContactsName.add(contactName);  
  40.     mContactsNumber.add(phoneNumber);  
  41.     mContactsPhonto.add(contactPhoto);  
  42.     }  
  43.   
  44.     phoneCursor.close();  
  45. }  
  46.    }  

获得手机sim卡联系人信息

sim卡和手机本人 获取的方式类似 只是url有点不一样 ,须要注意的一点是 sim卡  是没有联系人头像的。


[java]  view plain copy
  1.    /**得到手机SIM卡联系人人信息**/  
  2.    private void getSIMContacts() {  
  3. ContentResolver resolver = mContext.getContentResolver();  
  4. // 获取Sims卡联系人  
  5. Uri uri = Uri.parse("content://icc/adn");  
  6. Cursor phoneCursor = resolver.query(uri, PHONES_PROJECTION, nullnull,  
  7.     null);  
  8.   
  9. if (phoneCursor != null) {  
  10.     while (phoneCursor.moveToNext()) {  
  11.   
  12.     // 得到手机号码  
  13.     String phoneNumber = phoneCursor.getString(PHONES_NUMBER_INDEX);  
  14.     // 当手机号码为空的或者为空字段 跳过当前循环  
  15.     if (TextUtils.isEmpty(phoneNumber))  
  16.         continue;  
  17.     // 得到联系人名称  
  18.     String contactName = phoneCursor  
  19.         .getString(PHONES_DISPLAY_NAME_INDEX);  
  20.   
  21.     //Sim卡中没有联系人头像  
  22.       
  23.     mContactsName.add(contactName);  
  24.     mContactsNumber.add(phoneNumber);  
  25.     }  
  26.   
  27.     phoneCursor.close();  
  28. }  
  29.    }  

这个界面就可以看到联系人的 名称 号码 以及头像了。如果想在模拟器上看须要将图片拷贝到SD卡中,然后在联系人中设置一下,这里就可以看到头像了,或者 真机上会比较清楚、




任意点击一个联系人会调用系统拨打电话的界面 ,代码如下。

[java]  view plain copy
  1. //调用系统方法拨打电话  
  2. Intent dialIntent = new Intent(Intent.ACTION_CALL, Uri  
  3.     .parse("tel:" + mContactsNumber.get(position)));  
  4. startActivity(dialIntent);  












最重要的是须要AndroidManifest.xml中 加入权限 否则代码会报错的。 千万别忘了。


[html]  view plain copy
  1. <!-- 读取联系人权限 -->   
  2. <uses-permission android:name="android.permission.READ_CONTACTS"/>  
  3. <!-- 拨打电话权限 -->  
  4. <uses-permission android:name="android.permission.CALL_PHONE"/>    





下面给出完整的代码

[java]  view plain copy
  1. import java.io.InputStream;  
  2. import java.util.ArrayList;  
  3.   
  4. import android.app.ListActivity;  
  5. import android.content.ContentResolver;  
  6. import android.content.ContentUris;  
  7. import android.content.Context;  
  8. import android.content.Intent;  
  9. import android.database.Cursor;  
  10. import android.graphics.Bitmap;  
  11. import android.graphics.BitmapFactory;  
  12. import android.net.Uri;  
  13. import android.os.Bundle;  
  14. import android.provider.ContactsContract;  
  15. import android.provider.ContactsContract.CommonDataKinds.Phone;  
  16. import android.provider.ContactsContract.CommonDataKinds.Photo;  
  17. import android.text.TextUtils;  
  18. import android.view.LayoutInflater;  
  19. import android.view.View;  
  20. import android.view.ViewGroup;  
  21. import android.widget.AdapterView;  
  22. import android.widget.BaseAdapter;  
  23. import android.widget.ImageView;  
  24. import android.widget.ListView;  
  25. import android.widget.TextView;  
  26. import android.widget.AdapterView.OnItemClickListener;  
  27.   
  28. public class ContactsActivity extends ListActivity {  
  29.   
  30.     Context mContext = null;  
  31.   
  32.     /**获取库Phon表字段**/  
  33.     private static final String[] PHONES_PROJECTION = new String[] {  
  34.         Phone.DISPLAY_NAME, Phone.NUMBER, Photo.PHOTO_ID,Phone.CONTACT_ID };  
  35.      
  36.     /**联系人显示名称**/  
  37.     private static final int PHONES_DISPLAY_NAME_INDEX = 0;  
  38.       
  39.     /**电话号码**/  
  40.     private static final int PHONES_NUMBER_INDEX = 1;  
  41.       
  42.     /**头像ID**/  
  43.     private static final int PHONES_PHOTO_ID_INDEX = 2;  
  44.      
  45.     /**联系人的ID**/  
  46.     private static final int PHONES_CONTACT_ID_INDEX = 3;  
  47.       
  48.   
  49.     /**联系人名称**/  
  50.     private ArrayList<String> mContactsName = new ArrayList<String>();  
  51.       
  52.     /**联系人头像**/  
  53.     private ArrayList<String> mContactsNumber = new ArrayList<String>();  
  54.   
  55.     /**联系人头像**/  
  56.     private ArrayList<Bitmap> mContactsPhonto = new ArrayList<Bitmap>();  
  57.       
  58.     ListView mListView = null;  
  59.     MyListAdapter myAdapter = null;  
  60.   
  61.     @Override  
  62.     public void onCreate(Bundle savedInstanceState) {  
  63.     mContext = this;  
  64.     mListView = this.getListView();  
  65.     /**得到手机通讯录联系人信息**/  
  66.     getPhoneContacts();  
  67.   
  68.     myAdapter = new MyListAdapter(this);  
  69.     setListAdapter(myAdapter);  
  70.   
  71.   
  72.     mListView.setOnItemClickListener(new OnItemClickListener() {  
  73.   
  74.         @Override  
  75.         public void onItemClick(AdapterView<?> adapterView, View view,  
  76.             int position, long id) {  
  77.         //调用系统方法拨打电话  
  78.         Intent dialIntent = new Intent(Intent.ACTION_CALL, Uri  
  79.             .parse("tel:" + mContactsNumber.get(position)));  
  80.         startActivity(dialIntent);  
  81.         }  
  82.     });  
  83.   
  84.     super.onCreate(savedInstanceState);  
  85.     }  
  86.   
  87.     /**得到手机通讯录联系人信息**/  
  88.     private void getPhoneContacts() {  
  89.     ContentResolver resolver = mContext.getContentResolver();  
  90.   
  91.     // 获取手机联系人  
  92.     Cursor phoneCursor = resolver.query(Phone.CONTENT_URI,PHONES_PROJECTION, nullnullnull);  
  93.   
  94.   
  95.     if (phoneCursor != null) {  
  96.         while (phoneCursor.moveToNext()) {  
  97.   
  98.         //得到手机号码  
  99.         String phoneNumber = phoneCursor.getString(PHONES_NUMBER_INDEX);  
  100.         //当手机号码为空的或者为空字段 跳过当前循环  
  101.         if (TextUtils.isEmpty(phoneNumber))  
  102.             continue;  
  103.           
  104.         //得到联系人名称  
  105.         String contactName = phoneCursor.getString(PHONES_DISPLAY_NAME_INDEX);  
  106.           
  107.         //得到联系人ID  
  108.         Long contactid = phoneCursor.getLong(PHONES_CONTACT_ID_INDEX);  
  109.   
  110.         //得到联系人头像ID  
  111.         Long photoid = phoneCursor.getLong(PHONES_PHOTO_ID_INDEX);  
  112.           
  113.         //得到联系人头像Bitamp  
  114.         Bitmap contactPhoto = null;  
  115.   
  116.         //photoid 大于0 表示联系人有头像 如果没有给此人设置头像则给他一个默认的  
  117.         if(photoid > 0 ) {  
  118.             Uri uri =ContentUris.withAppendedId(ContactsContract.Contacts.CONTENT_URI,contactid);  
  119.             InputStream input = ContactsContract.Contacts.openContactPhotoInputStream(resolver, uri);  
  120.             contactPhoto = BitmapFactory.decodeStream(input);  
  121.         }else {  
  122.             contactPhoto = BitmapFactory.decodeResource(getResources(), R.drawable.contact_photo);  
  123.         }  
  124.           
  125.         mContactsName.add(contactName);  
  126.         mContactsNumber.add(phoneNumber);  
  127.         mContactsPhonto.add(contactPhoto);  
  128.         }  
  129.   
  130.         phoneCursor.close();  
  131.     }  
  132.     }  
  133.       
  134.     /**得到手机SIM卡联系人人信息**/  
  135.     private void getSIMContacts() {  
  136.     ContentResolver resolver = mContext.getContentResolver();  
  137.     // 获取Sims卡联系人  
  138.     Uri uri = Uri.parse("content://icc/adn");  
  139.     Cursor phoneCursor = resolver.query(uri, PHONES_PROJECTION, nullnull,  
  140.         null);  
  141.   
  142.     if (phoneCursor != null) {  
  143.         while (phoneCursor.moveToNext()) {  
  144.   
  145.         // 得到手机号码  
  146.         String phoneNumber = phoneCursor.getString(PHONES_NUMBER_INDEX);  
  147.         // 当手机号码为空的或者为空字段 跳过当前循环  
  148.         if (TextUtils.isEmpty(phoneNumber))  
  149.             continue;  
  150.         // 得到联系人名称  
  151.         String contactName = phoneCursor  
  152.             .getString(PHONES_DISPLAY_NAME_INDEX);  
  153.   
  154.         //Sim卡中没有联系人头像  
  155.           
  156.         mContactsName.add(contactName);  
  157.         mContactsNumber.add(phoneNumber);  
  158.         }  
  159.   
  160.         phoneCursor.close();  
  161.     }  
  162.     }  
  163.       
  164.     class MyListAdapter extends BaseAdapter {  
  165.     public MyListAdapter(Context context) {  
  166.         mContext = context;  
  167.     }  
  168.   
  169.     public int getCount() {  
  170.         //设置绘制数量  
  171.         return mContactsName.size();  
  172.     }  
  173.   
  174.     @Override  
  175.     public boolean areAllItemsEnabled() {  
  176.         return false;  
  177.     }  
  178.   
  179.     public Object getItem(int position) {  
  180.         return position;  
  181.     }  
  182.   
  183.     public long getItemId(int position) {  
  184.         return position;  
  185.     }  
  186.   
  187.     public View getView(int position, View convertView, ViewGroup parent) {  
  188.         ImageView iamge = null;  
  189.         TextView title = null;  
  190.         TextView text = null;  
  191.         if (convertView == null || position < mContactsNumber.size()) {  
  192.         convertView = LayoutInflater.from(mContext).inflate(  
  193.             R.layout.colorlist, null);  
  194.         iamge = (ImageView) convertView.findViewById(R.id.color_image);  
  195.         title = (TextView) convertView.findViewById(R.id.color_title);  
  196.         text = (TextView) convertView.findViewById(R.id.color_text);  
  197.         }  
  198.         //绘制联系人名称  
  199.         title.setText(mContactsName.get(position));  
  200.         //绘制联系人号码  
  201.         text.setText(mContactsNumber.get(position));  
  202.         //绘制联系人头像  
  203.         iamge.setImageBitmap(mContactsPhonto.get(position));  
  204.         return convertView;  
  205.     }  
  206.   
  207.     }  
  208. }  


列表的布局文件


[html]  view plain copy
  1. <?xml version="1.0" encoding="utf-8"?>    
  2.     
  3. <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"    
  4.     android:layout_width="fill_parent" android:layout_height="wrap_content">    
  5.     <ImageView android:id="@+id/color_image"    
  6.         android:layout_width="40dip" android:layout_height="40dip" />    
  7.     <TextView android:id="@+id/color_title"    
  8.         android:layout_width="fill_parent" android:layout_height="wrap_content"    
  9.         android:layout_toRightOf="@+id/color_image"    
  10.         android:layout_alignParentBottom="true"    
  11.         android:layout_alignParentRight="true" android:singleLine="true"    
  12.         android:ellipsize="marquee"     
  13.         android:textSize="15dip"  />    
  14.     <TextView android:id="@+id/color_text"    
  15.         android:layout_width="fill_parent" android:layout_height="wrap_content"    
  16.         android:layout_toRightOf="@+id/color_image"    
  17.         android:layout_below="@+id/color_title"    
  18.         android:layout_alignParentBottom="true"    
  19.         android:layout_alignParentRight="true"     
  20.         android:singleLine="true"    
  21.         android:ellipsize="marquee"     
  22.         android:textSize="20dip" />    
  23. </RelativeLayout>    




这章的内容如果有更熟悉的朋友欢迎和我一起讨论。老规矩每篇文章都会附带源代码,最后如果你还是觉得我写的不够详细 看的不够爽 不要紧我把源代码的下载地址贴出来 欢迎大家一起讨论学习雨松MOMO希望可以和大家一起进步。



下载地址:http://download.csdn.net/source/3559166


原文楼主追加评论:
确实有个BUG,请大家修改一下getView方法就可以了。
public View getView(int position, View convertView, ViewGroup parent) {
ImageView iamge = null;
TextView title = null;
TextView text = null;
if (convertView == null || position < mContactsNumber.size()) {
convertView = LayoutInflater.from(mContext).inflate(
R.layout.contactslist, null);
iamge = (ImageView) convertView.findViewById(R.id.color_image);
title = (TextView) convertView.findViewById(R.id.color_title);
text = (TextView) convertView.findViewById(R.id.color_text);
}
// 绘制联系人名称
title.setText(mContactsName.get(position));
// 绘制联系人号码
text.setText(mContactsNumber.get(position));
// 绘制联系人头像
iamge.setImageBitmap(mContactsPhonto.get(position));
return convertView;
}
查看原文请点击这里。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
目 录 1 前言 1 2 系统的需求分析 2 2.1 需求分析 2 2.1.1 基本功能需求 2 2.1.2 系统用例分析 2 2.2 总体设计方案 5 2.2.1 系统模块关系与划分 5 3 系统的概要设计 6 3.1 通讯录需求分析 6 3.1.1 新建、编辑联系人 6 3.1.2 查找联系人 6 3.1.3 通讯功能 6 3.1.4 个人中心 7 3.1.5 还原与备份功能 7 3.2 系统界面设计 8 4 系统编码实现 14 前 言 随着移动通信与Internet向移动终端的普及,网络和用户对移动终端的要求越来越高 ,而Symbian,Windows Mobile,PalmOS等手机平台过于封闭,不能很好的满足用户的需求,因此市场迫切需要 一个开发性很强的平台。经过多年的发展,第三代数字通信(3G)技术活动了广泛的接 受,它为移动终端用户带来了更快的数据传输速率。随着3G网络的使用,移动终端不再 仅是通讯网络的终端,还将成为互联网的终端。因此,移动终端的应用软件和需要的服 务将有很大的发展空间。Google为此与2007年11月推出了一个专为移动设备设计的软 件平台——AndroidAndroid 是一套真正意义上的开发性的移动设备综合平台,它包括操作系统、中间件和一些关键 的平台应用。Android 是由Linux+Java构成的开源软件,允许所有厂商和个人在其基础上进行开发。Android平 台的开放性等特点既能促进技术(包括平台本身)的创新,又有助于降低开发成本,还 可以是运营商能非常方便地制定自己的特色化的产品。因此,它具有很大的市场发展潜 力。 Android(Google公司)是Google开发的基于Linux平台的开源手机操作系统。它包括 操作系统、用户界面和应用程序 ——移动电话工作所需的全部软件,而且不存在任何以往阻碍移动产业创新的专有权障碍 。谷歌与开放手机联盟合作开发了 Android,这个联盟由包括中国移动、摩托罗拉、高通、宏达和 T-Mobile 在内的 30 多家技术和无线应用的领军企业组成。 1) 优点:具备触摸屏、高级图形显示和上网功能,界面强大,可以说是一种 融入全部Web应用的单一平台 2) 缺点:由于时时刻刻都需要和网络进行连接,因此在手机的能耗方面控制就较差, 导致待机能力不足;又由于其开源性,过分依赖开发商,缺少标准配置。 1. 需求分析 1. 基本功能需求 能要求:实现通信录的在线备份还原功能,能把系统的通信录一键导入导出。 实现要求:客户端基于Android平台实现,服务端技术自定 用例场景:小明丢了手机,只好去抢购了一个小米同时把手机卡补办回来,需要把之 前手机的200个联系人补上。好在小明之前把所有联系人都备份到服务器了,只需要下载 在线通信录后,登录平台,一键还原即可。 2. 系统用例分析 图1- 1显示了通话记录功能模块。包括了联系人详细信息查看,清空通话记录,在选择一个条 目后,可以对其进行拨打电话,发送短信功能的操作,也可以进行删除。 手机用户 图1-1 通话记录模块用例图 图1- 2显示了联系人功能模块的用例。包括了查看联系人详细信息,编辑联系人信息,新建联 系人,对选中的联系人,可以对其进行拨打电话、发送短信的操作。用户还可以进行联 系人搜索,这样方便用户快速找到想找的联系人信息。 图1-2 电话薄模块用例图 图1- 3为个人中心模块中设置个人详细信息子模块的用例。该模块的功能就是用户设置自己的 个人基本信息。 图1-3 设置个人信息模块用例图 图1-4显示了联系人还原与备份的用例图。该模块的功能就是联系人的还原与备份 图1-4 联系人导入导出 2. 总体设计方案 1. 系统模块关系与划分 一个好的系统设计的步骤决定了程序是否能按照设计者的目的按时完成,是否能在规定 的时间内按照设计者的要求高质量的完成程序必要的功能。并且按照标准的设计步骤对 程序进行调试,测试,以及后期的优化完善,使程序更加具有健壮性和可用性。通过对 通讯录功能、系统模块、用户需求方面进行全方位的分析制定开发流程。 采用标准的开发流程确定系统具有用户管理功能,联系人增删改功能,通讯功能,查找 功能,备份等功能。 图1-5 系统功能图 通过对系统的功能结构的分析,设计后系统运行流程是系统运行后用户将进入通讯录 主界面,可以看到联系人,增加联系人,。通过点击MENU界面的增加功能选项可以新增 联系人,通过查找按钮可以对联系人进行姓名、号码的操作。通过个人中心可以显示所 有联系人、还原所有联系人、并备份功能。在联系人详细信息界面点击MENU键弹出通讯 功能框选择拨打电话按钮或者发信息发邮件按键,系统的运行流程图如1-6所示。 图1-6 系统运行流程 2.1通讯录需求分析 根据手机功能调查显
通过使用Android Studio中的内容提供者,我们可以轻松地实现建立手机通讯录界面并读取系统联系人的功能。 首先,我们需要创建一个新的Android项目,并在项目的布局文件中设计通讯录界面。可以包括一个ListView用于显示联系人列表,以及可选的搜索框等组件。 接下来,我们需要编写一个自定义的内容提供者。这个内容提供者将封装我们对系统联系人数据的访问。可以参考Android提供的ContactsContract类,其中包含了通讯录的数据结构和相关的查询方法。 在内容提供者中,我们需要实现读取系统联系人的功能。可以使用ContentResolver的query()方法来执行查询操作,并通过向查询结果Cursor中添加特定的列,来获取我们需要的联系人字段,如姓名、电话号码等。 接着,我们需要在通讯录界面的Activity中使用Loader来加载联系人数据。我们可以使用CursorLoader来实现这一功能,并指定要加载的数据源为我们自定义的内容提供者。 加载完成后,我们将获取到的联系人数据填充到ListView中,以展示联系人列表。可以使用SimpleCursorAdapter来帮助我们将Cursor中的数据绑定到ListView的每一项。 最后,我们可以为通讯录界面的ListView添加点击事件,当用户点击某个联系人时,可以跳转到该联系人的详细信息页面,或执行其他操作。 通过上述步骤,我们可以实现一个简单的手机通讯录界面,并成功读取系统联系人。这样,用户就可以方便地查看、搜索和管理手机通讯录中的联系人信息了。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值