微信小程序微信登录的实现方法详解(JAVA后台)

作者:、楽. 时间:2023-08-29 12:37:25 

官方文档:https://developers.weixin.qq.com/miniprogram/dev/framework/open-ability/login.html

本文主要记录小程序实现微信登陆功能,后端为Java开发。

在开发之前我们先看一下官方提供的时序图,了解一下我们的大致开发流程:

微信小程序微信登录的实现方法详解(JAVA后台)

大致了解流程之后,我们便可以着手开发了。

1. 前提

一个可以测试的微信小程序

此微信小程序的APPID和APPscret(至开发者后台获取)

2. 开发流程

从时序图我们可以了解到流程大致分为两步:

  • 小程序端获取code后传给Java后台

  • Java后台获取code后向微信后台接口获取open_id

2.1 小程序端

在微信小程序的前端调用wx.login()获取一个code,这个code就像是我们去微信后台服务器获取用户信息的一个钥匙,微信通过获取这个code的过程给用户一个选择是否授权的选择,如果用户选择了授权就会返回一个code。这个code是一次性的,也是有时限的。由于我在Java后台进行了一次数据校验,所以我也会从getUserInfo接口中获取相关数据。代码如下:

微信小程序微信登录的实现方法详解(JAVA后台)

2.2 Java后端接口

后端的流程我将其大致分为如下几点:

  • 接收小程序发送的code

  • 开发者服务器 登录凭证校验接口 appi + appsecret + code

  • 接收微信接口服务 获取返回的参数

  • 校验签名 小程序发送的签名signature与服务器端生成的签名signature2 = sha1(rawData + sessionKey)

  • 根据返回的User实体类,判断用户是否是新用户,是的话,将用户信息存到数据库;

获取openId

后台接受了code以后通过建立一个http请求去访问微信后台服务器拉取这个用户的openid,如果一切正常就会得到这个用户对应这个小程序的openid。

请求的地址:

https://api.weixin.qq.com/sns/jscode2session?appid=APPID&secret=SECRET&js_code=JSCODE&grant_type=authorization_code

通过GET方式访问,其中的参数分别是:

  • appid:小程序的appid

  • secret:小程序的appsecret

  • js:小程序前端传来的code

  • grant_type:这个不用修改,表示授权的类型

请求工具类代码如下:(APPID自行替换)

public class WechatUtil {
   public static JSONObject getSessionKeyOrOpenId(String code) {
       String requestUrl = "https://api.weixin.qq.com/sns/jscode2session";
       Map<String, String> requestUrlParam = new HashMap<>();
       // https://mp.weixin.qq.com/wxopen/devprofile?action=get_profile&token=164113089&lang=zh_CN
       //小程序appId
       requestUrlParam.put("appid", WXConstant.APPID);
       //小程序secret
       requestUrlParam.put("secret", WXConstant.SECRET);
       //小程序端返回的code
       requestUrlParam.put("js_code", code);
       //默认参数
       requestUrlParam.put("grant_type", "authorization_code");
       //发送post请求读取调用微信接口获取openid用户唯一标识
       JSONObject jsonObject = JSON.parseObject(HttpClientUtil.doPost(requestUrl, requestUrlParam));
       return jsonObject;
   }
}

HTTP工具类如下:

需要添加相关依赖。

<!-- http请求工具包依赖 -->
<dependency>
<groupId>org.apache.httpcomponents</groupId>
<artifactId>httpclient</artifactId>
<version>4.5.2</version>
</dependency>
public class HttpClientUtil {

public static String doGet(String url, Map<String, String> param) {

// 创建Httpclient对象
       CloseableHttpClient httpclient = HttpClients.createDefault();

String resultString = "";
       CloseableHttpResponse response = null;
       try {
           // 创建uri
           URIBuilder builder = new URIBuilder(url);
           if (param != null) {
               for (String key : param.keySet()) {
                   builder.addParameter(key, param.get(key));
               }
           }
           URI uri = builder.build();

// 创建http GET请求
           HttpGet httpGet = new HttpGet(uri);

// 执行请求
           response = httpclient.execute(httpGet);
           // 判断返回状态是否为200
           if (response.getStatusLine().getStatusCode() == 200) {
               resultString = EntityUtils.toString(response.getEntity(), "UTF-8");
           }
       } catch (Exception e) {
           e.printStackTrace();
       } finally {
           try {
               if (response != null) {
                   response.close();
               }
               httpclient.close();
           } catch (IOException e) {
               e.printStackTrace();
           }
       }
       return resultString;
   }

public static String doGet(String url) {
       return doGet(url, null);
   }

public static String doPost(String url, Map<String, String> param) {
       // 创建Httpclient对象
       CloseableHttpClient httpClient = HttpClients.createDefault();
       CloseableHttpResponse response = null;
       String resultString = "";
       try {
           // 创建Http Post请求
           HttpPost httpPost = new HttpPost(url);
           // 创建参数列表
           if (param != null) {
               List<NameValuePair> paramList = new ArrayList<>();
               for (String key : param.keySet()) {
                   paramList.add(new BasicNameValuePair(key, param.get(key)));
               }
               // 模拟表单
               UrlEncodedFormEntity entity = new UrlEncodedFormEntity(paramList);
               httpPost.setEntity(entity);
           }
           // 执行http请求
           response = httpClient.execute(httpPost);
           resultString = EntityUtils.toString(response.getEntity(), "utf-8");
       } catch (Exception e) {
           e.printStackTrace();
       } finally {
           try {
               response.close();
           } catch (IOException e) {
               e.printStackTrace();
           }
       }

return resultString;
   }

public static String doPost(String url) {
       return doPost(url, null);
   }

public static String doPostJson(String url, String json) {
       // 创建Httpclient对象
       CloseableHttpClient httpClient = HttpClients.createDefault();
       CloseableHttpResponse response = null;
       String resultString = "";
       try {
           // 创建Http Post请求
           HttpPost httpPost = new HttpPost(url);
           // 创建请求内容
           StringEntity entity = new StringEntity(json, ContentType.APPLICATION_JSON);
           httpPost.setEntity(entity);
           // 执行http请求
           response = httpClient.execute(httpPost);
           resultString = EntityUtils.toString(response.getEntity(), "utf-8");
       } catch (Exception e) {
           e.printStackTrace();
       } finally {
           try {
               response.close();
           } catch (IOException e) {
               e.printStackTrace();
           }
       }

return resultString;
   }
}

