07-Access Token

Access Token

課程目標(biāo)

  • 自定義對(duì)象
  • 構(gòu)造函數(shù)
  • 歸檔 & 接檔

接口定義

文檔地址

http://open.weibo.com/wiki/OAuth2/access_token

接口地址

https://api.weibo.com/oauth2/access_token

HTTP 請(qǐng)求方式

  • POST

請(qǐng)求參數(shù)

參數(shù) 描述
client_id 申請(qǐng)應(yīng)用時(shí)分配的AppKey
client_secret 申請(qǐng)應(yīng)用時(shí)分配的AppSecret
grant_type 請(qǐng)求的類型,填寫 authorization_code
code 調(diào)用authorize獲得的code值
redirect_uri 回調(diào)地址,需需與注冊(cè)應(yīng)用里的回調(diào)地址一致

返回?cái)?shù)據(jù)

返回值字段 字段說(shuō)明
access_token 用于調(diào)用access_token浆劲,接口獲取授權(quán)后的access token
expires_in access_token的生命周期肿孵,單位是秒數(shù)
remind_in access_token的生命周期(該參數(shù)即將廢棄验靡,開發(fā)者請(qǐng)使用expires_in)
uid 當(dāng)前授權(quán)用戶的UID

用戶賬戶模型

加載AccessToken

  • HMOAuthViewController 中增加函數(shù)加載 AccessToken
/// 加載AccessToken的方法
///
/// - parameter code: 授權(quán)碼
private func loadAccessToken(code: String){
    let urlString = "https://api.weibo.com/oauth2/access_token"

    // 定義參數(shù)
    let params = [
        "client_id": WB_APPKEY,
        "client_secret": WB_APPSECRET,
        "grant_type": "authorization_code",
        "code": code,
        "redirect_uri": WB_REDIRECTURI]

    NetworkTools.shareTools.request(.POST, url: urlString, params: params) { (result, error) -> () in
        print(result)
    }
}
  • 在獲取授權(quán)碼成功之后調(diào)用方法
// 判斷是否包含 'code=' 字樣
if let query = request.URL?.query where query.containsString("code=") {
    // 截取授權(quán)碼
    let code = query.substringFromIndex("code=".endIndex)
    print("請(qǐng)求碼:\(code)")
    // 請(qǐng)求AccessToken
    loadAccessToken(code)
}else{
    // 用戶點(diǎn)擊了取消授權(quán),直接關(guān)閉頁(yè)面
    self.close()
}

運(yùn)行測(cè)試

  • 返回錯(cuò)誤信息
Error Domain=com.alamofire.error.serialization.response Code=-1016 "Request failed: unacceptable content-type: text/plain"
  • HMNetworkTools 單例初始化時(shí)設(shè)置反序列化數(shù)據(jù)格式
static let shareTools: HMNetworkTools = {
    let tools = HMNetworkTools()
    tools.responseSerializer.acceptableContentTypes?.insert("text/plain")
    return tools
}()

定義 HMUserAcount 模型

  • Model 目錄下添加 HMUserAccount
  • 定義模型屬性
/// 用戶帳號(hào)模型
/// - see: [http://open.weibo.com/wiki/OAuth2/access_token](http://open.weibo.com/wiki/OAuth2/access_token)
class HMUserAccount: NSObject {

    // 用于調(diào)用access_token,接口獲取授權(quán)后的access token
    var access_token: String?
    // access_token的生命周期,單位是秒數(shù)
    var expires_in: NSTimeInterval = 0
    // 當(dāng)前授權(quán)用戶的UID
    var uid: String?

    init(dict: [String: AnyObject]) {
        super.init()

        setValuesForKeysWithDictionary(dict)
    }

    override func setValue(value: AnyObject?, forUndefinedKey key: String) {}
}
  • 創(chuàng)建用戶帳號(hào)模型辫愉,在 HMOAuthViewController 的網(wǎng)絡(luò)回調(diào)代碼中添加如下代碼
let account = HMUserAccount(dict: result as! [String: AnyObject])
print(account)
  • 調(diào)試模型信息

  • 與 OC 不同,如果要在 Swift 1.2 中調(diào)試模型信息将硝,需要遵守 Printable 協(xié)議恭朗,并且重寫 descriptiongetter 方法,在 Swift 2.0 中依疼,description 屬性定義在 CustomStringConvertible 協(xié)議中

