通讯录开发主要是获取用户手机中的联系人,进而可以在应用中添加好友

一 .如何访问通讯录

  (1)在iOS9之前,有两个框架可以访问用户的通讯录

    AddressBookUI.framework: 提供了联系人列表界面,联系人详情界面,添加练习人界面等,一般用于选择联系人

    AddressBook.framework: 纯C语言的API,仅仅是获的联系人数据,没有提供UI界面展示,需要自己搭建联系人展示界面,里面的数据类型大部分基于Core Foundation框架,使用起来极其蛋疼

  (2)在iOS9开始,也有两个框架可以访问用户的通讯录

ContactsUI.framework: 对应AddressBookUI.framework

    Contacts.framework: 对应AddressBook.framework

二.代码演示

     (1)AddressBookUI的使用

        使用步骤

        1)创建选择联系人控制器

        2)设置代理

        3)实现代理方法(在代理方法中拿到用户选择的联系人)

        4)弹出控制器

       代码如下:

import UIKit
import AddressBookUI

class ViewController: UIViewController {

    override func viewDidLoad() {
        super.viewDidLoad()
    }

    override func touchesBegan(_ touches: Set<UITouch>, with event: UIEvent?) {
        // 1.创建联系人选择的控制器
        let ppnc = ABPeoplePickerNavigationController()

        // 2.设置代理
        ppnc.peoplePickerDelegate = self

        // 3.弹出控制器
        present(ppnc, animated: true, completion: nil)
    }
}

extension ViewController : ABPeoplePickerNavigationControllerDelegate {
    // 用户选中了某一个联系人
    func peoplePickerNavigationController(_ peoplePicker: ABPeoplePickerNavigationController, didSelectPerson person: ABRecord) {
        // 1.获取联系人的姓名
        /*
         Unmanaged<CFTypeRef>? : 非托管对象
            * 在Swift和C语言进行混编的过程中,产生的一个临时对象,真正使用的时候需要将非托管对象,转成真正的对象才能进行使用
            * takeUnretainedValue : 表示在转化的过程中,不会对对象进行一次retain操作
            * takeRetainedValue : 表示在转化的过程中,有对对象进行一次retain操作
                注意:一旦使用takeRetainedValue,那么必须对之前的非托管对象进行一次release(),否则就会产生内存泄漏
                 let UnManageObjc = ABRecordCopyValue(person, kABPersonLastNameProperty)
                 let lastname = UnManageObjc?.takeRetainedValue() as? String
                 UnManageObjc?.release()
        */
        guard let lastname = ABRecordCopyValue(person, kABPersonLastNameProperty).takeUnretainedValue() as? String else { return }
        guard let firstname = ABRecordCopyValue(person, kABPersonFirstNameProperty).takeUnretainedValue() as? String else { return }
        print("姓名:\(firstname) \(lastname)")

        // 2.获取联系人的电话号码
        // ABMultiValue 类似于一个字典,里面有key/value
        // 2.1.从person中拷贝出来所有的电话号码
        let phones = ABRecordCopyValue(person, kABPersonPhoneProperty).takeUnretainedValue() as ABMultiValue
        // 2.2.遍历ABMultiValue中的所有电话
        // guard let count = phones.count else { return } 错误写法
        let count = ABMultiValueGetCount(phones)

        ..<count {
            let phoneLabel = ABMultiValueCopyLabelAtIndex(phones, i).takeUnretainedValue() as String
            guard let phoneValue = ABMultiValueCopyValueAtIndex(phones, i).takeUnretainedValue() as? String else { continue }
            print("phoneLabel:\(phoneLabel) phoneValue:\(phoneValue)")
        }
    }
}

          注意: 这里有一种对象 :Unmanaged<CFTypeRef>? : 非托管对象,这种对象是在Swift和C语言进行混编的过程中,出现的.需要用takeUnretainedValue() 或者 takeRetainedValue()进行转化.

          运行结果如下图:

      (2)AddressBook的使用

        1)获取用户的授权

          获取授权状态

          如果用户是未决定状态,则请求授权

        2)获取联系人信息

          获取授权状态

          如果是已经授权,则获取联系人信息

          创建通讯录对象

          获取通信录中所有的联系人

          遍历所有的联系人,获取联系人信息

      获取用户授权的代码实现,通常在应用启动时就询问用户授权

          AppDelegate中代码如下所示:

