简单步骤:Android studio 内容提供者 - 实现建立手机通讯录界面,读取系统联系人

提示:本篇文章将会尽量保持精简,同时请诸位敲写代码时保持耐心,三连是最大的支持!


前言

本篇文章将会介绍如何使用 Android studio 内容提供者 实现 “读取手机通讯录” 的项目,文章是经由本人实际编写过后得出,同时项目中的名称等如有冲突可自行更改。文章尽量保持精简,也请诸位保持耐心,且会加以图文解释,方便读者能够更佳观看。

配置:Android studio 2021.1.1.21 windows  


一、项目介绍

本项目通过显示一个通讯录的界面,以列表的形式来显示 Android 设备通讯录中暴露的数据, 而 Android 便提供了一个组件 ContentProvider(内容提供者) 来充当一个 “中介” 的角色。具体有关 ContentProvider(内容提供者) 的细节在这就不再赘述。

二、使用步骤

1.创建程序

打开 Android studio,在 Android studio 创建一个新的应用程序,命名为 Contact ,指定包名为 com.itcast.contacts 。

2.添加 recyclerview-v7 库

因为我们通讯录界面将要使用到 RecyclerView 控件,以列表的形式展示,所以需要将拥有  RecyclerView 控件的 recyclerview-v7 库添加进程序。但是由于版本和补丁的更新,我们只需要在 Android 的 Gradle Scripts 下的 build.gradle(Module:Contact.app) 中填入一下代码

代码如下:

dependencies {
// 。。。。。
implementation 'androidx.recyclerview:recyclerview:1.0.0'
}

更为详细的说明可以移步到:简单步骤:解决 Android studio 2021.1.1 出现添加 “ recyclerview-v7 库 ”报错_蛇形刁手的博客-CSDN博客

3.放置界面控件

在 res/layout 文件夹的 activity_contact.xml 下放置一个 TextView控件 和 RecyclerView 控件来以列表的形式显示通讯录和其中联系人的信息。 

  代码如下:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:background="#eaeaea"
    android:orientation="vertical">
    <TextView
        android:layout_width="match_parent"
        android:layout_height="50dp"
        android:background="#4889f4"
        android:gravity="center"
        android:text="通讯录"
        android:textColor="@android:color/white"
        android:textSize="20sp" />
    <androidx.recyclerview.widget.RecyclerView
        android:id="@+id/rv_contact"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:layout_margin="5dp" />
</LinearLayout>

4.搭建界面布局

<1> 在 res/layout 文件夹下,创建布局文件 contact_item.xml , 编写界面控件。

 代码如下:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:layout_margin="5dp"
    android:background="@drawable/item_bg"
    android:orientation="horizontal"
    android:padding="8dp">
    <ImageView
        android:id="@+id/iv_photo"
        android:layout_width="60dp"
        android:layout_height="60dp"
        android:src="@drawable/contact_photo" />
    <LinearLayout
        android:layout_width="match_parent"
        android:layout_height="60dp"
        android:layout_marginLeft="8dp"
        android:gravity="center_vertical"
        android:orientation="vertical">
        <TextView
            android:id="@+id/tv_name"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:textColor="@android:color/black"
            android:textSize="16sp" />
        <TextView
            android:id="@+id/tv_phone"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_marginTop="6dp"
            android:textColor="@android:color/darker_gray"
            android:textSize="14sp" />
    </LinearLayout>
</LinearLayout>

<2> 导入图片

将通讯录界面所需要的图片 contact_photo.png 导入到 res/drewable 文件夹中。(如果文章会自动生成水印,那么推荐使用自己的图片)

图片如下:

<3> 创建条目界面的背景文件

在res/drewable 文件夹中创建 item_bg.xml文件设置条目背景。

代码如下:

<?xml version="1.0" encoding="utf-8"?>
<shape xmlns:android="http://schemas.android.com/apk/res/android">
    <solid android:color="#ffffff" />
    <corners android:radius="8dp" />
</shape>

 5.封装实体类

在程序的 java/com.itcast.contacts 包下创建一个 ContactInfo 类,用来创建联系人信息的属性。

 代码如下:

package com.itcast.contacts;
public class ContactInfo {
    private String contactName;   //联系人名称
    private String phoneNumber;   //电话号码
    public String getContactName() {
        return contactName;
    }
    public void setContactName(String contactName) {
        this.contactName = contactName;
    }
    public String getPhoneNumber() {
        return phoneNumber;
    }
    public void setPhoneNumber(String phoneNumber) {
        this.phoneNumber = phoneNumber;
    }
}

6.编写数据适配器

在程序的 java/com.itcast.contacts 包下创建一个 ContactAdapter 类,由于我们使用了 RecyclerView 控件,所以就需要创建一个数据适配器,来对 RecyclerView 控件进行数据适配。

 代码如下:

