iOS开发-通讯录同步

简介:

前段时间公司需要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

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

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这个框架的回调会调用多次,不知道是什么原因,有对这块比较了解的还希望能帮我解决这个问题。

你可能感兴趣的:(iOS开发-通讯录同步)