微信小程序调用微信登陆获取openid及java做为服务端示例

 更新时间:2018年01月29日 11:29:39   作者:未来_之路  
这篇文章主要介绍了微信小程序调用微信登陆获取openid及java做为服务端示例,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

一、微信小程序
第一步:调用 wx.login获取code 文档地址
第二步:判断用户是否授权读取用户信息 文档地址
第三步:调用wx.getUserInfo读取用户数据 文档地址
第四步:由于小程序后台授权域名无法授权微信的域名,所以我们只能通过我们自己的服务器去调用微信服务器去获取用户信息,故我们将wx.login获取code 和 wx.getUserInfo 获取的encryptedData与iv 通过wx.request 请求传入后台

服务器返回的数据:

小程序代码:

//调用登录接口,获取 code 
wx.login({ 
 success: function (res) { 
  wx.getSetting({ 
   success(setRes) { 
    // 判断是否已授权 
    if (!setRes.authSetting['scope.userInfo']) { 
     // 授权访问 
     wx.authorize({ 
      scope: 'scope.userInfo', 
      success() { 
       //获取用户信息 
       wx.getUserInfo({ 
        lang: "zh_CN", 
        success: function (userRes) { 
         //发起网络请求 
         wx.request({ 
          url: config.loginWXUrl, 
          data: { 
           code: res.code, 
           encryptedData: userRes.encryptedData, 
           iv: userRes.iv 
          }, 
          header: { 
           "Content-Type": "application/x-www-form-urlencoded" 
          }, 
          method: 'POST', 
          //服务端的回掉 
          success: function (result) { 
           var data = result.data.result; 
           data.expireTime = nowDate + EXPIRETIME; 
           wx.setStorageSync("userInfo", data); 
           userInfo = data; 
          } 
         }) 
        } 
       }) 
      } 
     }) 
    } else { 
     //获取用户信息 
     wx.getUserInfo({ 
      lang: "zh_CN", 
      success: function (userRes) { 
       //发起网络请求 
       wx.request({ 
        url: config.loginWXUrl, 
        data: { 
         code: res.code, 
         encryptedData: userRes.encryptedData, 
         iv: userRes.iv 
        }, 
        header: { 
         "Content-Type": "application/x-www-form-urlencoded" 
        }, 
        method: 'POST', 
        success: function (result) { 
         var data = result.data.result; 
         data.expireTime = nowDate + EXPIRETIME; 
         wx.setStorageSync("userInfo", data); 
         userInfo = data; 
        } 
       }) 
      } 
     }) 
    } 
   } 
  }) 
 } 
}) 

二、java服务端

根据code获取openid与解码用户信息 代码

所需要的jar包

<dependency> 
  <groupId>org.codehaus.xfire</groupId> 
  <artifactId>xfire-core</artifactId> 
  <version>1.2.6</version> 
</dependency> 
<dependency> 
  <groupId>org.bouncycastle</groupId> 
  <artifactId>bcprov-jdk16</artifactId> 
  <version>1.46</version> 
</dependency> 
/** 
 * 微信小程序信息获取 
 * 
 * @author zhy 
 */ 
public class WXAppletUserInfo { 
  private static Logger log = Logger.getLogger(WXAppletUserInfo.class); 
   
  /** 
   * 获取微信小程序 session_key 和 openid 
   * 
   * @author zhy 
   * @param code 调用微信登陆返回的Code 
   * @return 
   */ 
  public static JSONObject getSessionKeyOropenid(String code){ 
    //微信端登录code值 
    String wxCode = code; 
    ResourceBundle resource = ResourceBundle.getBundle("weixin");  //读取属性文件 
    String requestUrl = resource.getString("url"); //请求地址 https://api.weixin.qq.com/sns/jscode2session 
    Map<String,String> requestUrlParam = new HashMap<String,String>(); 
    requestUrlParam.put("appid", resource.getString("appId")); //开发者设置中的appId 
    requestUrlParam.put("secret", resource.getString("appSecret")); //开发者设置中的appSecret 
    requestUrlParam.put("js_code", wxCode); //小程序调用wx.login返回的code 
    requestUrlParam.put("grant_type", "authorization_code");  //默认参数 
     
    //发送post请求读取调用微信 https://api.weixin.qq.com/sns/jscode2session 接口获取openid用户唯一标识 
    JSONObject jsonObject = JSON.parseObject(UrlUtil.sendPost(requestUrl, requestUrlParam)); 
    return jsonObject; 
  } 
   
