微信公众平台开发实战Java版之微信获取用户基本信息
在关注者与公众号产生消息交互后,公众号可获得关注者的OpenID(加密后的微信号,每个用户对每个公众号的OpenID是唯一的。对于不同公众号,同一用户的openid不同)。
公众号可通过本接口来根据OpenID获取用户基本信息,包括昵称、头像、性别、所在城市、语言和关注时间。
开发者可通过OpenID来获取用户基本信息。请使用https协议。
我们可以看看官方的文档:获取用户的基本信息。
接口调用请求说明
http请求方式: GET
https://api.weixin.qq.com/cgi-bin/user/info?access_token=ACCESS_TOKEN&openid=OPENID&lang=zh_CN
参数说明
返回说明
正常情况下,微信会返回下述JSON数据包给公众号:
-
{
-
"subscribe": 1,
-
"openid": "o6_bmjrPTlm6_2sgVt7hMZOPfL2M",
-
"nickname": "Band",
-
"sex": 1,
-
"language": "zh_CN",
-
"city": "广州",
-
"province": "广东",
-
"country": "中国",
-
"headimgurl": "http://wx.qlogo.cn/mmopen/g3MonUZtNHkdmzicIlibx6iaFqAc56vxLSUfpb6n5WKSYVY0ChQKkiaJSgQ1dZuTOgvLLrhJbERQQ4eMsv84eavHiaiceqxibJxCfHe/0",
-
"subscribe_time": 1382694957,
-
"unionid": " o6_bmasdasdsad6_2sgVt7hMZOPfL"
-
"remark": "",
-
"groupid": 0
-
}
参数说明:
错误时微信会返回错误码等信息,JSON数据包示例如下(该示例为AppID无效错误):
{"errcode":40013,"errmsg":"invalid appid"}
根据上面的信息,我们定义一个用户信息类来存放用户的基本信息。
-
package com.souvc.weixin.pojo;
-
/**
-
* 类名: WeixinUserInfo </br>
-
* 描述: 微信用户的基本信息 </br>
-
* 开发人员: souvc </br>
-
* 创建时间: 2015-11-27 </br>
-
* 发布版本:V1.0 </br>
-
*/
-
public class WeixinUserInfo {
-
// 用户的标识
-
private String openId;
-
// 关注状态(1是关注,0是未关注),未关注时获取不到其余信息
-
private int subscribe;
-
// 用户关注时间,为时间戳。如果用户曾多次关注,则取最后关注时间
-
private String subscribeTime;
-
// 昵称
-
private String nickname;
-
// 用户的性别(1是男性,2是女性,0是未知)
-
private int sex;
-
// 用户所在国家
-
private String country;
-
// 用户所在省份
-
private String province;
-
// 用户所在城市
-
private String city;
-
// 用户的语言,简体中文为zh_CN
-
private String language;
-
// 用户头像
-
private String headImgUrl;
-
public String getOpenId() {
-
return openId;
-
}
-
public void setOpenId(String openId) {
-
this.openId = openId;
-
}
-
public int getSubscribe() {
-
return subscribe;
-
}
-
public void setSubscribe(int subscribe) {
-
this.subscribe = subscribe;
-
}
-
public String getSubscribeTime() {
-
return subscribeTime;
-
}
-
public void setSubscribeTime(String subscribeTime) {
-
this.subscribeTime = subscribeTime;
-
}
-
public String getNickname() {
-
return nickname;
-
}
-
public void setNickname(String nickname) {
-
this.nickname = nickname;
-
}
-
public int getSex() {
-
return sex;
-
}
-
public void setSex(int sex) {
-
this.sex = sex;
-
}
-
public String getCountry() {
-
return country;
-
}
-
public void setCountry(String country) {
-
this.country = country;
-
}
-
public String getProvince() {
-
return province;
-
}
-
public void setProvince(String province) {
-
this.province = province;
-
}
-
public String getCity() {
-
return city;
-
}
-
public void setCity(String city) {
-
this.city = city;
-
}
-
public String getLanguage() {
-
return language;
-
}
-
public void setLanguage(String language) {
-
this.language = language;
-
}
-
public String getHeadImgUrl() {
-
return headImgUrl;
-
}
-
public void setHeadImgUrl(String headImgUrl) {
-
this.headImgUrl = headImgUrl;
-
}
-
}
我们先来看看获取用户信息的接口:
https://api.weixin.qq.com/cgi-bin/user/info?access_token=ACCESS_TOKEN&openid=OPENID&lang=zh_CN
根据分析,获取用户的基本信息需要一个token。
-
package com.souvc.weixin.pojo;
-
/**
-
* 类名: Token </br>
-
* 描述: 凭证 </br>
-
* 开发人员: souvc </br>
-
* 创建时间: 2015-11-27 </br>
-
* 发布版本:V1.0 </br>
-
*/
-
public class Token {
-
// 接口访问凭证
-
private String accessToken;
-
// 凭证有效期,单位:秒
-
private int expiresIn;
-
public String getAccessToken() {
-
return accessToken;
-
}
-
public void setAccessToken(String accessToken) {
-
this.accessToken = accessToken;
-
}
-
public int getExpiresIn() {
-
return expiresIn;
-
}
-
public void setExpiresIn(int expiresIn) {
-
this.expiresIn = expiresIn;
-
}
-
}
https请求,需要的信任管理器:
-
package com.souvc.weixin.util;
-
import java.security.cert.CertificateException;
-
import java.security.cert.X509Certificate;
-
import javax.net.ssl.X509TrustManager;
-
/**
-
* 类名: MyX509TrustManager </br>
-
* 描述:信任管理器 </br>
-
* 开发人员: souvc </br>
-
* 创建时间: 2015-11-27 </br>
-
* 发布版本:V1.0 </br>
-
*/
-
public class MyX509TrustManager implements X509TrustManager {
-
// 检查客户端证书
-
public void checkClientTrusted(X509Certificate[] chain, String authType) throws CertificateException {
-
}
-
// 检查服务器端证书
-
public void checkServerTrusted(X509Certificate[] chain, String authType) throws CertificateException {
-
}
-
// 返回受信任的X509证书数组
-
public X509Certificate[] getAcceptedIssuers() {
-
return null;
-
}
-
}
封装了一个公共类:
-
package com.souvc.weixin.util;
-
import java.io.BufferedReader;
-
import java.io.InputStream;
-
import java.io.InputStreamReader;
-
import java.io.OutputStream;
-
import java.io.UnsupportedEncodingException;
-
import java.net.ConnectException;
-
import java.net.URL;
-
import javax.net.ssl.HttpsURLConnection;
-
import javax.net.ssl.SSLContext;
-
import javax.net.ssl.SSLSocketFactory;
-
import javax.net.ssl.TrustManager;
-
import net.sf.json.JSONException;
-
import net.sf.json.JSONObject;
-
import org.slf4j.Logger;
-
import org.slf4j.LoggerFactory;
-
import com.souvc.weixin.pojo.Token;
-
/**
-
* 类名: CommonUtil </br>
-
* 描述: 通用工具类 </br>
-
* 开发人员: souvc </br>
-
* 创建时间: 2015-11-27 </br>
-
* 发布版本:V1.0 </br>
-
*/
-
public class CommonUtil {
-
private static Logger log = LoggerFactory.getLogger(CommonUtil.class);
-
// 凭证获取(GET)
-
public final static String token_url = "https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid=APPID&secret=APPSECRET";
-
/**
-
* 发送https请求
-
*
-
* @param requestUrl 请求地址
-
* @param requestMethod 请求方式(GET、POST)
-
* @param outputStr 提交的数据
-
* @return JSONObject(通过JSONObject.get(key)的方式获取json对象的属性值)
-
*/
-
public static JSONObject httpsRequest(String requestUrl, String requestMethod, String outputStr) {
-
JSONObject jsonObject = null;
-
try {
-
// 创建SSLContext对象,并使用我们指定的信任管理器初始化
-
TrustManager[] tm = { new MyX509TrustManager() };
-
SSLContext sslContext = SSLContext.getInstance("SSL", "SunJSSE");
-
sslContext.init(null, tm, new java.security.SecureRandom());
-
// 从上述SSLContext对象中得到SSLSocketFactory对象
-
SSLSocketFactory ssf = sslContext.getSocketFactory();
-
URL url = new URL(requestUrl);
-
HttpsURLConnection conn = (HttpsURLConnection) url.openConnection();
-
conn.setSSLSocketFactory(ssf);
-
conn.setDoOutput(true);
-
conn.setDoInput(true);
-
conn.setUseCaches(false);
-
// 设置请求方式(GET/POST)
-
conn.setRequestMethod(requestMethod);
-
// 当outputStr不为null时向输出流写数据
-
if (null != outputStr) {
-
OutputStream outputStream = conn.getOutputStream();
-
// 注意编码格式
-
outputStream.write(outputStr.getBytes("UTF-8"));
-
outputStream.close();
-
}
-
// 从输入流读取返回内容
-
InputStream inputStream = conn.getInputStream();
-
InputStreamReader inputStreamReader = new InputStreamReader(inputStream, "utf-8");
-
BufferedReader bufferedReader = new BufferedReader(inputStreamReader);
-
String str = null;
-
StringBuffer buffer = new StringBuffer();
-
while ((str = bufferedReader.readLine()) != null) {
-
buffer.append(str);
-
}
-
// 释放资源
-
bufferedReader.close();
-
inputStreamReader.close();
-
inputStream.close();
-
inputStream = null;
-
conn.disconnect();
-
jsonObject = JSONObject.fromObject(buffer.toString());
-
} catch (ConnectException ce) {
-
log.error("连接超时:{}", ce);
-
} catch (Exception e) {
-
log.error("https请求异常:{}", e);
-
}
-
return jsonObject;
-
}
-
/**
-
* 获取接口访问凭证
-
*
-
* @param appid 凭证
-
* @param appsecret **
-
* @return
-
*/
-
public static Token getToken(String appid, String appsecret) {
-
Token token = null;
-
String requestUrl = token_url.replace("APPID", appid).replace("APPSECRET", appsecret);
-
// 发起GET请求获取凭证
-
JSONObject jsonObject = httpsRequest(requestUrl, "GET", null);
-
if (null != jsonObject) {
-
try {
-
token = new Token();
-
token.setAccessToken(jsonObject.getString("access_token"));
-
token.setExpiresIn(jsonObject.getInt("expires_in"));
-
} catch (JSONException e) {
-
token = null;
-
// 获取token失败
-
log.error("获取token失败 errcode:{} errmsg:{}", jsonObject.getInt("errcode"), jsonObject.getString("errmsg"));
-
}
-
}
-
return token;
-
}
-
/**
-
* URL编码(utf-8)
-
*
-
* @param source
-
* @return
-
*/
-
public static String urlEncodeUTF8(String source) {
-
String result = source;
-
try {
-
result = java.net.URLEncoder.encode(source, "utf-8");
-
} catch (UnsupportedEncodingException e) {
-
e.printStackTrace();
-
}
-
return result;
-
}
-
/**
-
* 根据内容类型判断文件扩展名
-
*
-
* @param contentType 内容类型
-
* @return
-
*/
-
public static String getFileExt(String contentType) {
-
String fileExt = "";
-
if ("image/jpeg".equals(contentType))
-
fileExt = ".jpg";
-
else if ("audio/mpeg".equals(contentType))
-
fileExt = ".mp3";
-
else if ("audio/amr".equals(contentType))
-
fileExt = ".amr";
-
else if ("video/mp4".equals(contentType))
-
fileExt = ".mp4";
-
else if ("video/mpeg4".equals(contentType))
-
fileExt = ".mp4";
-
return fileExt;
-
}
-
}
获取用户基本信息的方法:
-
/**
-
* 获取用户信息
-
*
-
* @param accessToken 接口访问凭证
-
* @param openId 用户标识
-
* @return WeixinUserInfo
-
*/
-
public static WeixinUserInfo getUserInfo(String accessToken, String openId) {
-
WeixinUserInfo weixinUserInfo = null;
-
// 拼接请求地址
-
String requestUrl = "https://api.weixin.qq.com/cgi-bin/user/info?access_token=ACCESS_TOKEN&openid=OPENID";
-
requestUrl = requestUrl.replace("ACCESS_TOKEN", accessToken).replace("OPENID", openId);
-
// 获取用户信息
-
JSONObject jsonObject = CommonUtil.httpsRequest(requestUrl, "GET", null);
-
if (null != jsonObject) {
-
try {
-
weixinUserInfo = new WeixinUserInfo();
-
// 用户的标识
-
weixinUserInfo.setOpenId(jsonObject.getString("openid"));
-
// 关注状态(1是关注,0是未关注),未关注时获取不到其余信息
-
weixinUserInfo.setSubscribe(jsonObject.getInt("subscribe"));
-
// 用户关注时间
-
weixinUserInfo.setSubscribeTime(jsonObject.getString("subscribe_time"));
-
// 昵称
-
weixinUserInfo.setNickname(jsonObject.getString("nickname"));
-
// 用户的性别(1是男性,2是女性,0是未知)
-
weixinUserInfo.setSex(jsonObject.getInt("sex"));
-
// 用户所在国家
-
weixinUserInfo.setCountry(jsonObject.getString("country"));
-
// 用户所在省份
-
weixinUserInfo.setProvince(jsonObject.getString("province"));
-
// 用户所在城市
-
weixinUserInfo.setCity(jsonObject.getString("city"));
-
// 用户的语言,简体中文为zh_CN
-
weixinUserInfo.setLanguage(jsonObject.getString("language"));
-
// 用户头像
-
weixinUserInfo.setHeadImgUrl(jsonObject.getString("headimgurl"));
-
} catch (Exception e) {
-
if (0 == weixinUserInfo.getSubscribe()) {
-
log.error("用户{}已取消关注", weixinUserInfo.getOpenId());
-
} else {
-
int errorCode = jsonObject.getInt("errcode");
-
String errorMsg = jsonObject.getString("errmsg");
-
log.error("获取用户信息失败 errcode:{} errmsg:{}", errorCode, errorMsg);
-
}
-
}
-
}
-
return weixinUserInfo;
-
}
测试的方法:注意将以下替换为自己的appid和秘钥。
-
public static void main(String args[]) {
-
// 获取接口访问凭证
-
String accessToken = CommonUtil.getToken("xxxx", "xxxx").getAccessToken();
-
/**
-
* 获取用户信息
-
*/
-
WeixinUserInfo user = getUserInfo(accessToken, "ooK-yuJvd9gEegH6nRIen-gnLrVw");
-
System.out.println("OpenID:" + user.getOpenId());
-
System.out.println("关注状态:" + user.getSubscribe());
-
System.out.println("关注时间:" + user.getSubscribeTime());
-
System.out.println("昵称:" + user.getNickname());
-
System.out.println("性别:" + user.getSex());
-
System.out.println("国家:" + user.getCountry());
-
System.out.println("省份:" + user.getProvince());
-
System.out.println("城市:" + user.getCity());
-
System.out.println("语言:" + user.getLanguage());
-
System.out.println("头像:" + user.getHeadImgUrl());
-
}
效果如下:
OpenID:ooK-yuJvd9gEegH6nRIen-gnLrVw
关注状态:1
关注时间:1449021142
昵称:风少
性别:1
国家:中国
省份:广东
城市:广州
语言:zh_CN
头像:http://wx.qlogo.cn/mmopen/lOZIEvyfCa7aZQ7CkiamdpQicUDnGDEC0nzb7ZALjdl3TzFVFEHWM1AFqEXnicNIDeh0IQYTt0NrIP06ibg4W5WflASfFfX9qqib0/0