override var description: String {
    let keys = ["access_token", "expires_in", "uid"]
    return dictionaryWithValuesForKeys(keys).description
}

建議description此代碼抽取到Xcode的代碼塊中痰腮,方便使用

設(shè)置過(guò)期日期

過(guò)期日期

  • 在新浪微博返回的數(shù)據(jù)中,過(guò)期日期是以當(dāng)前系統(tǒng)時(shí)間加上秒數(shù)計(jì)算的律罢,為了方便后續(xù)使用膀值,增加過(guò)期日期屬性

  • 定義屬性

/// token過(guò)期日期
var expiresDate: NSDate?
  • expires_indidSet 方法里面給 expiresDate 賦值
var expires_in: NSTimeInterval = 0 {
    didSet{
        expiresDate = NSDate(timeIntervalSinceNow: expires_in)
    }
}
  • 修改 description
let keys = ["access_token", "expires_in", "expiresDate", "uid"]

加載用戶信息

課程目標(biāo)

  • 通過(guò) AccessToken 獲取新浪微博網(wǎng)絡(luò)數(shù)據(jù)

接口定義

文檔地址

http://open.weibo.com/wiki/2/users/show

接口地址

https://api.weibo.com/2/users/show.json

HTTP 請(qǐng)求方式

  • GET

請(qǐng)求參數(shù)

參數(shù) 描述
access_token 采用OAuth授權(quán)方式為必填參數(shù),其他授權(quán)方式不需要此參數(shù),OAuth授權(quán)后獲得
uid 需要查詢的用戶ID

返回?cái)?shù)據(jù)

返回值字段 字段說(shuō)明
name 友好顯示名稱
avatar_large 用戶頭像地址(大圖)沧踏,180×180像素

測(cè)試 URL

https://api.weibo.com/2/users/show.json?access_token=2.00ml8IrF0qLZ9W5bc20850c50w9hi9&uid=5365823342

代碼實(shí)現(xiàn)

  • HMOAuthViewController 中添加 loadUserInfo 方法
/// 加載用戶數(shù)據(jù)
///
/// - parameter userAccount: 用戶賬戶模型
private func loadUserInfo(userAccount: HMUserAccount) {

    let urlString = "https://api.weibo.com/2/users/show.json"
    // 定義參數(shù)
    let params = [
        "uid": userAccount.uid!,
        "access_token": userAccount.access_token!
    ]

    HMNetworkTools.shareTools.request(.GET, url: urlString, params: params) { (result, error) -> () in
        if error != nil {
            print("請(qǐng)求失敗:\(error)")
            return
        }
        print(result)
    }
}
  • loadAccessToken 方法中請(qǐng)求accessToken成功后調(diào)用該方法
// 將返回字典轉(zhuǎn)成模型
let account = HMUserAccount(dict: result as! [String: AnyObject])
print(account)
// 請(qǐng)求用戶數(shù)據(jù)
self.loadUserInfo(account)

注意:在 Swift 中歌逢,閉包中輸入代碼的智能提示非常不好,因此新建一個(gè)函數(shù)單獨(dú)處理加載用戶功能

擴(kuò)展用戶模型

  • HMUserAccount 中增加用戶名和頭像屬性
/// 友好顯示名稱
var screen_name: String?
/// 用戶頭像地址(大圖)悦冀,180×180像素
var avatar_large: String?
  • 擴(kuò)展 description 中的屬性
override var description: String {
    let keys = ["access_token", "expires_in", "expiresDate", "uid", "screen_name", "avatar_large"]

    return dictionaryWithValuesForKeys(keys).description
}
  • 在用戶信息請(qǐng)求成功之后獲取昵稱與頭像
// 發(fā)送get請(qǐng)求
HMNetworkTools.shareTools.request(.GET, url: urlString, params: params) { (result, error) -> () in
    if error != nil {
        print("請(qǐng)求失敗:\(error)")
        return
    }
    guard let dict = result as? [String: AnyObject] else {
        return
    }
    // 設(shè)置請(qǐng)求回來(lái)的用戶昵稱和頭像
    userAccount.screen_name = dict["screen_name"] as? String
    userAccount.avatar_large = dict["avatar_large"] as? String
}