import UIKit
import AddressBook

@UIApplicationMain
class AppDelegate: UIResponder, UIApplicationDelegate {

    var window: UIWindow?

    func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplicationLaunchOptionsKey: Any]?) -> Bool {

        // 1.获取用户的授权状态
        let status = ABAddressBookGetAuthorizationStatus()

        // 2.判断授权状态是否未决定
        if status == .notDetermined {
            // 2.1.创建通信录对象
            let addressBook = ABAddressBookCreate().takeUnretainedValue()

            // 2.2.请求授权
            ABAddressBookRequestAccessWithCompletion(addressBook, { (isFlag : Bool, error : CFError?) in
                if isFlag {
                    print("授权成功")
                } else {
                    print("授权失败")
                }
            })
        }

        return true
    }
}

      ViewController中代码如下所示:

import UIKit
import AddressBook

class ViewController: UIViewController {

    override func viewDidLoad() {
        super.viewDidLoad()
    }

    override func touchesBegan(_ touches: Set<UITouch>, with event: UIEvent?) {
        // 1.获取用户授权状态
        let status = ABAddressBookGetAuthorizationStatus()

        // 2.判断是否是已经授权
        guard status == .authorized else {
            return
        }

        // 3.创建通信录对象
        let addressBook = ABAddressBookCreate().takeUnretainedValue()

        // 4.从对象中,拷贝出来所有的联系人
        let peopleArray = ABAddressBookCopyArrayOfAllPeople(addressBook).takeUnretainedValue()

        // 5.遍历数组,获取每一个联系人
        let count = CFArrayGetCount(peopleArray)
        ..<count {
            // 5.1.获取指针
            let pointer = CFArrayGetValueAtIndex(peopleArray, i)

            // 5.2.获取指针指向的对象
            // unsafeBitCast : 将指针转成某一个对象
            let person = unsafeBitCast(pointer, to: ABRecord.self)

            // 5.3.获取该联系人的姓名
            guard let lastname = ABRecordCopyValue(person, kABPersonLastNameProperty).takeUnretainedValue() as? String else { continue }
            guard let firstname = ABRecordCopyValue(person, kABPersonFirstNameProperty).takeUnretainedValue() as? String else { continue }
            print("姓名:\(firstname) \(lastname)")

            // 5.4.获取电话号码
            let phones = ABRecordCopyValue(person, kABPersonPhoneProperty).takeUnretainedValue() as ABMultiValue
            let phoneCount = ABMultiValueGetCount(phones)
            ..<phoneCount {
                // let phoneLabel = ABMultiValueCopyLabelAtIndex(phones, i).takeUnretainedValue() as String
                guard let phoneValue = ABMultiValueCopyValueAtIndex(phones, i).takeUnretainedValue() as? String else { continue }
                print(phoneValue)
            }
        }
    }
}

      注意: 由于需要访问请求授权,需要在info.plist配置NSContactsUsageDescription这个key

      (3)ContactsUI的使用

        1)使用步骤

          创建选择联系人控制器

          设置代理

          实现代理方法(在代理中拿到用户选择的联系人)

          弹出控制器

        2)代码实现

import UIKit
import ContactsUI

class ViewController: UIViewController {

    override func viewDidLoad() {
        super.viewDidLoad()
    }

    override func touchesBegan(_ touches: Set<UITouch>, with event: UIEvent?) {
        // 1.创建联系人选择的控制器
        let cpvc = CNContactPickerViewController()

        // 2.设置代理
        cpvc.delegate = self

        // 3.弹出控制器
        present(cpvc, animated: true, completion: nil)
    }
}

