iOS開發——keychain的使用



通常情況下,我們用NSUserDefaults存儲數據信息,但是對於一些私密信息,比如密碼、證書等等,就需要使用更爲安全的keychain了。keychain裏保存的信息不會因App被刪除而丟失,在用戶重新安裝App後依然有效,數據還在。

使用蘋果官方發佈的KeychainItemWrapper或者SFHFKeychainUtils很方便,後來看到 iphone使用keychain來存取用戶名和密碼 一文,覺得對了解keychain有很大的幫助,於是ARC控也嘗試了一把。

需要導入Security.framework 

  1. @implementation WQKeyChain  
  2. + (NSMutableDictionary *)getKeychainQuery:(NSString *)service {  
  3. return [NSMutableDictionary dictionaryWithObjectsAndKeys:  
  4.         (__bridge_transfer id)kSecClassGenericPassword,(__bridge_transfer id)kSecClass,  
  5.         service, (__bridge_transfer id)kSecAttrService,  
  6.         service, (__bridge_transfer id)kSecAttrAccount,  
  7.         (__bridge_transfer id)kSecAttrAccessibleAfterFirstUnlock,(__bridge_transfer id)kSecAttrAccessible,  
  8.         nil];  
  9. }  
  10.   
  11. + (void)save:(NSString *)service data:(id)data {  
  12.     //Get search dictionary  
  13.     NSMutableDictionary *keychainQuery = [self getKeychainQuery:service];  
  14.     //Delete old item before add new item  
  15.     SecItemDelete((__bridge_retained CFDictionaryRef)keychainQuery);  
  16.     //Add new object to search dictionary(Attention:the data format)  
  17.     [keychainQuery setObject:[NSKeyedArchiver archivedDataWithRootObject:data] forKey:(__bridge_transfer id)kSecValueData];  
  18.     //Add item to keychain with the search dictionary  
  19.     SecItemAdd((__bridge_retained CFDictionaryRef)keychainQuery, NULL);  
  20. }  
  21.   
  22. + (id)load:(NSString *)service {  
  23.     id ret = nil;  
  24.     NSMutableDictionary *keychainQuery = [self getKeychainQuery:service];  
  25.     //Configure the search setting  
  26.     [keychainQuery setObject:(id)kCFBooleanTrue forKey:(__bridge_transfer id)kSecReturnData];  
  27.     [keychainQuery setObject:(__bridge_transfer id)kSecMatchLimitOne forKey:(__bridge_transfer id)kSecMatchLimit];  
  28.     CFDataRef keyData = NULL;  
  29.     if (SecItemCopyMatching((__bridge_retained CFDictionaryRef)keychainQuery, (CFTypeRef *)&keyData) == noErr) {  
  30.         @try {  
  31.             ret = [NSKeyedUnarchiver unarchiveObjectWithData:(__bridge_transfer NSData *)keyData];  
  32.         } @catch (NSException *e) {  
  33.             NSLog(@"Unarchive of %@ failed: %@", service, e);  
  34.         } @finally {  
  35.         }  
  36.     }  
  37.     return ret;  
  38. }  
  39.   
  40. + (void)delete:(NSString *)service {  
  41.     NSMutableDictionary *keychainQuery = [self getKeychainQuery:service];  
  42.     SecItemDelete((__bridge_retained CFDictionaryRef)keychainQuery);  
  43. }  
  44. @end  
@implementation WQKeyChain
+ (NSMutableDictionary *)getKeychainQuery:(NSString *)service {
return [NSMutableDictionary dictionaryWithObjectsAndKeys:
        (__bridge_transfer id)kSecClassGenericPassword,(__bridge_transfer id)kSecClass,
        service, (__bridge_transfer id)kSecAttrService,
        service, (__bridge_transfer id)kSecAttrAccount,
        (__bridge_transfer id)kSecAttrAccessibleAfterFirstUnlock,(__bridge_transfer 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((__bridge_retained CFDictionaryRef)keychainQuery);
    //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((__bridge_retained CFDictionaryRef)keychainQuery, NULL);
}

+ (id)load:(NSString *)service {
    id ret = nil;
    NSMutableDictionary *keychainQuery = [self getKeychainQuery:service];
    //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: %@", service, e);
        } @finally {
        }
    }
    return ret;
}

