swift

swift中如何存储user到本地

2016-07-09  本文已影响0人  mkb2

在ios开发过程中,有个烂大街的事情就是要做user的本地存储,将来登录的时候,直接先去从本地获取,如果有,就去登录页面,否则进入其他页面,ios的都会了,我们看看用swift如何去写,可能有些方法有些不同,但是都要程序员安静的做几件事:

  • 1.建立一个模型,用于字典转模型
  • 2.存储对象的时候,对外实现两个方法,1.存储User模型 2.获取User模型
  • 3.告诉外界是否登录(用一个参数就好)
  • 4.为外界提供一个单利User的属性,避免每一次不必要的解档

1.建立一个模型,用于字典转模型

首先,我们去写一个WXUser,然后写几个属性,因为属性不一定有,所以用 ?去声明一下

    var accessToken:String?
    var expiresIn:NSNumber?//经过测试,新浪的文档中这个属性是NSNumber
    var did:String?

在oc中我们复写了description方法,然后自定义打印的样式,这里swift也可以,但是写法略有不同,description是一个属性,我们要去重写一些他的get方法,这就是区别

    //修改复写的东西
    override var description:String{
        let arr = ["accessToken","expiresIn","uid"]
       let params = self.dictionaryWithValuesForKeys(arr)
        return "\(params)"
    }

对外去声明一个字典转模型的方法,但是可能是xcode的bug,所以我们去写的时候,init()方法可能就出不来,所以我们复写一下这个方法,如果你的没毛病,就不用复写了

    override init() {
        
    }
    
    init(dict:[String:AnyObject]) {
        accessToken = dict["access_token"] as? String
        expiresIn = dict["expires_in"] as? NSNumber
        uid = dict["uid"] as? String
    }

因为对数据进行本地化,其实有几种方法分别是

  • plist(基本数据类型)
  • NSUserDefault(也是基本数据类型),
  • SQLite(存储比较多的数据,如新闻类,微博类的很多数据)
  • 归档(xcode提供的数据类型都行,但是自定义的不行,除非让对象遵守nscoding,但是swift中不用准守,只要去复写归档解档的方法就好了,单一的USER模型,选择归档比较靠谱)
  //数据本地化存储
    //归档
    func encodeWithCoder(aCoder:NSCoder)
    {
        aCoder.encodeObject(accessToken,forKey: "accessToken")
        aCoder.encodeObject(expiresIn,forKey: "expiresIn")
        aCoder.encodeObject(uid,forKey: "uid")
    }
    //解档
    init(coder deCoder:NSCoder) {
         accessToken =  deCoder.decodeObjectForKey("accessToken") as? String
         expiresIn = deCoder.decodeObjectForKey("expiresIn") as?  NSNumber
         uid = deCoder.decodeObjectForKey("uid") as? String
    }

2.存储对象的时候,对外实现两个方法

一个是保存用户,一个是获取用户,认为用类方法比较靠谱

    //save
    class  func saveUserInfo(user : WXUser)
   {
    let path = NSSearchPathForDirectoriesInDomains(NSSearchPathDirectory.DocumentDirectory, NSSearchPathDomainMask.UserDomainMask,true).last!
    //因为swift中的字符串函数比较难用,所以我们就去将swift中的字符串转变成oc的,然后使用oc的方法处理
    let filePath = (path as NSString).stringByAppendingPathComponent("user.plist")
//    print("filePath --\(filePath)")
    NSKeyedArchiver.archiveRootObject(user, toFile: filePath)
    }
    
    //getUser
   class func fetchUserInfo() -> WXUser?   //因为有可能这个第一次没有数据,所以我们获取不到,就给个问号?
    {
        let path = NSSearchPathForDirectoriesInDomains(NSSearchPathDirectory.DocumentDirectory, NSSearchPathDomainMask.UserDomainMask,true).last!
        //因为swift中的字符串函数比较难用,所以我们就去将swift中的字符串转变成oc的,然后使用oc的方法处理
        let filePath = (path as NSString).stringByAppendingPathComponent("user.plist")
       let user =  NSKeyedUnarchiver.unarchiveObjectWithFile(filePath) as? WXUser
        return user
    }