extension ViewController : CNContactPickerDelegate {
    func contactPicker(_ picker: CNContactPickerViewController, didSelect contact: CNContact) {
        // 1.获取用户的姓名
        // lastname --> familyName
        // firstname --> givenname
        let lastname = contact.familyName
        let firstname = contact.givenName
        print("姓名:\(firstname) \(lastname)")

        // 2.获取用户电话号码(ABMultivalue)
        let phones = contact.phoneNumbers
        for phone in phones {
            let phoneLabel = phone.label
            let phoneValue = phone.value.stringValue
            print("phoneLabel:\(phoneLabel). phoneValue:\(phoneValue)")
        }
    }

}

输出结果和AddressBookUI.framework中的输出结果类似

      4)Contacts

1)获取用户的授权

          获取授权状态

          如果用户是未决定状态,则请求授权

        2)获取联系人信息

          获取授权状态

          如果是已经授权,则获取联系人信息

          创建通讯录对象

          获取通信录中所有的联系人

          遍历所有的联系人,获取联系人信息

      获取用户授权的代码实现,通常在应用启动时就询问用户授权,请求授权需要在info.plist配置NSContactsUsageDescription这个key

AppDelegate代码如下所示:

import UIKit
import Contacts

@UIApplicationMain
class AppDelegate: UIResponder, UIApplicationDelegate {

    var window: UIWindow?

    func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplicationLaunchOptionsKey: Any]?) -> Bool {

        // 1.获取授权状态
        // CNContactStore --> 通信录对象
        let status = CNContactStore.authorizationStatus(for: .contacts)

        // 2.判断如果是未决定状态,请求授权
        if status == .notDetermined {
            // 2.1.创建通信录对象
            let store = CNContactStore()

            // 2.2.请求授权
            store.requestAccess(for: .contacts, completionHandler: { (isFlag : Bool, error : Error?) in
                if isFlag {
                    print("授权成功")
                } else {
                    print("授权失败")
                }
            })
        }

        return true
    }
}

ViewController中代码如下:

import UIKit
import Contacts

class ViewController: UIViewController {

    override func viewDidLoad() {
        super.viewDidLoad()
    }

    override func touchesBegan(_ touches: Set<UITouch>, with event: UIEvent?) {
        // 1.获取授权状态
        let status = CNContactStore.authorizationStatus(for: .contacts)

        // 2.判断是否是已经授权
        guard status == .authorized else {
            return
        }

        // 3.创建通信录对象
        let store = CNContactStore()

        // 4.从通信录中获取所有的联系人
        // 4.1.获取fetch,并且指定之后要获取联系人中的什么属性
        let keys = [CNContactFamilyNameKey as NSString, CNContactGivenNameKey as NSString, CNContactPhoneNumbersKey as NSString]

        // 4.2.创建请求对象
        let request = CNContactFetchRequest(keysToFetch: keys)

        // 4.3.遍历所有的联系人
        do {
            try store.enumerateContacts(with: request, usingBlock: { (contact : CNContact, stop : UnsafeMutablePointer<ObjCBool>) -> Void in
                // 1.获取姓名
                let lastname = contact.familyName
                let firstname = contact.givenName
                print(lastname, firstname)

                // 2.获取电话号码
                let phoneNumers = contact.phoneNumbers
                for phone in phoneNumers {
                    print(phone.label ?? "没有Label")
                    print(phone.value.stringValue)
                }
            })
        } catch {
            print(error)
        }

    }
}

