简介:
前段时间公司需要iOS端获取手机通讯录,然后通过MFi(后续将写一写关于BLE以及MFi的相关文章)同步到外设中,在网上找了一些文章,对于通讯录同步(因为只需要将手机联系人信息获取并同步,所以未涉及界面和在写入),现在苹果有两种框架可以实现,分别是AddressBook和Contacts。
Contacts:
在iOS9.0之后可以使用Contacts进行通讯录操作,Contacts是iOS9.0之后苹果推出的一个框架,这个框架我们看着都比较顺眼一点,因为它是使用OC写的,对于类和对象的调用相对熟悉一点。
1、首先导入Contacts.framework;
2、因为iOS10之后私有权限的限制,所以需要在plist文件中设置
Privacy - Contacts Usage Description添加相关描述。
3、准备工作到这就OK了,下面开始码代码了。
1)导入头文件 #import<Contacts/Contacts.h>
2) 对于通讯录操作,是使用CNContactStore这个类来进行操作的,创建一个对象。
3)进行授权,代码如下
self.contactStore=[[CNContactStore alloc]init];
//授权
CNAuthorizationStatus authorizationStatus=[CNContactStore authorizationStatusForEntityType:CNEntityTypeContacts];
//第一次安装应用
if(authorizationStatus==CNAuthorizationStatusNotDetermined) {
[self.contactStore requestAccessForEntityType:CNEntityTypeContacts completionHandler:^(BOOLgranted, NSError *_Nullableerror) {
if(error) {
NSLog(@"未授权");
}else{
NSLog(@"授权成功");
//获取联系人操作
[self getContacts];
}
}];
}elseif(authorizationStatus==CNAuthorizationStatusAuthorized){
NSLog(@"status--->success");
//获取联系人操作
[self getContacts];
}
- (void)getContacts{
//获取联系人获取所需联系人内容
NSArray *array=@[CNContactGivenNameKey,CNContactFamilyNameKey,CNContactPhoneNumbersKey,CNContactIdentifierKey];
CNContactFetchRequest *request=[[CNContactFetchRequest alloc]initWithKeysToFetch:array];
[self.contactStore enumerateContactsWithFetchRequest:request error:nilusingBlock:^(CNContact *_Nonnull contact,BOOL*_Nonnull stop) {
CNContact是一个通讯录的对象,因为功能只需要名字和联系人,所以只需要获取familyName,giveName,和phoneNumbers.
对于phoneContacts是一个数组,
for(CNLabeledValue *labelValueincontact.phoneNumbers) {
//获取电话号码的identifier
//NSString *phoneIdentifier=labelValue.identifier;
//获取电话号码
CNPhoneNumber *phoneNumber=labelValue.value;
NSString *phoneValue=phoneNumber.stringValue;
[numArray addObject:phoneValue];
}
})];
}
最后还有就是关于通讯录改变的代理,当通讯录改变时会调用系统的代理,在Appdelegate注册通知[[NSNotificationCenter defaultCenter]addObserver:self selector:@selector(contactChange:) name:CNContactStoreDidChangeNotification object:nil];就可以在程序存活期间监听通讯录变化(回调信息中没有指出是哪个联系人改变),但是如果程序被杀死,我测试的是重新打开程序,不会调用,这显然不是我想要的,所以我在查找通讯录的时候将联系人信息加入了数据库中,通过对数据库的查找判断是否发生了增删改。
很开心的做完了这个功能,然后提交测试,测试返回一个bug,说是安装了程序之后无限崩溃,这是什么鬼,再一问她iOS版本,竟然是8.0,竟然还有不升级的。坑死人呀,没办法了,只能改用另一个框架。
AddressBook:
这个框架是基于c语言的,所以用起来感觉没有Contacts这个框架顺手一点,不过大概的步骤还是差不多的。
1、首先导入Address.framework;
2、在plist文件中设置Privacy - Contacts Usage Description添加相关描述。
3、代码如下。
1)导入头文件 #import <AddressBook/AddressBook.h>
2) 和Contacts类似,需要先判断是否有权限访问,
ABAddressBookRefaddressBook=nil;
addressBook=ABAddressBookCreateWithOptions(NULL,NULL);
if(ABAddressBookGetAuthorizationStatus()==kABAuthorizationStatusDenied) {
NSLog(@"kABAuthorizationStatusDenied");
}elseif(ABAddressBookGetAuthorizationStatus()==kABAuthorizationStatusAuthorized){
NSLog(@"kABAuthorizationStatusAuthorized");
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT,0), ^{
//获取通讯录操作
[selfgetAddressBook:addressBook];
});
}else{
NSLog(@"Not determined");
ABAddressBookRequestAccessWithCompletion(ABAddressBookCreateWithOptions(NULL,nil), ^(boolgranted,CFErrorReferror) {
if(granted) {
NSLog(@"Just authorized");
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT,0), ^{
//获取通讯录操作
[selfgetAddressBook:addressBook];
});
}else{
NSLog(@"Just deieny");
}
});
}
- (void)getAddressBook:(ABAdressBookRef)addressBook
{
//按照添加时间请求所有的联系人
//CFArrayRef contants=ABAddressBookCopyArrayOfAllPeople(addressBook);
//按照排序规则请求所有的联系人
ABRecordRefrecordRef =ABAddressBookCopyDefaultSource(addressBook);
//这里的firstName是按照姓名属性中的firstName来排序的,并不是按照全称来排序
CFArrayRefcontants =ABAddressBookCopyArrayOfAllPeopleInSourceWithSortOrdering(addressBook, recordRef,kABPersonSortByFirstName);
for(inti=0; i
//获得people
ABRecordRefrecordRef=CFArrayGetValueAtIndex(contants, i);
NSString*firstName=(__bridgeNSString*)(ABRecordCopyValue(recordRef,kABPersonFirstNameProperty));
NSString*lastName=(__bridgeNSString*)(ABRecordCopyValue(recordRef,kABPersonLastNameProperty));
//kABPersonSocialProfileUserIdentifierKey
intcontacts_id=ABRecordGetRecordID(recordRef);
ABMultiValueRefvalues=ABRecordCopyValue(recordRef,kABPersonPhoneProperty);
NSMutableArray*phoneArray=[NSMutableArrayarray];
for(inti=0; i
[phoneArrayaddObject:(__bridgeNSString*)ABMultiValueCopyValueAtIndex(values, i)];
}
NSString*phoneString=[[NSStringalloc]initWithData:[NSJSONSerializationdataWithJSONObject:phoneArrayoptions:0error:nil]encoding:NSUTF8StringEncoding];
if(!firstName) {
firstName=@"";
}
if(!lastName) {
lastName=@"";
}
NSLog(@"%d------%@------%@",contacts_id,[lastNamestringByAppendingString:firstName],phoneString);
[selfupdateCollectListWithIdentifier:[NSStringstringWithFormat:@"%d",contacts_id]Name:[lastNamestringByAppendingString:firstName]phone:phoneString];
}
}
当通讯录改变时,可以通过通知来获取
ABAddressBookRegisterExternalChangeCallback(_addressBookRef,addressBookChanged, (__bridgevoid*)(self));
(在c的函数中调用OC的方法,staticAppDelegate*selfClass=nil; selfClass=self ,使用selfClass调用)
总结:
通过对这两个框架的使用,感觉还是Contacts这个框架更简洁明了,还有一个遗留问题就是,当通讯录改变时,使用AddressBook这个框架的回调会调用多次,不知道是什么原因,有对这块比较了解的还希望能帮我解决这个问题。