+ (void)delete:(NSString *)service {
    NSMutableDictionary *keychainQuery = [self getKeychainQuery:service];
    SecItemDelete((__bridge_retained CFDictionaryRef)keychainQuery);
}
@end

比如,保存密碼

  1. @interface WQUserDataManager : NSObject  
  2.   
  3. /** 
  4.  *  @brief  存儲密碼 
  5.  * 
  6.  *  @param  password    密碼內容 
  7.  */  
  8. +(void)savePassWord:(NSString *)password;  
  9.   
  10. /** 
  11.  *  @brief  讀取密碼 
  12.  * 
  13.  *  @return 密碼內容 
  14.  */  
  15. +(id)readPassWord;  
  16.   
  17. /** 
  18.  *  @brief  刪除密碼數據 
  19.  */  
  20. +(void)deletePassWord;  
  21.   
  22. @end  
@interface WQUserDataManager : NSObject

/**
 *	@brief	存儲密碼
 *
 *	@param 	password 	密碼內容
 */
+(void)savePassWord:(NSString *)password;

/**
 *	@brief	讀取密碼
 *
 *	@return	密碼內容
 */
+(id)readPassWord;

/**
 *	@brief	刪除密碼數據
 */
+(void)deletePassWord;

@end

  1. #import "WQUserDataManager.h"  
  2.   
  3. @implementation WQUserDataManager  
  4.   
  5. static NSString * const KEY_IN_KEYCHAIN = @"com.wuqian.app.allinfo";  
  6. static NSString * const KEY_PASSWORD = @"com.wuqian.app.password";  
  7.   
  8. +(void)savePassWord:(NSString *)password  
  9. {  
  10.     NSMutableDictionary *usernamepasswordKVPairs = [NSMutableDictionary dictionary];  
  11.     [usernamepasswordKVPairs setObject:password forKey:KEY_PASSWORD];  
  12.     [WQKeyChain save:KEY_IN_KEYCHAIN data:usernamepasswordKVPairs];  
  13. }  
  14.   
  15. +(id)readPassWord  
  16. {  
  17.     NSMutableDictionary *usernamepasswordKVPair = (NSMutableDictionary *)[WQKeyChain load:KEY_IN_KEYCHAIN];  
  18.     return [usernamepasswordKVPair objectForKey:KEY_PASSWORD];  
  19. }  
  20.   
  21. +(void)deletePassWord  
  22. {  
  23.     [WQKeyChain delete:KEY_IN_KEYCHAIN];  
  24. }  
  25. @end  
#import "WQUserDataManager.h"

@implementation WQUserDataManager

static NSString * const KEY_IN_KEYCHAIN = @"com.wuqian.app.allinfo";
static NSString * const KEY_PASSWORD = @"com.wuqian.app.password";

+(void)savePassWord:(NSString *)password
{
    NSMutableDictionary *usernamepasswordKVPairs = [NSMutableDictionary dictionary];
    [usernamepasswordKVPairs setObject:password forKey:KEY_PASSWORD];
    [WQKeyChain save:KEY_IN_KEYCHAIN data:usernamepasswordKVPairs];
}

+(id)readPassWord
{
    NSMutableDictionary *usernamepasswordKVPair = (NSMutableDictionary *)[WQKeyChain load:KEY_IN_KEYCHAIN];
    return [usernamepasswordKVPair objectForKey:KEY_PASSWORD];
}

+(void)deletePassWord
{
    [WQKeyChain delete:KEY_IN_KEYCHAIN];
}
@end

實現一個簡單的界面,把設定的密碼存起來,然後立即讀取顯示出來看看效果

  1. -(IBAction)btnAciton:(id)sender  
  2. {  
  3.     [WQUserDataManager savePassWord:self.textfield.text];  
  4.     self.label.text = [WQUserDataManager readPassWord];  
  5. }  
-(IBAction)btnAciton:(id)sender
{
    [WQUserDataManager savePassWord:self.textfield.text];
    self.label.text = [WQUserDataManager readPassWord];
}



達到了預期的效果。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章