iOS开发 - 继udid,Mac地址等一系列唯一标识无效后,如何用KeyChain来实现设备唯一性

简介: iOS开发 - 继udid,Mac地址等一系列唯一标识无效后,如何用KeyChain来实现设备唯一性

苹果本着为用户安全考虑的初衷导致UDID和Mac地址相继阵亡,IMEI也不例外,为了设备的唯一性,一代代开发者绞尽脑汁,后来KeyChain被他们瞄上了,终于可以继续判别社别的唯一性。


原理是利用UUID,有人说,UUID是非唯一的,很容易变化,对,没错,但是UUID绝对不会重复对吧?所以UUID我只需要获取一次,然后存入KeyChain,即使App被删除了,KeyChain中的UUID也不会被删除,如果你们公司有两款App,完全可以通过存入的key值读取出另一个App的UUID,做一些简单的交互。


具体的实现方法,网上有很多种方法,很少有全套的使用方法,博主这里重新总结了下:

先说说获取UUID的方法:

 CFUUIDRef myUUID = CFUUIDCreate( nil );
    CFStringRef uuidString = CFUUIDCreateString( nil, myUUID );
    NSString * result = (NSString *)CFBridgingRelease(CFStringCreateCopy( NULL, uuidString));

关于实际的存储,都是属于套路的代码了,没什么可读性,直接贴出来了:


有两个管理类:

#import <Foundation/Foundation.h>
@interface LHUUIDManager : NSObject
+(void)saveUUID; //保存UUID
+(id)readUUID; //获取UUID
+(void)deleteUUID; //删除
@end
#import "LHUUIDManager.h"
#import "LHKeyChain.h"
@implementation LHUUIDManager : NSObject 
static NSString * const KEY_IN_KEYCHAIN = @"KEY_IN_KEYCHAIN";
static NSString * const KEY_IN_UUID = @"KEY_IN_UUID";
+(void)saveUUID
{
    NSMutableDictionary *myUUIDDic = [NSMutableDictionary dictionary];
    [myUUIDDic setObject:[self creatUUID] forKey:KEY_IN_UUID];
    [LHKeyChain save:KEY_IN_KEYCHAIN data:myUUIDDic];
}
+(id)readUUID
{
    NSMutableDictionary *myUUIDDic = (NSMutableDictionary *)[LHKeyChain read:KEY_IN_KEYCHAIN];
    return [myUUIDDic objectForKey:KEY_IN_UUID];
}
+(void)deleteUUID
{
    [LHKeyChain lhDelete:KEY_IN_KEYCHAIN];
}
+(NSString *)creatUUID
{
    CFUUIDRef myUUID = CFUUIDCreate( nil );
    CFStringRef uuidString = CFUUIDCreateString( nil, myUUID );
    NSString * result = (NSString *)CFBridgingRelease(CFStringCreateCopy( NULL, uuidString));
    return result;
}
@end
#import <Foundation/Foundation.h>
@interface LHKeyChain : NSObject
+ (NSMutableDictionary *)getKeychain:(NSString *)keyChain ;
+ (void)save:(NSString *)keyChain data:(id)data;
+ (id)read:(NSString *)keyChain;
+ (void)lhDelete:(NSString *)keyChain;
@end
#import "LHKeyChain.h"
@implementation LHKeyChain
+ (NSMutableDictionary *)getKeychain:(NSString *)keyChain {
    return [NSMutableDictionary dictionaryWithObjectsAndKeys:
            (__bridge_transfer id)kSecClassGenericPassword,(__bridge_transfer id)kSecClass,
            keyChain, (__bridge_transfer id)kSecAttrService,
            keyChain, (__bridge_transfer id)kSecAttrAccount,
            (__bridge_transfer id)kSecAttrAccessibleAfterFirstUnlock,(__bridge_transfer id)kSecAttrAccessible,
            nil];
}
+ (void)save:(NSString *)keyChain data:(id)data {
    //Get search dictionary
    NSMutableDictionary *keychainQuery = [self getKeychain:keyChain];
    //Delete old item before add new item
    CFDictionaryRef aRef = (__bridge_retained CFDictionaryRef)keychainQuery;
    SecItemDelete(aRef);
    //Add new object to search dictionary(Attention:the data format)
    [keychainQuery setObject:[NSKeyedArchiver archivedDataWithRootObject:data] forKey:(__bridge_transfer id)kSecValueData];
    //Add item to keychain with the search dictionary
    SecItemAdd(aRef, NULL);
}
+ (id)read:(NSString *)keyChain {
    id ret = nil;
    NSMutableDictionary *keychainQuery = [self getKeychain:keyChain];
    //Configure the search setting
    [keychainQuery setObject:(id)kCFBooleanTrue forKey:(__bridge_transfer id)kSecReturnData];
    [keychainQuery setObject:(__bridge_transfer id)kSecMatchLimitOne forKey:(__bridge_transfer id)kSecMatchLimit];
    CFDataRef keyData = NULL;
    if (SecItemCopyMatching((__bridge_retained CFDictionaryRef)keychainQuery, (CFTypeRef *)&keyData) == noErr) {
        @try {
            ret = [NSKeyedUnarchiver unarchiveObjectWithData:(__bridge_transfer NSData *)keyData];
        } @catch (NSException *e) {
            NSLog(@"Unarchive of %@ failed: %@", keyChain, e);
        } @finally {
        }
    }
    return ret;
}
+ (void)lhDelete:(NSString *)keyChain {
    NSMutableDictionary *keychainQuery = [self getKeychain:keyChain];
    SecItemDelete((__bridge_retained CFDictionaryRef)keychainQuery);
}
@end