package com.itcast.contacts;

import android.content.Context;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ImageView;
import android.widget.TextView;

import androidx.recyclerview.widget.RecyclerView;

import java.util.List;

public class ContactAdapter extends RecyclerView.Adapter<ContactAdapter
        .MyViewHolder>  {
    private Context mContext;
    private List<ContactInfo> contactInfoList;
    public ContactAdapter(Context context, List<ContactInfo> contactInfoList){
        this.mContext=context;
        this.contactInfoList=contactInfoList;
    }
    @Override
    public MyViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
        MyViewHolder holder = new MyViewHolder(
                LayoutInflater.from(mContext).inflate(
                        R.layout.contact_item, parent, false));
        return holder;
    }
    @Override
    public void onBindViewHolder(MyViewHolder holder, int position) {
        holder.tv_name.setText(contactInfoList.get(position).getContactName());
        holder.tv_phone.setText(contactInfoList.get(position).getPhoneNumber());
    }
    @Override
    public int getItemCount() {
        return contactInfoList.size();
    }
    class MyViewHolder extends RecyclerView.ViewHolder {
        TextView tv_name,tv_phone;
        ImageView iv_photo;
        public MyViewHolder(View view) {
            super(view);
            tv_name = view.findViewById(R.id.tv_name);
            tv_phone = view.findViewById(R.id.tv_phone);
            iv_photo = view.findViewById(R.id.iv_photo);
        }
    }
}

7.实现显示界面数据功能

在程序的 java/com.itcast.contacts 包下创建一个 ContactActivity 类, 来申请读取手机通讯录的权限,同时重写 onRequestPermissionsResult () 方法获取通讯录权限是否申请成功。

 代码如下:

package com.itcast.contacts;

import android.annotation.SuppressLint;
import android.content.pm.PackageManager;
import android.database.Cursor;
import android.os.Build;
import android.os.Bundle;
import android.provider.ContactsContract;
import android.widget.Toast;

import androidx.appcompat.app.AppCompatActivity;
import androidx.core.app.ActivityCompat;
import androidx.recyclerview.widget.LinearLayoutManager;
import androidx.recyclerview.widget.RecyclerView;

import java.util.ArrayList;
import java.util.List;

public class ContactActivity extends AppCompatActivity {
    private ContactAdapter adapter;
    private RecyclerView rv_contact;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_contact);
        init();
    }
    private void setData(){
        List<ContactInfo> contactInfos=getContacts();
        adapter=new ContactAdapter(ContactActivity.this,contactInfos);
        rv_contact.setAdapter(adapter);
    }
    public List<ContactInfo> getContacts() {
        List<ContactInfo> contactInfos = new ArrayList<>();
        Cursor cursor = getContentResolver().query(ContactsContract.
                Contacts.CONTENT_URI, null, null, null, null);
        if (contactInfos!=null)contactInfos.clear();//清除集合中的数据
        while (cursor.moveToNext()) {
            @SuppressLint("Range") String id = cursor.getString(
                    cursor.getColumnIndex(ContactsContract.Contacts._ID));
            @SuppressLint("Range") String name = cursor.getString (cursor.getColumnIndex(ContactsContract.
                    Contacts.DISPLAY_NAME));
            @SuppressLint("Range") int isHas = Integer.parseInt(cursor.getString(cursor.getColumnIndex(
                    ContactsContract.Contacts.HAS_PHONE_NUMBER)));
            if (isHas > 0) {
                Cursor c = getContentResolver().query(ContactsContract.
                                CommonDataKinds.Phone.CONTENT_URI, null,
                        ContactsContract.CommonDataKinds.Phone.CONTACT_ID +
                                " = " + id, null, null);
                while (c.moveToNext()) {
                    ContactInfo info = new ContactInfo();
                    info.setContactName(name);
                    @SuppressLint("Range") String number = c.getString(c.getColumnIndex(ContactsContract.
                            CommonDataKinds.Phone.NUMBER)).trim();
                    number = number.replace(" ", "");
                    number = number.replace("-", "");
                    info.setPhoneNumber(number);
                    contactInfos.add(info);
                }
                c.close();
            }
        }
        cursor.close();
        return contactInfos;
    }
    private void init(){
        rv_contact=findViewById(R.id.rv_contact);
        rv_contact.setLayoutManager(new LinearLayoutManager(this));
        getPermissions();
    }
    String[] permissionList;
    public void getPermissions() {
        if (android.os.Build.VERSION.SDK_INT >= Build.VERSION_CODES.M) {
            permissionList = new String[]{"android.permission.READ_CONTACTS"};
            ArrayList<String> list = new ArrayList<String>();
            // 循环判断所需权限中有哪个尚未被授权
            for (int i = 0; i < permissionList.length; i++) {
                if (ActivityCompat.checkSelfPermission(this, permissionList[i])
                        != PackageManager.PERMISSION_GRANTED)
                    list.add(permissionList[i]);
            }
            if (list.size() > 0) {
                ActivityCompat.requestPermissions(this,
                        list.toArray(new String[list.size()]), 1);
            } else {
                setData();//后续创建该方法
            }
        } else {
            setData();  //后续创建该方法
        }
    }
    @Override
    public void onRequestPermissionsResult(int requestCode, String[] permissions,
                                           int[] grantResults) {
        super.onRequestPermissionsResult(requestCode, permissions, grantResults);
        if (requestCode == 1) {
            for (int i = 0; i < permissions.length; i++) {
                if(permissions[i].equals("android.permission.READ_CONTACTS")
                        && grantResults[i] == PackageManager.PERMISSION_GRANTED){
                    Toast.makeText(this, "读取通讯录权限申请成功",
                            Toast.LENGTH_SHORT).show();
                    setData();//后续创建该方法
                }else{
                    Toast.makeText(this,"读取通讯录权限申请失败",
                            Toast.LENGTH_SHORT).show();
                }
            }
        }
    }

}