每一個(gè)令牌授權(quán)一個(gè) 特定的網(wǎng)站特定的時(shí)段內(nèi) 訪問(wèn) 特定的資源

歸檔 & 解檔

課程目標(biāo)

  • 對(duì)比 OC 的歸檔 & 解檔實(shí)現(xiàn)

  • 利用歸檔 & 解檔保存用戶信息

  • 遵守協(xié)議

class HMUserAccount: NSObject, NSCoding
  • 實(shí)現(xiàn)協(xié)議方法
// MARK: - NSCoding
/// 歸檔 - 將對(duì)象以二進(jìn)制形式保存至磁盤前被調(diào)用趋翻,與網(wǎng)絡(luò)的序列化類似
func encodeWithCoder(aCoder: NSCoder) {
    aCoder.encodeObject(access_token, forKey: "access_token")
    aCoder.encodeObject(expiresDate, forKey: "expiresDate")
    aCoder.encodeObject(uid, forKey: "uid")
    aCoder.encodeObject(name, forKey: "name")
    aCoder.encodeObject(avatar_large, forKey: "avatar_large")
}

/// 解檔 - 將二進(jìn)制數(shù)據(jù)從磁盤讀取并且轉(zhuǎn)換成對(duì)象時(shí)被調(diào)用睛琳,與網(wǎng)絡(luò)的反序列化類似
required init?(coder aDecoder: NSCoder) {
    access_token = aDecoder.decodeObjectForKey("access_token") as? String
    expiresDate = aDecoder.decodeObjectForKey("expiresDate") as? NSDate
    uid = aDecoder.decodeObjectForKey("uid") as? String
    name = aDecoder.decodeObjectForKey("name") as? String
    avatar_large = aDecoder.decodeObjectForKey("avatar_large") as? String
}
  • 實(shí)現(xiàn)將當(dāng)前對(duì)象歸檔保存的函數(shù)
/// 歸檔保存當(dāng)前對(duì)象
func saveUserAccount(){
    let path = (NSSearchPathForDirectoriesInDomains(.DocumentDirectory, .UserDomainMask, true).last! as NSString).stringByAppendingPathExtension("userAccount.archive")!
    print("歸檔保存路徑:\(path)")
    // 歸檔
    NSKeyedArchiver.archiveRootObject(self, toFile: path)
}

視圖模型

目標(biāo)一:建立視圖模型盒蟆,抽取網(wǎng)絡(luò)請(qǐng)求代碼

  • 建立用戶賬戶的視圖模型
/// 用戶賬戶視圖模型
class HMUserAccountViewModel: NSObject {

    /// 單例
    static let sharedUserAccount = HMUserAccountViewModel()

    /// 用戶賬戶模型
    var userAccount: HMUserAccount?
}
  • 抽取網(wǎng)絡(luò)請(qǐng)求代碼到視圖模型中
/// 加載AccessToken并且加載用戶信息
///
/// - parameter code:     授權(quán)碼
/// - parameter complete: 完成回調(diào)
func loadAccessToken(code: String, complete:(isSuccessed: Bool)->()){

    let urlString = "https://api.weibo.com/oauth2/access_token"

    // 定義參數(shù)
    let params = [
        "client_id": WB_APPKEY,
        "client_secret": WB_APPSECRET,
        "grant_type": "authorization_code",
        "code": code,
        "redirect_uri": WB_REDIRECTURI]

    HMNetworkTools.sharedTools.request(.POST, urlString: urlString, parameters: params) { (response, error) -> () in

        if error != nil {
            print(error)
            return
        }

        let account = HMUserAccount(dict: response as! [String: AnyObject])
        print(account)
        self.userAccount = account
        // 請(qǐng)求用戶數(shù)據(jù)
        self.loadUserInfo(account.uid!, accessToken: account.access_token!, complete: complete)
    }
}

