美文网首页IOS丶信步沧桑IOS面试裤
SSKeychain简单实战-获取唯一标识符和密码自动填充功能

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

作者: 赤脊山的豺狼人 | 来源:发表于2016-05-17 16:25 被阅读1526次

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

    日常项目中我们总会遇到获取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支持库

    Security.framework

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

    .entitlements

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

    AppIdentifierPrefix

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

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

    关注豺狼的订阅号, 更新的新文章会第一时间收到通知~

    相关文章

      网友评论

      • liyaoyao:楼主,SAMKeychain.h 文件中最后一行 的 #import <SAMKeychain/SAMKeychainQuery.h>报错你是怎么弄的 ?
        赤脊山的豺狼人:@谢谢生活 cocoapod 导入没问题的 你是不是直接拖拽的?
        谢谢生活:我靠 我也是。

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

      本文链接:https://www.haomeiwen.com/subject/qlsirttx.html