  /** 
   * 解密用户敏感数据获取用户信息 
   * 
   * @author zhy 
   * @param sessionKey 数据进行加密签名的密钥 
   * @param encryptedData 包括敏感数据在内的完整用户信息的加密数据 
   * @param iv 加密算法的初始向量 
   * @return 
   */ 
  public static JSONObject getUserInfo(String encryptedData,String sessionKey,String iv){ 
    // 被加密的数据 
    byte[] dataByte = Base64.decode(encryptedData); 
    // 加密秘钥 
    byte[] keyByte = Base64.decode(sessionKey); 
    // 偏移量 
    byte[] ivByte = Base64.decode(iv); 
    try { 
        // 如果密钥不足16位,那么就补足. 这个if 中的内容很重要 
      int base = 16; 
      if (keyByte.length % base != 0) { 
        int groups = keyByte.length / base + (keyByte.length % base != 0 ? 1 : 0); 
        byte[] temp = new byte[groups * base]; 
        Arrays.fill(temp, (byte) 0); 
        System.arraycopy(keyByte, 0, temp, 0, keyByte.length); 
        keyByte = temp; 
      } 
      // 初始化 
      Security.addProvider(new BouncyCastleProvider()); 
      Cipher cipher = Cipher.getInstance("AES/CBC/PKCS7Padding","BC"); 
      SecretKeySpec spec = new SecretKeySpec(keyByte, "AES"); 
      AlgorithmParameters parameters = AlgorithmParameters.getInstance("AES"); 
      parameters.init(new IvParameterSpec(ivByte)); 
      cipher.init(Cipher.DECRYPT_MODE, spec, parameters);// 初始化 
      byte[] resultByte = cipher.doFinal(dataByte); 
      if (null != resultByte && resultByte.length > 0) { 
        String result = new String(resultByte, "UTF-8"); 
        return JSON.parseObject(result); 
      } 
    } catch (NoSuchAlgorithmException e) { 
      log.error(e.getMessage(), e); 
    } catch (NoSuchPaddingException e) { 
      log.error(e.getMessage(), e); 
    } catch (InvalidParameterSpecException e) { 
      log.error(e.getMessage(), e); 
    } catch (IllegalBlockSizeException e) { 
      log.error(e.getMessage(), e); 
    } catch (BadPaddingException e) { 
      log.error(e.getMessage(), e); 
    } catch (UnsupportedEncodingException e) { 
      log.error(e.getMessage(), e); 
    } catch (InvalidKeyException e) { 
      log.error(e.getMessage(), e); 
    } catch (InvalidAlgorithmParameterException e) { 
      log.error(e.getMessage(), e); 
    } catch (NoSuchProviderException e) { 
      log.error(e.getMessage(), e); 
    } 
    return null; 
  } 
} 

发送请求的代码

   /** 
* 向指定 URL 发送POST方法的请求 
* 
* @param url 发送请求的 URL 
* @param param 请求参数 
* @return 所代表远程资源的响应结果 
*/ 
ublic static String sendPost(String url, Map<String, ?> paramMap) { 
   PrintWriter out = null; 
   BufferedReader in = null; 
   String result = ""; 
    
   String param = ""; 
Iterator<String> it = paramMap.keySet().iterator(); 
 
while(it.hasNext()) { 
  String key = it.next(); 
  param += key + "=" + paramMap.get(key) + "&"; 
} 
 
   try { 
     URL realUrl = new URL(url); 
     // 打开和URL之间的连接 
     URLConnection conn = realUrl.openConnection(); 
     // 设置通用的请求属性 
     conn.setRequestProperty("accept", "*/*"); 
     conn.setRequestProperty("connection", "Keep-Alive"); 
     conn.setRequestProperty("Accept-Charset", "utf-8"); 
     conn.setRequestProperty("user-agent", "Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.1;SV1)"); 
     // 发送POST请求必须设置如下两行 
     conn.setDoOutput(true); 
     conn.setDoInput(true); 
     // 获取URLConnection对象对应的输出流 
     out = new PrintWriter(conn.getOutputStream()); 
     // 发送请求参数 
     out.print(param); 
     // flush输出流的缓冲 
     out.flush(); 
     // 定义BufferedReader输入流来读取URL的响应 
     in = new BufferedReader(new InputStreamReader(conn.getInputStream(), "UTF-8")); 
     String line; 
     while ((line = in.readLine()) != null) { 
       result += line; 
     } 
   } catch (Exception e) { 
    log.error(e.getMessage(), e); 
   } 
   //使用finally块来关闭输出流、输入流 
   finally{ 
     try{ 
       if(out!=null){ 
         out.close(); 
       } 
       if(in!=null){ 
         in.close(); 
       } 
     } 
     catch(IOException ex){ 
       ex.printStackTrace(); 
     } 
   } 
   return result; 
 } 

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