8.去掉默认标题栏,添加读取系统通讯录权限

<1> 在 AndroidManifest.xml 文件的 <application> 标签下修改标题栏,使标题栏更加美观。

<2> 因为我们需要使用到系统的通讯录,所以还要在 AndroidManifest.xml 文件中添加读取系统通讯录权限

<3> 同时解决可能会出现虚拟器运行后“ xxx has stopped"的错误,详细内容可以移步到:简单步骤:解决 Android studio 出现 “ xxx(项目名) has stopped ” 的错误_蛇形刁手的博客-CSDN博客

 代码如下:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    package="com.itcast.contacts">

    <application
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:roundIcon="@mipmap/ic_launcher_round"
        android:supportsRtl="true"
        android:theme="@style/Theme.AppCompat.NoActionBar">
        <activity
            android:name=".ContactActivity"
            android:exported="true"
            tools:ignore="MissingClass">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
    </application>
    <uses-permission android:name="android.permission.READ_CONTACTS" />
</manifest>

9.运行程序

第一次运行时,会弹出一个 “ Allow Contacts to access your contacts?” 窗口,有两个选项: “ ALLOW ” 表示允许读取 和 “ DENY ” 拒绝读取,我们点击允许读取。当第一次运行时,我们的通讯录界面是一片空白,需要我们在系统的通讯录输入联系人信息予以提供。重新切换到我们的通讯录界面后便会出现信息。

图文解释如下:


总结

以上就是 Android studio 内容提供者 - 实现读取手机通讯录的具体操作,由于每个人的配置、编写习惯等等多种原因,如果在实际操作中出现问题,或者发现文章中的错误,也可以在评论区中发表出来,尽能所答,欢迎指点与建议。

如果感到有帮助!!!

编写不易,关注、三连是最大的支持!!!

欢迎建议,感谢支持!!!

  • 59
    点赞
  • 234
    收藏
    觉得还不错? 一键收藏
  • 19
    评论