需要注意的是let filePath = (path as NSString).stringByAppendingPathComponent("user.plist")这句话,因为swift的方法特别难用,所以我们选择使用oc的方法,这里就涉及到了swift->OC (path as NSString),然后就切换过来了,这里面可以使用分类去将地址的获取分装一下,突然发现路径基本是一样的,有点恶心,所以可以拿出来作为一个变量来使用,但是感觉略有不妥,直接拿出来做一个stirng的category将来快速使用,
ctr+n->source->swift File,命名为String+Category即可,将improt Foundation改成import UIKit,然后去写对象方法就好了,我的目的是写出这样的:任何的字符串,只有后面加上一个对象方法,就可以获取到这个路径的字符串命名的文件,如"account.plist".docDir,就能获取到/Users/wangxin/Library/Developer/CoreSimulator/Devices/7D2A98A9-0339-4664-B900-0CF935D62FE5/data/Containers/Data/Application/4ED101E7-3B6C-46DA-BC1A-8CEB7E29CFC7/Documents/https:/www.wangqiujia.com/account.plist,问题是如果某君写了这个方法"https://www.wangqiujia.com/user.plist",打印出来的是有问题的,所以通过函数lastPathComponent()就可以解决
pod出分类的具体代码

import UIKit
extension String{
    /**
     *  先要使用的时候比较简便,如 “wxlo”.docunDir  "wxlo".cacheDir
     */
    func docDir() -> String
    {
       let docPath = NSSearchPathForDirectoriesInDomains(NSSearchPathDirectory.DocumentDirectory, NSSearchPathDomainMask.UserDomainMask, true).last! as NSString
        let filePath = docPath.stringByAppendingPathComponent((self as NSString).lastPathComponent)
        return filePath
    }
    //获取cache目录
    func cacheDir() -> String
    {
       let cache = NSSearchPathForDirectoriesInDomains(NSSearchPathDirectory.CachesDirectory, NSSearchPathDomainMask.UserDomainMask, true).last! as NSString
        let filePath = cache.stringByAppendingPathComponent((self as NSString).lastPathComponent)
        return filePath
    }
    //获取临时目录
    func tempDir() -> String
    {
       let temp = NSTemporaryDirectory() as NSString
        let filePath = temp.stringByAppendingPathComponent((self as NSString).lastPathComponent)
        return filePath
    }
}

下面的是修改后的存储和保存模型的方法

    //save
    class  func saveUserInfo(user : WXUser)
    {
        NSKeyedArchiver.archiveRootObject(user, toFile: "https://www.wangqiujia.com/name.plist".docDir())
    }
    
    //getUser
    class func fetchUserInfo() -> WXUser?   //因为有可能这个第一次没有数据,所以我们获取不到,就给个问号?
    {
        return NSKeyedUnarchiver.unarchiveObjectWithFile("https://www.wangqiujia.com/name.plist".docDir()) as? WXUser
    }

是不是清爽很多?

3.告诉外界是否登录(用一个参数就好)

要去告诉外界是否登录,那么最好提供一个类方法,如下

    //判断是不是已经有用户登录了
    class func isLogin() -> Bool
    {
        //从本地获取对象
        return (self.fetchUserInfo() != nil);
    }

4.为外界提供一个单利User的属性,避免每一次不必要的解档

每一次使用fetch用户的时候,都要去解档,很浪费时间,虽然计算机的解档时间很快,可以忽略不计,但是为了让我们代码优化,简单操作一下就好
对外声明一个对象static var shardUser:WXUser?,然后在fetchUser的方法中加一个判断,代码如下

    //getUser
    class func fetchUserInfo() -> WXUser?   //因为有可能这个第一次没有数据,所以我们获取不到,就给个问号?
    {
        //因为每一次去fetchUserInfo的话,都要要去解档,虽然现在数据很快,但是还是不太好,每次都这样做,所以我们可以用两种方法处理,1.单利,2.直接给定一个属性,判断他的有无即可
        if shardUser == nil
        {
            shardUser = NSKeyedUnarchiver.unarchiveObjectWithFile("https://www.wangqiujia.com/name.plist".docDir()) as? WXUser
            return shardUser
        }
        return shardUser
        
    }

测试与打印

下面是具体的调用,和打印结果


json -> Model 没有去用Mantle,MJExtension等第三方,就自己写一个简单的,用于理解字典转模型,这里swift逼迫我们使用do-catch的方法,就是这个

       do{
                    let data = try NSJSONSerialization.dataWithJSONObject(JSON!, options: NSJSONWritingOptions.PrettyPrinted);
                    let paras = try NSJSONSerialization.JSONObjectWithData(data, options: NSJSONReadingOptions.AllowFragments) as! [String:AnyObject]
                    let user = WXUser(dict: paras)
                    //通过类方法将对象存储到本地
                    WXUser.saveUserInfo(user)
                    //为了验证是否存储进来了,我们去打印一下
                    let tempsavedUser = WXUser.fetchUserInfo()
                    print("tempsavedUser  、\(tempsavedUser)")
                }catch{
                
                }

打印的结果是这样的

2016-07-09 19:58:45.128 WXWeibo[3359:96217]
tempsavedUser  、Optional(["expiresIn": 112308, "accessToken": 2.00Y28UWGCuhaJDdd1c6632e6bNqIXE, "uid": 5976894708])
上一篇下一篇

猜你喜欢

热点阅读