基于第三方微信授权登录的iOS代码分析

来源:互联网 发布:51单片机复位电路详细 编辑:程序博客网 时间:2024/05/17 22:45

http://blog.csdn.net/mapboo/article/details/39475015


基于第三方的微信授权登录是之前笔者博文《QQ和新浪微博授权登录》的姊妹篇,微信已经深入到每一个APP的缝隙,最常用的莫过分享和登录了,接下来就以代码的形式来展开微信登录的相关说明,至于原理级别的oauth2.0认证体系请参考微信开放平台的相关说明和图示 https://open.weixin.qq.com/


三,微信登录授权开发:

1,到微信开发平台注册相关APP,现在是等待审核成功后才能获取到对应的key和secret;获取成功后需要单独申请开通登录和支付接口,如图


2,和QQ类似,需要填写Url Schemes,如demo中的wxd930ea5d5a258f4f ,然后引入相应framework;

3,在AppDelegate中注册和实现授权后的回调函数,代码如下:

[objc] view plaincopyprint?在CODE上查看代码片派生到我的代码片
  1. //向微信注册  
  2.   [WXApi registerApp:kWXAPP_ID withDescription:@"weixin"];  
[objc] view plaincopyprint?在CODE上查看代码片派生到我的代码片
  1. //授权后回调 <span style="font-family: Arial;">WXApiDelegate</span>  
  2. -(void)onResp:(BaseReq *)resp  
  3. {  
  4.    /* 
  5.     ErrCode ERR_OK = 0(用户同意) 
  6.     ERR_AUTH_DENIED = -4(用户拒绝授权) 
  7.     ERR_USER_CANCEL = -2(用户取消) 
  8.     code    用户换取access_token的code,仅在ErrCode为0时有效 
  9.     state   第三方程序发送时用来标识其请求的唯一性的标志,由第三方程序调用sendReq时传入,由微信终端回传,state字符串长度不能超过1K 
  10.     lang    微信客户端当前语言 
  11.     country 微信用户当前国家信息 
  12.     */      
  13.     SendAuthResp *aresp = (SendAuthResp *)resp;  
  14.     if (aresp.errCode== 0) {  
  15.         NSString *code = aresp.code;  
  16.         NSDictionary *dic = @{@"code":code};  
  17.     }  
  18. }  

[objc] view plaincopyprint?在CODE上查看代码片派生到我的代码片
  1. //和QQ,新浪并列回调句柄  
[objc] view plaincopyprint?在CODE上查看代码片派生到我的代码片
  1. - (BOOL)application:(UIApplication *)application openURL:(NSURL *)url sourceApplication:(NSString *)sourceApplication annotation:(id)annotation  
  2. {  
  3.     return [TencentOAuth HandleOpenURL:url] ||  
  4.     [WeiboSDK handleOpenURL:url delegate:self] ||  
  5.     [WXApi handleOpenURL:url delegate:self];;  
  6. }  
  7.   
  8. - (BOOL)application:(UIApplication *)application handleOpenURL:(NSURL *)url  
  9. {  
  10.     return [TencentOAuth HandleOpenURL:url] ||  
  11.     [WeiboSDK handleOpenURL:url delegate:self] ||  
  12.     [WXApi handleOpenURL:url delegate:self];;  
  13. }  

4,微信登录授权比较复杂,相比QQ,新浪多了几步,简单说就是需要三步,第一步,获取code,这个用来获取token,第二步,就是带上code获取token,第三步,根据第二步获取的token和openid来获取用户的相关信息;

下面用代码来实现:

第一步:code

[objc] view plaincopyprint?在CODE上查看代码片派生到我的代码片
  1. - (IBAction)weixinLogin:(id)sender  
  2. {  
  3.     [self sendAuthRequest];  
  4. }  
  5.   
  6. -(void)sendAuthRequest  
  7. {  
  8.     SendAuthReq* req =[[SendAuthReq alloc ] init];  
  9.     req.scope = @"snsapi_userinfo,snsapi_base";  
  10.     req.state = @"0744" ;  
  11.     [WXApi sendReq:req];  
  12. }  
这里获取后会调用之前在AppDelegate里面的对应oauthResp回调,获得得到的code。

第二步:token和openid

