前段时间由于项目进度紧, 一直没有倒开空来照顾豺狼的, 写博客是学习的一个非常好的途径, 豺狼的技术并不好, 所以博客内容也没有特别高端, 只是把一些学到的知识在分享的过程中整合吸收而已, 能帮到各位看官当然就更好了...
日常项目中我们总会遇到获取DeviceId的需求, 比如定点推送消息, 未登录收藏等功能, Keychain具体的解释就不赘述了, 先摘录一下网上关于获取设备唯一标识符的方法:这里吐槽下iOS对比安卓繁琐的获取方法,
iOS中获取设备唯一标示符的方法一直随版本的更新而变化。iOS 2.0版本以后UIDevice提供一个获取设备唯一标识符的方法uniqueIdentifier,通过该方法我们可以获取设备的序列号,这个也是目前为止唯一可以确认唯一的标示符。好景不长,因为该唯一标识符与手机一一对应,苹果觉得可能会泄露用户隐私,所以在 iOS 5.0之后该方法就被废弃掉了;iOS 6.0系统新增了两个用于替换uniqueIdentifier的接口,分别是:identifierForVendor,advertisingIdentifier,但这两个接口会在应用重新安装时改变数值,并不是唯一的标示符,所以开发者改为使用WiFi的mac地址来取代;iOS 7中苹果又封杀mac地址,所以开发者再次改变思路使用KeyChain来保存获取到的UDID,这样以后即使APP删了再装回来,也可以从KeyChain中读取回来。
由此我们知道在iOS7之后的策略是将每次重装都会改变的UUID保存到Keychain中, 之后每次使用就从Keychain中取出来, 由于Keychain不会随着APP的删除而清空, 所以达到了唯一标识符的作用. 脑洞大开下, 会不会出现超小几率的UUID重复呢?
同样的原理, 某些APP在卸载重装之后依然可以记住你的密码就是这样实现的, 比如招商银行的手机客户端, 我猜想就是根据唯一标识符来记录具体的图像密码实现的, 而一些APP的密码自动填充功能也是同样的实现方法. 理论说到这里, 具体的还是要到实践中去体验.非广告
-->Demo<--
首先豺狼看了下苹果关于Keychain的Demo, 发现竟然是MRC的代码, 有警告听说还有内存泄露...现在向大家推荐良心三方库Sam Soffes的SSKeychain, SSKeychain的功能很简单, 想要详细了解的可以看SSKeychain Documentation.
将SSKeychain导入工程后, 需要添加Security.framework
支持库
然后在TARGETS
-->Capabilities
-->Keychain Sharing
打开, 会提示登录开发者账号, 选择证书, 打开之后会生成一个.entitlements
文件
其中的AppIdentifierPrefix
是的你的App IDs的Prefix
这样准备工作就完成了, 之后我们把UI设计好, 具体功能如下图. label
用来显示设备唯一标识符, 删除APP之后并不会改变. 第一次输入用户名和密码之后, 再次输入用户名会自动填充密码. 点击清空按钮, 会把账号对应的本地密码删除. 点击查询会在控制台输出APP内保存的所有账号信息.
关于SSKeychain, 这里用到了如下几个方法:
// 保存对应账户密码
+ (BOOL)setPassword:(NSString *)password forService:(NSString *)serviceName account:(NSString *)account;
// 提取对应账户密码
+ (NSString *)passwordForService:(NSString *)serviceName account:(NSString *)account;
// 删除对应账户密码
+ (BOOL)deletePasswordForService:(NSString *)serviceName account:(NSString *)account;
// 获取APP下所有账户
+ (NSArray *)accountsForService:(NSString *)serviceName;
详细的代码贴出来, 推荐下载Demo跑起来看看, 并不难, 都是对Keychain的最基本的用法
#import "ViewController.h"
#import "SSKeychain.h"
static NSString *kKeychainService = @"com.xuhaoran.keychaindemo";
static NSString *kKeychainDeviceId = @"KeychainDeviceId";
@interface ViewController ()
@property (weak, nonatomic) IBOutlet UITextField *accountTextField;
@property (weak, nonatomic) IBOutlet UITextField *passwordTextField;
@property (weak, nonatomic) IBOutlet UILabel *uuidLabel;
@end
@implementation ViewController
#pragma mark - life cycle
- (void)viewDidLoad {
[super viewDidLoad];
self.uuidLabel.text = [NSString stringWithFormat:@"设备号:\n%@", [self getDeviceId]];
}
- (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event {
[self.accountTextField resignFirstResponder];
[self.passwordTextField resignFirstResponder];
}
#pragma mark - UITextFieldDelegate
- (void)textFieldDidBeginEditing:(UITextField *)textField {
if (textField == self.accountTextField) {
self.accountTextField.text = nil;
self.passwordTextField.text = nil;
}
}
- (BOOL)textFieldShouldReturn:(UITextField *)textField {
if (textField == self.accountTextField) {
[self.passwordTextField becomeFirstResponder];
}
else if (textField == self.passwordTextField) {
[self loginAction:nil];
}
return YES;
}
- (void)textFieldDidEndEditing:(UITextField *)textField {
if (textField == self.accountTextField) {
// 提取本地密码
NSString *localPassword = [SSKeychain passwordForService:kKeychainService account:textField.text];
if (localPassword) {
self.passwordTextField.text = localPassword;
}
}
}
#pragma mark - responce action
- (IBAction)loginAction:(id)sender {
if (!self.accountTextField.text || !self.passwordTextField.text) {
[self showMsg:@"输入账号和密码!"];
return;
}
[self showMsg:[NSString stringWithFormat:@"账户名:%@\n密码:%@", self.accountTextField.text, self.passwordTextField.text]];
// 保存账号密码
[SSKeychain setPassword:self.passwordTextField.text forService:kKeychainService account:self.accountTextField.text];
}
- (IBAction)clearAction:(id)sender {
if (!self.accountTextField.text) {
return;
}
if ([SSKeychain deletePasswordForService:kKeychainService account:self.accountTextField.text]) {
[self showMsg:[NSString stringWithFormat:@"账户%@的密码已清空!", self.accountTextField.text]];
self.passwordTextField.text = nil;
}
else {
[self showMsg:@"删除失败了"];
}
}
- (IBAction)searchAllAction:(id)sender {
NSArray *accounts = [SSKeychain accountsForService:kKeychainService];
NSLog(@"accounts:\n%@", accounts);
[self showMsg:@"看下控制台输出"];
}
#pragma mark - private method
- (NSString *)getDeviceId {
// 读取设备号
NSString *localDeviceId = [SSKeychain passwordForService:kKeychainService account:kKeychainDeviceId];
if (!localDeviceId) {
// 保存设备号
CFUUIDRef deviceId = CFUUIDCreate(NULL);
assert(deviceId != NULL);
CFStringRef deviceIdStr = CFUUIDCreateString(NULL, deviceId);
[SSKeychain setPassword:[NSString stringWithFormat:@"%@", deviceIdStr] forService:kKeychainService account:kKeychainDeviceId];
localDeviceId = [NSString stringWithFormat:@"%@", deviceIdStr];
}
return localDeviceId;
}
- (void)showMsg:(NSString *)msg {
UIAlertController *alert = [UIAlertController alertControllerWithTitle:@"Tip" message:msg preferredStyle:UIAlertControllerStyleAlert];
UIAlertAction *cancel = [UIAlertAction actionWithTitle:@"OK" style:UIAlertActionStyleCancel handler:nil];
[alert addAction:cancel];
[self showViewController:alert sender:nil];
}
@end
因为看网上大多都是简单的方法介绍, 所以豺狼就写了一个实战型的Demo, 虽说是实战, 但也都是SSKeychain的基本用法, 希望对各位多有帮助, 如果喜欢请点个赞鼓励一下豺狼, 谢谢.