iOS中通讯录的开发的更多相关文章

  1. IOS中微博正文开发步骤总结

    微博正文开发步骤总结 1.新建正文控制器,在点击首页的某一条微博时跳转过去 2.在MainController中设置导航控制器的代理,监听所有导航控制器的跳转 1> 如果即将显示的不是根控制器 ...

  2. iOS中通讯录电话号码空格问题

    今天在读取通讯录的时候,读取到的手机号码格式为* (***) ***-****的,乍看下,数字中间有空格."-".(.)的非数字字符. 然后我就打算替换这些非数字字符,结果替换完, ...

  3. iOS中的地图和定位

    文章摘自http://www.cnblogs.com/kenshincui/p/4125570.html#location  如有侵权,请联系删除. 概览 现在很多社交.电商.团购应用都引入了地图和定 ...

  4. iOS中获取本地通讯录联系人以及汉字首字母排序

    iOS中获取手机通讯录中的联系人信息: /*** 加载本地联系人*/ - (void)loadLocalContacts { //新建一个通讯录类 ABAddressBookRef addressBo ...

  5. IOS开发数据存储篇—IOS中的几种数据存储方式

    IOS开发数据存储篇—IOS中的几种数据存储方式 发表于2016/4/5 21:02:09  421人阅读 分类: 数据存储 在项目开发当中,我们经常会对一些数据进行本地缓存处理.离线缓存的数据一般都 ...

  6. iOS开发——高级篇——iOS中常见的设计模式(MVC/单例/委托/观察者)

    关于设计模式这个问题,在网上也找过一些资料,下面是我自己总结的,分享给大家 如果你刚接触设计模式,我们有好消息告诉你!首先,多亏了Cocoa的构建方式,你已经使用了许多的设计模式以及被鼓励的最佳实践. ...

  7. iOS开发小技巧--iOS中设置applicationIconBadgeNumber遇到的问题

    iOS中设置applicationIconBadgeNumber 在iOS7中直接设置applicationIconBadgeNumber没有问题,但是在iOS8之后设置applicationIcon ...

  8. 《转》iOS音频视频初级开发

    代码改变世界 Posts - 73, Articles - 0, Comments - 1539 Cnblogs Dashboard Logout HOME CONTACT GALLERY RSS   ...

  9. IOS中调用系统的电话、短信、邮件、浏览功能

    iOS开发系列--通讯录.蓝牙.内购.GameCenter.iCloud.Passbook系统服务开发汇总 2015-01-13 09:16 by KenshinCui, 26990 阅读, 35 评 ...

随机推荐

  1. [LeetCode] Fraction to Recurring Decimal 分数转循环小数

    Given two integers representing the numerator and denominator of a fraction, return the fraction in ...

  2. Fatal signal xx (SIGSEGV) at

    Fatal signal 11问题的解决方法 http://blog.csdn.net/tankai19880619/article/details/9004619 如何定位Android NDK开发 ...

  3. 如何记录搜索引擎爬行记录php版

    写博客也有一段时间了,为什么搜索引擎迟迟不收录你的页面呢?想知道每天都有哪些蜘蛛“拜访”你的网站吗?作为一名网站长,有必要知道每天都有哪些蜘蛛爬行过你的网站,以便于了解各搜索引擎蜘蛛爬行频率,对网站进 ...

  4. 从一个数组中提取出第start位到第end位

    假设通过数组in来表示一个很大的数(in[0]表示最低bit),提取该数的第start位到第end位(计数起始位为0): #define MAX_BYTE_LEN ( 48 ) int getData ...

  5. ID

    id 编辑 身份标识号.账号.唯一编码.专属号码.工业设计.国家简称.法律词汇.通用账户.译码器.软件公司等,各类专有词汇缩写. 身份证,身份识别,是一种身份证明. 中文名 身份证,帐号,工业设计,通 ...

  6. 提高PHP开发质量的36个方法(精品)

    提高PHP开发质量的36个方法 林涛 发表于:2016-3-25 0:00 分类:26点 标签: 62次 1.不要使用相对路径 常常会看到: require_once('../../lib/some_ ...

  7. 学习微信小程序之css13关于盒子外边距的合并

  8. js中window对象的opener属性的一个坑

    2018-05-08 17:48:33 今天我编写代码碰到了一个让我纠结了很久的坑,特别想在此说一下,让其他人避免我踏过的这个坑. 这个坑就是:在我自己写的子窗口中用opener属性却获取不到父窗口的 ...

  9. appium运行报错java.net.SocketException: socket write error

    这个错我调了 快两天一点头绪没有,脚本正常跑没问题,但是就是控制台输出信息报错,没法定位问题在哪.报错如图: 虽然这个报错不影响测试结果,但是本人有强迫症,一定要查出究竟: 我的尝试: 1.那天试验, ...

  10. html-文件上传设置accept类型延时问题

       今天在做文件上传时,采用了jQuery的upload插件,使用过程中发现了一个很有意思也很头疼的问题. 上传按钮,第一次点击时瞬间就可以打开文件选择框,之后再点击则需要等待恐怖的8s以上. 百度 ...