/// 加載用戶信息
///
/// - parameter uid:         用戶uid
/// - parameter accessToken: accessToken
/// - parameter complete:    完成回調(diào)
private func loadUserInfo(uid: String, accessToken: String, complete:(isSuccessed: Bool)->()){

    let urlString = "https://api.weibo.com/2/users/show.json"
    // 組織參數(shù)
    let params = [
        "access_token": accessToken,
        "uid": uid
    ]
    // 發(fā)送請(qǐng)求
    HMNetworkTools.sharedTools.request(.GET, urlString: urlString, parameters: params) { (response, error) -> () in

        if error != nil {
            print("請(qǐng)求失敗\(error)")
            complete(isSuccessed: false)
            return
        }
        print(response)

        // 在 if let 或者 guard let 里面 使用 as 都使用 ?
        guard let responseDic = response as? [String: AnyObject] else {
            print("返回?cái)?shù)據(jù)不是一個(gè)字典")
            complete(isSuccessed: false)
            return
        }
        // 賦值用戶的頭像與昵稱
        self.userAccount?.screen_name = responseDic["screen_name"] as? String
        self.userAccount?.avatar_large = responseDic["avatar_large"] as? String

        complete(isSuccessed: true)
    }
}
  • 修改 HMOAuthViewController 中的代碼調(diào)用
// 請(qǐng)求  AccessToken以及用戶數(shù)據(jù)
HMUserAccountViewModel.sharedUserAccount.loadAccessToken(code, complete: { (isSuccessed) -> () in
    if isSuccessed {
        print("個(gè)人信息請(qǐng)求成功,跳轉(zhuǎn)界面")
    }else{
        print("個(gè)人信息請(qǐng)求失敗")
    }
})
  • 刪除控制器里面 loadAccessTokenloadUserInfo 兩個(gè)方法

控制器的代碼簡(jiǎn)單了师骗!

目標(biāo)二:加載歸檔保存的 token历等,避免重復(fù)登錄

  • 抽取歸檔代碼到 ViewModel中,并且實(shí)現(xiàn)解檔方法
// MARK: - 保存 & 加載
/// 解檔歸檔路徑
    // 歸檔與解檔的路徑
    private var archivePath: String = (NSSearchPathForDirectoriesInDomains(.DocumentDirectory, .UserDomainMask, true).last! as NSString).stringByAppendingPathComponent("useraccount.archive")

// 歸檔當(dāng)前對(duì)象
func saveAccount(account: HMUserAccount) {
    // 歸檔
    NSKeyedArchiver.archiveRootObject(account, toFile: archivePath)
}

// 解檔
func loadUserAccount() -> HMUserAccount? {
    // 解檔
    let result = NSKeyedUnarchiver.unarchiveObjectWithFile(archivePath) as? HMUserAccount
    return result
}
  • 在 AppDelegate 中添加以下代碼測(cè)試
printLog(HMUserAccountViewModel.sharedUserAccount.loadUserAccount())
  • 代碼優(yōu)化辟癌,在視圖模型初始化方法里面添加解檔的邏輯
private override init() {
    super.init()
    userAccount = loadUserAccount()
}

private init 可以避免外部調(diào)用構(gòu)造函數(shù)創(chuàng)建對(duì)象

  • view model 中添加 accessToken 計(jì)算型屬性
// 提供給外界accessToken的值
var accessToken: String? {
    return userAccount?.access_token
}
  • 添加 isExpires 判斷帳號(hào)是否過(guò)期
// 帳號(hào)是否過(guò)期
var isExpires: Bool {
    if userAccount?.expiresDate?.compare(NSDate()) == NSComparisonResult.OrderedDescending {
        return false
    }
    return true
}
  • 更改 accessToken 的 get 方法
// 訪問(wèn)令牌
var accessToken: String? {
    // 其內(nèi)部判斷 accessToken 是否有
    if !isExpires {
        return userAccount?.access_token
    }
    return nil
}
  • 修改 HMVisitorViewController 中的用戶登錄標(biāo)記
var userLogon = HMUserAccountViewModel.sharedUserAccount.accessToken != nil
  • 在視圖模型中增加 userLogon 計(jì)算型屬性
/// 用戶登錄標(biāo)記
var userLogon: Bool {
    return userAccount?.access_token != nil
}
  • 再次修改 HMVisitorViewController 中的用戶登錄標(biāo)記
