博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
使用keychain来存取用户名和密码
阅读量:2201 次
发布时间:2019-05-03

本文共 3309 字,大约阅读时间需要 11 分钟。

 iOSkeychain服务提供了一种安全的保存私密信息(密码,序列号,证书等)的方式。每个ios程序都有一个独立的keychain存储。从ios 3.0开始,跨程序分享keychain变得可行。

下面就使用keychain来实现存取用户名和密码。

苹果已经有现成的类封装好了keychain,KeychainItemWrapper.hKeychainItemWrapper.m文件,可以在实例里找到。

但是这里我只需要存取用户名和密码,就不用苹果提供的类了,自己写个简单的类来实现就好。

代码如下:

CHKeychain.h

[cpp] 
  1. #import <Foundation/Foundation.h>  
  2. #import <Security/Security.h>  
  3.   
  4.   
  5. @interface CHKeychain : NSObject  
  6.   
  7. + (void)save:(NSString *)service data:(id)data;  
  8. + (id)load:(NSString *)service;  
  9. + (void)delete:(NSString *)service;  
  10.   
  11.   
  12. @end  


CHKeychain.m

[cpp] 
  1. #import "CHKeychain.h"  
  2.   
  3. @implementation CHKeychain  
  4. + (NSMutableDictionary *)getKeychainQuery:(NSString *)service {  
  5.     return [NSMutableDictionary dictionaryWithObjectsAndKeys:  
  6.             (id)kSecClassGenericPassword,(id)kSecClass,  
  7.             service, (id)kSecAttrService,  
  8.             service, (id)kSecAttrAccount,  
  9.             (id)kSecAttrAccessibleAfterFirstUnlock,(id)kSecAttrAccessible,  
  10.             nil];  
  11. }  
  12.   
  13. + (void)save:(NSString *)service data:(id)data {  
  14.     //Get search dictionary  
  15.     NSMutableDictionary *keychainQuery = [self getKeychainQuery:service];  
  16.     //Delete old item before add new item  
  17.     SecItemDelete((CFDictionaryRef)keychainQuery);  
  18.     //Add new object to search dictionary(Attention:the data format)  
  19.     [keychainQuery setObject:[NSKeyedArchiver archivedDataWithRootObject:data] forKey:(id)kSecValueData];  
  20.     //Add item to keychain with the search dictionary  
  21.     SecItemAdd((CFDictionaryRef)keychainQuery, NULL);  
  22. }  
  23.   
  24. + (id)load:(NSString *)service {  
  25.     id ret = nil;  
  26.     NSMutableDictionary *keychainQuery = [self getKeychainQuery:service];  
  27.     //Configure the search setting  
  28.     //Since in our simple case we are expecting only a single attribute to be returned (the password) we can set the attribute kSecReturnData to kCFBooleanTrue  
  29.     [keychainQuery setObject:(id)kCFBooleanTrue forKey:(id)kSecReturnData];  
  30.     [keychainQuery setObject:(id)kSecMatchLimitOne forKey:(id)kSecMatchLimit];  
  31.     CFDataRef keyData = NULL;  
  32.     if (SecItemCopyMatching((CFDictionaryRef)keychainQuery, (CFTypeRef *)&keyData) == noErr) {  
  33.         @try {  
  34.             ret = [NSKeyedUnarchiver unarchiveObjectWithData:(NSData *)keyData];  
  35.         } @catch (NSException *e) {  
  36.             NSLog(@"Unarchive of %@ failed: %@", service, e);  
  37.         } @finally {  
  38. }  
  39.     }  
  40.     if (keyData)   
  41.     CFRelease(keyData);  
  42.     return ret;  
  43. }  
  44.   
  45. + (void)delete:(NSString *)service {  
  46.     NSMutableDictionary *keychainQuery = [self getKeychainQuery:service];  
  47.     SecItemDelete((CFDictionaryRef)keychainQuery);  
  48. }  
  49.   
  50.   
  51. @end  


首先需要定义几个字符串用来做key

[cpp] 
  1. NSString * const KEY_USERNAME_PASSWORD = @"com.company.app.usernamepassword";  
  2. NSString * const KEY_USERNAME = @"com.company.app.username";  
  3. NSString * const KEY_PASSWORD = @"com.company.app.password";  

把用户名和密码存入keychain

[cpp] 
  1. NSMutableDictionary *usernamepasswordKVPairs = [NSMutableDictionary dictionary];  
  2. [usernamepasswordKVPairs setObject:txtfldUsername.text forKey:KEY_USERNAME];  
  3. [usernamepasswordKVPairs setObject:txtfldPassword.text forKey:KEY_PASSWORD];  
  4. [CHKeychain save:KEY_USERNAME_PASSWORD data:usernamepasswordKVPairs];  

keychain中取出用户名和密码:

[cpp] 
  1. NSMutableDictionary *usernamepasswordKVPairs = (NSMutableDictionary *)[CHKeychain load:KEY_USERNAME_PASSWORD];  
  2. txtfldUsername.text = [usernamepasswordKVPairs objectForKey:KEY_USERNAME];  
  3. txtfldPassword.text = [usernamepasswordKVPairs objectForKey:KEY_PASSWORD];  

删除一个keychain item:

[cpp] 
  1. [CHKeychain delete:KEY_USERNAME_PASSWORD];  

这样一个简单的使用keychain存取用户名密码的功能就做好了。

另外附上一篇写得更详细的e文:

如果还有问题请看此篇文章:http://ken.bokele.com/?ArticleID=86996

转载地址:http://nqgub.baihongyu.com/

你可能感兴趣的文章
Intellij IDEA使用(十三)—— 在Intellij IDEA中配置Maven
查看>>
面试题 —— 关于main方法的十个面试题
查看>>
集成测试(一)—— 使用PHP页面请求Spring项目的Java接口数据
查看>>
使用Maven构建的简单的单模块SSM项目
查看>>
Intellij IDEA使用(十四)—— 在IDEA中创建包(package)的问题
查看>>
FastDFS集群架构配置搭建(转载)
查看>>
HTM+CSS实现立方体图片旋转展示效果
查看>>
FFmpeg 命令操作音视频
查看>>
问题:Opencv(3.1.0/3.4)找不到 /opencv2/gpu/gpu.hpp 问题
查看>>
目的:使用CUDA环境变量CUDA_VISIBLE_DEVICES来限定CUDA程序所能使用的GPU设备
查看>>
问题:Mysql中字段类型为text的值, java使用selectByExample查询为null
查看>>
程序员--学习之路--技巧
查看>>
解决问题之 MySQL慢查询日志设置
查看>>
contOS6 部署 lnmp、FTP、composer、ThinkPHP5、docker详细步骤
查看>>
TP5.1模板布局中遇到的坑,配置完不生效解决办法
查看>>
PHPstudy中遇到的坑No input file specified,以及传到linux环境下遇到的坑,模板文件不存在
查看>>
TP5.1事务操作和TP5事务回滚操作多表
查看>>
composer install或composer update 或 composer require phpoffice/phpexcel 失败解决办法
查看>>
TP5.1项目从windows的Apache服务迁移到linux的Nginx服务需要注意几点。
查看>>
win10安装软件 打开时报错 找不到 msvcp120.dll
查看>>