Android开发实验---通讯录 实验报告 通讯录 姓名: 学号: 课程名称: 移动应用开发 所在学院: 信息科学与工程学院 专业班级: 计算机 任课教师: Android开发实验---通讯录全文共25页,当前为第1页。 Android开发实验---通讯录全文共25页,当前为第1页。 实验项目名称 通讯录 实验目的与要求: 目的:练习掌握 Android 软件开发基本编程技术、Android 系统 SQLite 数据库的使用、通话、短信的使用等,设计制作一 Android 通讯录软件。 要求: (1)每位同学独立设计软件功能、完成软件的开发与测试。 (2)软件完成后由任课教师检查、提问软件功能、软件的设计和开发的代码。(3)每位同学独立完成实验报告(包括学号、姓名,实验目的、实验内容、软件功能、设计思想、实验总结等),并将实验报告和开发软件的工程文件压缩包提交任课教师。 实验设备及软件: 一台电脑,Androidstudio2.3.3软件 实验方法(原理、流程图) 实现通讯录功能和界面可在实验开发中由每位同学自己设计,但必须使用 SQLite 数据库保存通讯录信息。 以下功能和软件界面供参考:仿照一般手机通讯录的使用界面和功能,主要功能可包括: (1)可以添加、删减联系人 (2)联系人的信息包括:姓名、手机号码、工作单位、群组、电子邮件、手机铃声 (3)选择联系人后,可以快速进行操作,如:拨打电话发送短信、查看详细、移出群组、移动分组、删除联系人等。查看详细时显示手机号码、群组、和设定的手机铃声以及同该联系人的通话记录。 (4)在联系人界面,可以查看各群组。点击群组,跳出对应联系人。 (5)有拨号键盘,点击数字将号码显示出来,并可以对手机号码进行删减 (6)可以发送信息,显示信息记录 (7)发送信息时,可以快速选择现有联系人。 实验过程、步骤内容 通讯录界面要求布局合理简约,颜色舒,控制按钮简单明了,让用户一眼就能看出各个按钮的作用及操方法。能实要求的功能,但又不重复。要考到不同手机屏幕大小可能不一样,应尽兼容大多数手机屏幕尺寸,使之显示无障碍。 实现思路与简单设计 1. 关于通讯录获取联系人信息以及联系人信息的存储结构的分析。 2. 解决添加联系人的表中添加数据以及读取已保存的数据 3. 实现通讯录的基本功能的具体实现。 4. 对通讯录界面的具体设计,给人一种方便的操作通道。 5. 灵活应用各种系统库函数实现完整的通讯录Android开发实验---通讯录全文共25页,当前为第2页。开发流程图: Android开发实验---通讯录全文共25页,当前为第2页。 主要功能分析: 增加、删除、编联系人 点击通信录界面中的增加按钮,入增加联系人面。输入联系人的基本信息,并可根据用户需求增加个性化信息如头像、姓名、手机号码、办室电话、家庭电话、职务职称、单位名称、地址、邮政编码、Email、其他联系方式、备注这些信息,击确认返回主界面。点击通信录中一个已存在的联系人,进入联系人编辑界面,可修改系人的资料或进行删除联系人操作,完成后退回到主界面。对列表中联系人的标记,点mnu键弹出功能界面上的删除按键也可进行删除。还可以在菜单上选择删除全部联系人清空通讯录。在删除联系人的过程中,系统将提示用户是否继续操作,若放弃操作,则系人信息将继续保存。 2、查找联系人 用户点击menu键打开底部菜单框,底部菜单框为查询系人提供入口,进入通讯录的缺省页面为联系人列表,在列表中看到所有联系人的姓名、电话息排列,用户点击查找按键输入联系人基本信息,,通讯录显所有符合查询条件的联系人列表,用户选择一个联系人进入联系人基本信息页面进行其他操作;查询完成,用户按返回键返回主界面。 3、通功能 用户在通录选择联系人进入联系详细信息界面,这时点击menu键打开通信功能框,选择打电话、发信息的功能进行操作。 4、菜单能 通过对menu按的点击,显示底部菜框,包含有增加查找、除、菜单、返回功能,菜单按键则包含显示所有、删除所有等实用功能。 增加联系人: 这个功能由ContactAdder类完成,具体实现不是比较麻烦,保存动作由ContentResolver类解决,但实现方式有所不同,可分为一次性批量增加与挨个增加。 由于界面设置繁琐,代码页过多,所以挑出其中的一部分进行说明 首先是DB数据库建库过程: Android开发实验---通讯录全文共25页,当前为第3页。package com.xample.hivian.my_contact_manager.models.db; import android.content.ContentValues; import android.database.Cursor; import android.database.
通过使用Android Studio中的内容提供者,我们可以轻松地实现建立手机通讯录界面读取系统联系人的功能。 首先,我们需要创建一个新的Android项目,并在项目的布局文件中设计通讯录界面。可以包括一个ListView用于显示联系人列表,以及可选的搜索框等组件。 接下来,我们需要编写一个自定义的内容提供者。这个内容提供者将封装我们对系统联系人数据的访问。可以参考Android提供的ContactsContract类,其中包含了通讯录的数据结构和相关的查询方法。 在内容提供者中,我们需要实现读取系统联系人的功能。可以使用ContentResolver的query()方法来执行查询操作,并通过向查询结果Cursor中添加特定的列,来获取我们需要的联系人字段,如姓名、电话号码等。 接着,我们需要在通讯录界面的Activity中使用Loader来加载联系人数据。我们可以使用CursorLoader来实现这一功能,并指定要加载的数据源为我们自定义的内容提供者。 加载完成后,我们将获取到的联系人数据填充到ListView中,以展示联系人列表。可以使用SimpleCursorAdapter来帮助我们将Cursor中的数据绑定到ListView的每一项。 最后,我们可以为通讯录界面的ListView添加点击事件,当用户点击某个联系人时,可以跳转到该联系人的详细信息页面,或执行其他操作。 通过上述步骤,我们可以实现一个简单手机通讯录界面,并成功读取系统联系人。这样,用户就可以方便地查看、搜索和管理手机通讯录中的联系人信息了。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值