首先我們知道獲取設備標識的三種方法
/** 卸載應用重新安裝后會不一致*/ + (NSString *)getUUID{ CFUUIDRef uuid = CFUUIDCreate(NULL); NSString *UUID = (__bridge_transfer NSString *)CFUUIDCreateString(NULL, uuid); CFRelease(uuid); return UUID; } /** 卸載應用重新安裝后會不一致*/ + (NSString *)getUUID{ return [UIDevice currentDevice].identifierForVendor.UUIDString;; } /** 不會因為應用卸載改變 * 但是用戶在設置-隱私-廣告里面限制廣告跟蹤后會變成@"00000000-0000-0000-0000-000000000000" * 重新打開后會變成另一個,還原廣告標識符也會變 */ + (NSString *)getUUID{ return [[[ASIdentifierManager sharedManager] advertisingIdentifier] UUIDString]; }
我們看這三個方法是不是都不穩妥,可能都會改變,那我們得想個辦法把值存好了,存到鑰匙串
首先打開鑰匙串權限
然后是核心存儲代碼
#import <Foundation/Foundation.h> @interface KeyChainStore : NSObject + (void)save:(NSString*)service data:(id)data; + (id)load:(NSString*)service; + (void)deleteKeyData:(NSString*)service; @end
#import "KeyChainStore.h" @implementation KeyChainStore + (NSMutableDictionary*)getKeychainQuery:(NSString*)service { return[NSMutableDictionary dictionaryWithObjectsAndKeys: (id)kSecClassGenericPassword,(id)kSecClass, service,(id)kSecAttrService, service,(id)kSecAttrAccount, (id)kSecAttrAccessibleAfterFirstUnlock,(id)kSecAttrAccessible, nil]; } + (void)save:(NSString*)service data:(id)data{ //Get search dictionary NSMutableDictionary*keychainQuery = [self getKeychainQuery:service]; //Delete old item before add new item SecItemDelete((CFDictionaryRef)keychainQuery); //Add new object to searchdictionary(Attention:the data format) [keychainQuery setObject:[NSKeyedArchiver archivedDataWithRootObject:data]forKey:(id)kSecValueData]; //Add item to keychain with the searchdictionary SecItemAdd((CFDictionaryRef)keychainQuery,NULL); } + (id)load:(NSString*)service { id ret =nil; NSMutableDictionary*keychainQuery = [self getKeychainQuery:service]; //Configure the search setting //Since in our simple case we areexpecting only a single attribute to be returned (the password) wecan set the attribute kSecReturnData to kCFBooleanTrue [keychainQuery setObject:(id)kCFBooleanTrue forKey:(id)kSecReturnData]; [keychainQuery setObject:(id)kSecMatchLimitOne forKey:(id)kSecMatchLimit]; CFDataRef keyData =NULL; if(SecItemCopyMatching((CFDictionaryRef)keychainQuery,(CFTypeRef*)&keyData) ==noErr){ @try{ ret =[NSKeyedUnarchiver unarchiveObjectWithData:(__bridge NSData*)keyData]; }@catch(NSException *e) { NSLog(@"Unarchiveof %@ failed: %@",service, e); }@finally{ } } if(keyData) CFRelease(keyData); return ret; } + (void)deleteKeyData:(NSString*)service { NSMutableDictionary*keychainQuery = [self getKeychainQuery:service]; SecItemDelete((CFDictionaryRef)keychainQuery); } @end
調用代碼
/** 獲取UUID*/ + (NSString *)getUUIDByKeyChain{ // 這個key的前綴最好是你的BundleID NSString*strUUID = (NSString*)[KeyChainStore load:@"com.mycompany.myapp.usernamepassword"]; //首次執行該方法時,uuid為空 if([strUUID isEqualToString:@""]|| !strUUID) { // 獲取UUID 這個是要引入<AdSupport/AdSupport.h>的 strUUID = [[[ASIdentifierManager sharedManager] advertisingIdentifier] UUIDString]; if(strUUID.length ==0 || [strUUID isEqualToString:@"00000000-0000-0000-0000-000000000000"]) { //生成一個uuid的方法 CFUUIDRef uuidRef= CFUUIDCreate(kCFAllocatorDefault); strUUID = (NSString*)CFBridgingRelease(CFUUIDCreateString(kCFAllocatorDefault,uuidRef)); CFRelease(uuidRef); } //將該uuid保存到keychain [KeyChainStore save:@"com.mycompany.myapp.usernamepassword" data:strUUID]; } return strUUID; }
借鑒鏈接 https://blog.csdn.net/lg767201403/article/details/80568798