SSKeychain简单实战-获取唯一标识符和密码自动填充功能

前段时间由于项目进度紧, 一直没有倒开空来照顾豺狼的, 写博客是学习的一个非常好的途径, 豺狼的技术并不好, 所以博客内容也没有特别高端, 只是把一些学到的知识在分享的过程中整合吸收而已, 能帮到各位看官当然就更好了...

SSKeychain简单实战-获取唯一标识符和密码自动填充功能_第1张图片

日常项目中我们总会遇到获取DeviceId的需求, 比如定点推送消息, 未登录收藏等功能, 这里吐槽下iOS对比安卓繁琐的获取方法,Keychain具体的解释就不赘述了, 先摘录一下网上关于获取设备唯一标识符的方法:

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支持库

SSKeychain简单实战-获取唯一标识符和密码自动填充功能_第2张图片
Security.framework

然后在TARGETS-->Capabilities-->Keychain Sharing打开, 会提示登录开发者账号, 选择证书, 打开之后会生成一个.entitlements文件

.entitlements

其中的AppIdentifierPrefix是的你的App IDs的Prefix

SSKeychain简单实战-获取唯一标识符和密码自动填充功能_第3张图片
AppIdentifierPrefix

这样准备工作就完成了, 之后我们把UI设计好, 具体功能如下图. label用来显示设备唯一标识符, 删除APP之后并不会改变. 第一次输入用户名和密码之后, 再次输入用户名会自动填充密码. 点击清空按钮, 会把账号对应的本地密码删除. 点击查询会在控制台输出APP内保存的所有账号信息.

SSKeychain简单实战-获取唯一标识符和密码自动填充功能_第4张图片
UI

关于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的基本用法, 希望对各位多有帮助, 如果喜欢请点个赞鼓励一下豺狼, 谢谢.

SSKeychain简单实战-获取唯一标识符和密码自动填充功能_第5张图片
关注豺狼的订阅号, 更新的新文章会第一时间收到通知~

你可能感兴趣的:(SSKeychain简单实战-获取唯一标识符和密码自动填充功能)