到这里是所有关键代码,放上下载地址:点击下载


最后,忘了件重要的事情,要在xcode中启用KeyChain:

1.png

当显示为ON的时候才能使用KeyChain。

目录
相关文章
|
10天前
|
开发框架 前端开发 Android开发
安卓与iOS开发中的跨平台策略
在移动应用开发的战场上,安卓和iOS两大阵营各据一方。随着技术的演进,跨平台开发框架成为开发者的新宠,旨在实现一次编码、多平台部署的梦想。本文将探讨跨平台开发的优势与挑战,并分享实用的开发技巧,帮助开发者在安卓和iOS的世界中游刃有余。
|
1天前
|
Java 调度 Android开发
安卓与iOS开发中的线程管理差异解析
在移动应用开发的广阔天地中,安卓和iOS两大平台各自拥有独特的魅力。如同东西方文化的差异,它们在处理多线程任务时也展现出不同的哲学。本文将带你穿梭于这两个平台之间,比较它们在线程管理上的核心理念、实现方式及性能考量,助你成为跨平台的编程高手。
|
3天前
|
存储 前端开发 Swift
探索iOS开发:从新手到专家的旅程
本文将带您领略iOS开发的奇妙之旅,从基础概念的理解到高级技巧的掌握,逐步深入iOS的世界。文章不仅分享技术知识,还鼓励读者在编程之路上保持好奇心和创新精神,实现个人成长与技术突破。
|
14天前
|
设计模式 Swift iOS开发
探索iOS开发:从基础到高级,打造你的第一款App
【10月更文挑战第40天】在这个数字时代,掌握移动应用开发已成为许多技术爱好者的梦想。本文将带你走进iOS开发的世界,从最基础的概念出发,逐步深入到高级功能实现,最终指导你完成自己的第一款App。无论你是编程新手还是有志于扩展技能的开发者,这篇文章都将为你提供一条清晰的学习路径。让我们一起开始这段旅程吧!
|
15天前
|
iOS开发 开发者
探索iOS开发中的SwiftUI框架
【10月更文挑战第39天】在苹果的生态系统中,SwiftUI框架以其声明式语法和易用性成为开发者的新宠。本文将深入SwiftUI的核心概念,通过实际案例展示如何利用这一框架快速构建用户界面,并探讨其对iOS应用开发流程的影响。
|
存储 iOS开发 数据安全/隐私保护
ios设备唯一标识获取策略
In iOS 7 and later, if you ask for the MAC address of an iOS device, the system returns the value 02:00:00:00:00:00.
963 0
|
2月前
|
Java Android开发 Swift
安卓与iOS开发对比:平台选择对项目成功的影响
【10月更文挑战第4天】在移动应用开发的世界中,选择合适的平台是至关重要的。本文将深入探讨安卓和iOS两大主流平台的开发环境、用户基础、市场份额和开发成本等方面的差异,并分析这些差异如何影响项目的最终成果。通过比较这两个平台的优势与挑战,开发者可以更好地决定哪个平台更适合他们的项目需求。
118 1
|
2月前
|
设计模式 安全 Swift
探索iOS开发:打造你的第一个天气应用
【9月更文挑战第36天】在这篇文章中,我们将一起踏上iOS开发的旅程,从零开始构建一个简单的天气应用。文章将通过通俗易懂的语言,引导你理解iOS开发的基本概念,掌握Swift语言的核心语法,并逐步实现一个具有实际功能的天气应用。我们将遵循“学中做,做中学”的原则,让理论知识和实践操作紧密结合,确保学习过程既高效又有趣。无论你是编程新手还是希望拓展技能的开发者,这篇文章都将为你打开一扇通往iOS开发世界的大门。
|
18天前
|
安全 数据处理 Swift
深入探索iOS开发中的Swift语言特性
本文旨在为开发者提供对Swift语言在iOS平台开发的深度理解,涵盖从基础语法到高级特性的全面分析。通过具体案例和代码示例,揭示Swift如何简化编程过程、提高代码效率,并促进iOS应用的创新。文章不仅适合初学者作为入门指南,也适合有经验的开发者深化对Swift语言的认识。
38 9
|
17天前
|
Android开发 Swift iOS开发
探索安卓与iOS开发的差异和挑战
【10月更文挑战第37天】在移动应用开发的广阔舞台上,安卓和iOS这两大操作系统扮演着主角。它们各自拥有独特的特性、优势以及面临的开发挑战。本文将深入探讨这两个平台在开发过程中的主要差异,从编程语言到用户界面设计,再到市场分布的不同影响,旨在为开发者提供一个全面的视角,帮助他们更好地理解并应对在不同平台上进行应用开发时可能遇到的难题和机遇。