接口代码

具体代码如下所示:

判断用户是否存在后的代码根据自己的业务逻辑进行修改即可。

@PostMapping("/wx/login")
public R user_login(@RequestParam(value = "code", required = false) String code,
                   @RequestParam(value = "rawData", required = false) String rawData,
                   @RequestParam(value = "signature", required = false) String signature) {
   // 用户非敏感信息:rawData
   // 签名:signature
   JSONObject rawDataJson = JSON.parseObject(rawData);
   // 1.接收小程序发送的code
   // 2.开发者服务器 登录凭证校验接口 appi + appsecret + code
   JSONObject SessionKeyOpenId = WechatUtil.getSessionKeyOrOpenId(code);
   // 3.接收微信接口服务 获取返回的参数
   String openid = SessionKeyOpenId.getString("openid");
   String sessionKey = SessionKeyOpenId.getString("session_key");

// 4.校验签名 小程序发送的签名signature与服务器端生成的签名signature2 = sha1(rawData + sessionKey)
   String signature2 = DigestUtils.sha1Hex(rawData + sessionKey);
   if (!signature.equals(signature2)) {
       return R.error().message("签名校验失败");
   }
   // 5.根据返回的User实体类,判断用户是否是新用户,是的话,将用户信息存到数据库;
   LambdaQueryWrapper<User> lqw = Wrappers.lambdaQuery();
   lqw.eq(User::getOpenId, openid);
   User user = userService.getOne(lqw);
   if (user == null) {
       // 用户信息入库
       String nickName = rawDataJson.getString("nickName");
       String avatarUrl = rawDataJson.getString("avatarUrl");
       user = new User();
       user.setOpenId(openid);
       user.setAvatar(avatarUrl);
       user.setNickName(nickName);
       userService.save(user);
   }
   return R.ok().data(user);
}

来源:https://blog.csdn.net/qq_41432730/article/details/123617323

标签:微信小程序,微信,登录
0
投稿

猜你喜欢

  • java使用dom4j生成与解析xml文档的方法示例

    2022-06-18 23:41:56
  • Android应用中使用SharedPreferences类存储数据的方法

    2022-06-11 20:27:08
  • JAVA通过HttpClient发送HTTP请求的方法示例

    2023-08-24 18:45:47
  • SpringBoot整合WebSocket实现聊天室流程全解

    2021-07-03 23:35:21
  • Android仿最新微信相机功能

    2022-06-08 00:35:08
  • Java实现经典游戏飞机大战-I的示例代码

    2023-07-30 15:45:22
  • 深入理解Spring事务的传播行为

    2022-08-31 11:25:21
  • C#实现合并及拆分PDF文件的方法

    2021-06-15 12:55:23
  • Unity实现粒子光效导出成png序列帧

    2023-08-30 21:22:52
  • VS2019配置OpenCV4.1.0详细教程与测试代码(推荐)

    2021-11-06 21:55:24
  • springcloud feign传输List的坑及解决

    2023-06-20 18:31:57
  • 零基础写Java知乎爬虫之获取知乎编辑推荐内容

    2023-11-29 04:33:39
  • DialogFragment运行原理及使用方法详解

    2023-10-27 04:00:26
  • redis redisson 集合的使用案例(RList、Rset、RMap)

    2023-11-13 13:50:21
  • swing分割窗口控件JSplitPane使用方法详解

    2021-07-28 14:15:20
  • Java下SpringBoot创建定时任务详解

    2023-10-03 01:25:23
  • 一些比较实用的 Android adb 命令分享

    2023-12-22 12:39:41
  • Java equals 方法与hashcode 方法的深入解析

    2022-03-05 19:20:15
  • 完美解决Spring Boot前端的Access-Control-Allow-Origin跨域问题

    2023-09-19 10:16:28
  • C#编程获取IP地址的方法示例

    2023-06-16 21:18:20
  • asp之家 软件编程 m.aspxhome.com