var userLogon = HMUserAccountViewModel.sharedUserAccount.userLogon
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末寒屯,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子黍少,更是在濱河造成了極大的恐慌寡夹,老刑警劉巖,帶你破解...
    沈念sama閱讀 219,539評(píng)論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件厂置,死亡現(xiàn)場(chǎng)離奇詭異菩掏,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)昵济,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,594評(píng)論 3 396
  • 文/潘曉璐 我一進(jìn)店門智绸,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人访忿,你說(shuō)我怎么就攤上這事瞧栗。” “怎么了海铆?”我有些...
    開封第一講書人閱讀 165,871評(píng)論 0 356
  • 文/不壞的土叔 我叫張陵迹恐,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我卧斟,道長(zhǎng)系草,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,963評(píng)論 1 295
  • 正文 為了忘掉前任唆涝,我火速辦了婚禮找都,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘廊酣。我一直安慰自己能耻,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,984評(píng)論 6 393
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著晓猛,像睡著了一般饿幅。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上戒职,一...
    開封第一講書人閱讀 51,763評(píng)論 1 307
  • 那天栗恩,我揣著相機(jī)與錄音,去河邊找鬼洪燥。 笑死磕秤,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的捧韵。 我是一名探鬼主播市咆,決...
    沈念sama閱讀 40,468評(píng)論 3 420
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼再来!你這毒婦竟也來(lái)了蒙兰?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,357評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤芒篷,失蹤者是張志新(化名)和其女友劉穎搜变,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體针炉,經(jīng)...
    沈念sama閱讀 45,850評(píng)論 1 317
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡挠他,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 38,002評(píng)論 3 338
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了糊识。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片绩社。...
    茶點(diǎn)故事閱讀 40,144評(píng)論 1 351
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖赂苗,靈堂內(nèi)的尸體忽然破棺而出愉耙,到底是詐尸還是另有隱情,我是刑警寧澤拌滋,帶...
    沈念sama閱讀 35,823評(píng)論 5 346
  • 正文 年R本政府宣布朴沿,位于F島的核電站,受9級(jí)特大地震影響败砂,放射性物質(zhì)發(fā)生泄漏赌渣。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,483評(píng)論 3 331
  • 文/蒙蒙 一昌犹、第九天 我趴在偏房一處隱蔽的房頂上張望坚芜。 院中可真熱鬧,春花似錦斜姥、人聲如沸鸿竖。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,026評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)缚忧。三九已至悟泵,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間闪水,已是汗流浹背糕非。 一陣腳步聲響...
    開封第一講書人閱讀 33,150評(píng)論 1 272
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留球榆,地道東北人朽肥。 一個(gè)月前我還...
    沈念sama閱讀 48,415評(píng)論 3 373
  • 正文 我出身青樓响牛,卻偏偏與公主長(zhǎng)得像介陶,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,092評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容

  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見(jiàn)模式的工具(例如配置管理右钾,服務(wù)發(fā)現(xiàn),斷路器旱爆,智...
    卡卡羅2017閱讀 134,672評(píng)論 18 139
  • 發(fā)現(xiàn) 關(guān)注 消息 iOS 第三方庫(kù)舀射、插件、知名博客總結(jié) 作者大灰狼的小綿羊哥哥關(guān)注 2017.06.26 09:4...
    肇東周閱讀 12,109評(píng)論 4 62
  • 5 分享內(nèi)容到你的網(wǎng)站 上一章中怀伦,你在網(wǎng)站中構(gòu)建了用戶注冊(cè)和認(rèn)證脆烟。你學(xué)會(huì)了如何為用戶創(chuàng)建自定義的個(gè)人資料模型,并添...
    lakerszhy閱讀 1,664評(píng)論 5 16
  • 自學(xué)寫下第一篇產(chǎn)品體驗(yàn)報(bào)告房待,誠(chéng)邀互聯(lián)網(wǎng)前輩們給出意見(jiàn)~ 產(chǎn)品體驗(yàn)報(bào)告框架產(chǎn)品體驗(yàn)報(bào)告框架圖 1.市場(chǎng)分析(戰(zhàn)略層)...
    叮咚醬閱讀 2,811評(píng)論 4 10
  • 一眨眼來(lái)澳洲四年多了邢羔,輾轉(zhuǎn)了兩個(gè)城市,換了四個(gè)工作桑孩,突然想分享一下自己的一些經(jīng)驗(yàn)和感受拜鹤,或許能對(duì)新來(lái)的或者準(zhǔn)備來(lái)的...
    aussieweever閱讀 3,924評(píng)論 2 4