[objc] view plaincopyprint?在CODE上查看代码片派生到我的代码片
  1. -(void)getAccess_token  
  2. {  
  3.     //https://api.weixin.qq.com/sns/oauth2/access_token?appid=APPID&secret=SECRET&code=CODE&grant_type=authorization_code  
  4.       
  5.     NSString *url =[NSString stringWithFormat:@"https://api.weixin.qq.com/sns/oauth2/access_token?appid=%@&secret=%@&code=%@&grant_type=authorization_code",kWXAPP_ID,kWXAPP_SECRET,self.wxCode.text];  
  6.       
  7.     dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{  
  8.         NSURL *zoneUrl = [NSURL URLWithString:url];  
  9.         NSString *zoneStr = [NSString stringWithContentsOfURL:zoneUrl encoding:NSUTF8StringEncoding error:nil];  
  10.         NSData *data = [zoneStr dataUsingEncoding:NSUTF8StringEncoding];  
  11.         dispatch_async(dispatch_get_main_queue(), ^{  
  12.             if (data) {  
  13.                 NSDictionary *dic = [NSJSONSerialization JSONObjectWithData:data options:NSJSONReadingMutableContainers error:nil];  
  14.               /* 
  15.                { 
  16.                "access_token" = "OezXcEiiBSKSxW0eoylIeJDUKD6z6dmr42JANLPjNN7Kaf3e4GZ2OncrCfiKnGWiusJMZwzQU8kXcnT1hNs_ykAFDfDEuNp6waj-bDdepEzooL_k1vb7EQzhP8plTbD0AgR8zCRi1It3eNS7yRyd5A"; 
  17.                "expires_in" = 7200; 
  18.                openid = oyAaTjsDx7pl4Q42O3sDzDtA7gZs; 
  19.                "refresh_token" = "OezXcEiiBSKSxW0eoylIeJDUKD6z6dmr42JANLPjNN7Kaf3e4GZ2OncrCfiKnGWi2ZzH_XfVVxZbmha9oSFnKAhFsS0iyARkXCa7zPu4MqVRdwyb8J16V8cWw7oNIff0l-5F-4-GJwD8MopmjHXKiA"; 
  20.                scope = "snsapi_userinfo,snsapi_base"; 
  21.                } 
  22.                */  
  23.                   
  24.                 self.access_token.text = [dic objectForKey:@"access_token"];  
  25.                 self.openid.text = [dic objectForKey:@"openid"];  
  26.                 
  27.             }  
  28.         });  
  29.     });  
  30. }  

利用GCD来获取对应的token和openID.

第三步:userinfo

[objc] view plaincopyprint?在CODE上查看代码片派生到我的代码片
  1. -(void)getUserInfo  
  2. {  
  3.    // https://api.weixin.qq.com/sns/userinfo?access_token=ACCESS_TOKEN&openid=OPENID  
  4.       
  5.     NSString *url =[NSString stringWithFormat:@"https://api.weixin.qq.com/sns/userinfo?access_token=%@&openid=%@",self.access_token.text,self.openid.text];  
  6.       
  7.     dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{  
  8.         NSURL *zoneUrl = [NSURL URLWithString:url];  
  9.         NSString *zoneStr = [NSString stringWithContentsOfURL:zoneUrl encoding:NSUTF8StringEncoding error:nil];  
  10.         NSData *data = [zoneStr dataUsingEncoding:NSUTF8StringEncoding];  
  11.         dispatch_async(dispatch_get_main_queue(), ^{  
  12.             if (data) {  
  13.                 NSDictionary *dic = [NSJSONSerialization JSONObjectWithData:data options:NSJSONReadingMutableContainers error:nil];  
  14.                 /* 
  15.                  { 
  16.                  city = Haidian; 
  17.                  country = CN; 
  18.                  headimgurl = "http://wx.qlogo.cn/mmopen/FrdAUicrPIibcpGzxuD0kjfnvc2klwzQ62a1brlWq1sjNfWREia6W8Cf8kNCbErowsSUcGSIltXTqrhQgPEibYakpl5EokGMibMPU/0"; 
  19.                  language = "zh_CN"; 
  20.                  nickname = "xxx"; 
  21.                  openid = oyAaTjsDx7pl4xxxxxxx; 
  22.                  privilege =     ( 
  23.                  ); 
  24.                  province = Beijing; 
  25.                  sex = 1; 
  26.                  unionid = oyAaTjsxxxxxxQ42O3xxxxxxs; 
  27.                  } 
  28.                  */  
  29.                   
  30.                 self.nickname.text = [dic objectForKey:@"nickname"];  
  31.                 self.wxHeadImg.image = [UIImage imageWithData:[NSData dataWithContentsOfURL:[NSURL URLWithString:[dic objectForKey:@"headimgurl"]]]];  
  32.   
  33.             }  
  34.         });  
  35.   
  36.     });  
  37. }  
执行到这一步就算完成了整个授权登录的功能,能把昵称和头像显示出来,剩下的就是及时刷新你的token,详情可参考开发文档。

下面是登录成功后的QQ,新浪微博,微信的真机运行成功截图:



评价:微信的开发文档相比容易理解和调试,虽然没有demo,但是文档比较详细,所以可以在一定程度上减轻了开发的困难,但是相比之下微信的授权步骤比较麻烦,需要三步才能彻底获取用户信息,这点没有QQ和新浪微博简洁,需要有一定的阅读和代码功底,希望能给大家带来帮助。


后记:微信授权登录是对QQ和新浪微博登录的姊妹篇,前两者登录代码分析请参考上一篇博文,如需三种登录方式的demo源代码,该demo只需换掉默认宏定义和Url Schemes中默认的key即可正式使用,详情请发Email:mmw05@163.com ,特别注意代码不免费提供,需要支付一定金额,谢谢合作!


附言:本博文属于原创博客,如需转载请注明出处,否则将追究法律责任,博客来源:http://blog.csdn.net/mapboo/article/details/39475015


0 0