目录
准备工作
需要小程序账号(可以去微信公众平台创建一个测试号或者正式号)
+ appid:小程序id
+ appsecret:小程序密钥,
需要工具
+ HBuilder X(用来启动小程序代码的)
+ 微信开发者工具(用来调试小程序的)
只有后端的实现,没有前端的代码
实现流程
1.首先需要前端在小程序中调用wx.login去获取code
2.前端将code交给后端,后端使用code,appid,sectret调用微信的api获取openid和session_key(用来解密用的),union_id(要注册到微信开放平台才会有)
我们获取到了openid去数据库查询是否存在,如果存在直接返回token完成登录
如果数据库中并没有这个openid,需要进行第三步后的操作,获取用户基本信息和手机号进行注册
3.返回数据给前端,让前端通过使用wx.getUserInfo
接口或者wx.getUserProfile
(官方推荐这种,前面那个快要废弃)接口跳转到授权页面,获取用户的基本数据userInfo,和加密后的手机号数据encryptedData 以及 iv(小程序测试号不能获取手机号授权)
4.前端将获取到的userInfo 和 encryptedData 以及 iv交给后端,后端将通过code获取到的session_key配合encryptedData 和iv对手机号进行解密,然后将获取到的手机号,基本信息,openid等信息进行注册后,返回前端token注册登录完成
5.完成
实现代码
下面的代码使用的是微信的sdk实现方式(提供了很多工具类,不需要我们定义解密等代码,直接调用sdk中的方法,非常非常非常简单)
公共部分
WxMaDefaultConfigImpl config = new WxMaDefaultConfigImpl();
// 替换为你的小程序 appId
config.setAppid("your_app_id");
// 替换为你的小程序 appSecret
config.setSecret("your_app_secret");
WxMaService service = new WxMaServiceImpl();
service.setWxMaConfig(config)
通过code获取openid等信息
service是上面公共部分定义的
//code是前端返回的
WxMaJscode2SessionResult wxMaJscode2SessionResult = service.getUserService().getSessionInfo(code);
String openid = wxMaJscode2SessionResult.getOpenid;
String sessionKey = wxMaJscode2SessionResult.getSessionKey;
解密手机号
service是上面公共部分定义的
//sessionKey通过code换取的
//encryptedData, iv前端返回的
WxMaPhoneNumberInfo phoneNumberInfo = wxMaService.getUserService().getPhoneNoInfo(sessionKey, encryptedData, iv);
String phoneNumber = phoneNumberInfo.getPhoneNumber();
扩展
不借助工具类实现解密
public static String decryptData(String encryptedData, String sessionKey, String iv) throws Exception {
// 解密
byte[] encryptedDataBytes = Base64.getDecoder().decode(encryptedData);
byte[] sessionKeyBytes = Base64.getDecoder().decode(sessionKey);
byte[] ivBytes = Base64.getDecoder().decode(iv);
Cipher cipher = Cipher.getInstance("AES/CBC/PKCS5Padding");
SecretKeySpec secretKeySpec = new SecretKeySpec(sessionKeyBytes, "AES");
IvParameterSpec ivParameterSpec = new IvParameterSpec(ivBytes);
cipher.init(Cipher.DECRYPT_MODE, secretKeySpec, ivParameterSpec);
byte[] decryptedBytes = cipher.doFinal(encryptedDataBytes);
return new String(decryptedBytes);
}
借助工具类获取access_token
service是上面公共部分定义的
String accessToken = service.getAccessToken();