相关文章

  • Java建造者模式构建复杂对象的最佳实践

    Java建造者模式构建复杂对象的最佳实践

    建造者模式,是一种对象构建模式 它可以将复杂对象的建造过程抽象出来,使这个抽象过程的不同实现方法可以构造出不同表现的对象。本文将通过示例讲解建造者模式,需要的可以参考一下
    2023-04-04
  • Java实现局域网聊天室功能(私聊、群聊)

    Java实现局域网聊天室功能(私聊、群聊)

    这篇文章主要为大家详细介绍了Java实现局域网聊天室功能,包括私聊、群聊,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-05-05
  • 带你全面认识Java中的异常处理

    带你全面认识Java中的异常处理

    在你所写过的代码中,你已经接触过一些异常了,我们可以通过一些简单的代码让我们理解一些简单的异常,下面这篇文章主要给大家介绍了关于Java中异常处理的相关资料,文中通过图文介绍的非常详细,需要的朋友可以参考下
    2022-12-12
  • SpringBoot日志级别与日志分组详解

    SpringBoot日志级别与日志分组详解

    文章介绍了日志级别(ALL至OFF)及其作用,说明SpringBoot默认日志级别为INFO,可通过application.properties调整全局或特定类的日志级别,并展示了日志调试技巧和分组配置方法,感兴趣的朋友跟随小编一起看看吧
    2025-08-08
  • springboot启动读取外部配置文件实现方式

    springboot启动读取外部配置文件实现方式

    Spring Boot通过外部配置文件优先级(外部>内部)加载运维定义的配置,需将文件放置在jar包同级或config目录下,相同配置项以外部为准,不同配置项互补生效
    2025-09-09
  • java8新特性之stream流中reduce()求和知识总结

    java8新特性之stream流中reduce()求和知识总结

    今天带大家回顾Java8的新特性,文中对stream流中reduce()求和的相关知识作了详细的介绍,对正在学习java的小伙伴们有很好地帮助,需要的朋友可以参考下
    2021-05-05
  • 用java实现冒泡排序算法

    用java实现冒泡排序算法

    两两比较待排序记录的关键字,发现两个记录的次序相反时即进行交换,直到没有反序的记录为止
    2013-01-01
  • springBoot动态加载jar及如何将类注册到IOC

    springBoot动态加载jar及如何将类注册到IOC

    在SpringBoot项目中动态加载jar文件并将其类注册到IOC容器是一种高级应用方式,,这种方法为SpringBoot项目提供了更灵活的扩展能力,使得项目可以在不修改原有代码的基础上增加新的功能模块,感兴趣的朋友一起看看吧
    2024-11-11
  • springboot中如何使用openfeign进行接口调用

    springboot中如何使用openfeign进行接口调用

    这篇文章主要介绍了springboot中如何使用openfeign进行接口调用问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-07-07
  • SpringBoot中SM2公钥加密、私钥解密的实现示例详解

    SpringBoot中SM2公钥加密、私钥解密的实现示例详解

    本文介绍了如何在SpringBoot项目中实现SM2公钥加密和私钥解密的功能,通过使用Hutool库和BouncyCastle依赖,简化了SM2加密的实现过程,本文结合实例代码给大家介绍的非常详细,感兴趣的朋友一起看看吧
    2025-07-07

最新评论