有很多的朋友再问我怎么获取code,openid之类的问题,在这里我就给大家分享一下。
在做微信支付是需要获取openid的,推送模板消息也是需要openid包括其他一些功能分享等也都是需要的,openid是什么呢,这里给大家解释一下,是用户在公众号中的一个标识,就相当于你数据表中的ID一样,一个身份标识。
不多说废话了这里直接上代码了:
这里写一个了a标签来获取code,需要注意的是redirect_uri=接收code的回调地址(是带http://格式)可以指向下面的register
获取openid
appid:是公众号的ID,appsecrect:是公众号的密钥,code:就是上面a标签重定向的获取到的
@RequestMapping(params = "method=register", method = RequestMethod.POST)
public @ResponseBody Map register(HttpServletRequest request) throws Exception {
//接收来之页面a标签重定向的参数
String code = request.getParameter("code");
//返回openid
String openid = WeiXinUtils.getOpenid(appid, appsecrt, code);
//推送模板消息给用户
WeiXinUtils.sendMsg(openid, "标题", "内容");
}
给一个封装的WeiXinUtils类:
import java.io.IOException;
import java.util.HashMap;
import java.util.Map;
import javax.servlet.http.Cookie;
import javax.servlet.http.HttpServletResponse;
import org.joda.time.DateTime;
import com.alibaba.fastjson.JSONObject;
public class WeiXinUtils {
/**
* Description: 获取微信公众号token
*
* @author dsn
* @date 2018年9月21日 上午9:53:26
* @param appid
* @param secret
* @return
* @version 1.0
*/
public static String getAccessToken(String appid, String secret) {
String token = "";
String token_url = "https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid=" + appid
+ "&secret=" + secret;
JSONObject result = PayCommonUtil.httpsRequest(token_url, "POST");
if (result.get("access_token") != null) {
token = result.get("access_token").toString();
}
return token;
}
/**
* Description: 获取微信ticket
*
* @author dsn
* @date 2018年9月21日 上午9:54:03
* @param token
* @return
* @version 1.0
*/
public static String getTicket(String token) {
if ("".equalsIgnoreCase(token) || null == token) {
return "";
}
String ticket_url = "https://api.weixin.qq.com/cgi-bin/ticket/getticket?access_token=" + token + "&type=jsapi";
JSONObject result = PayCommonUtil.httpsRequest(ticket_url, "POST");
return result.get("ticket").toString();
}
/**
* Description: 获取code
*
* @author dsn
* @date 2018年9月21日 下午12:02:43
* @param redirect_uri
* @param appid
* @return
* @version 1.0
*/
public static void getCode(String redirect_uri, String appid, HttpServletResponse response) {
String code_url = "https://open.weixin.qq.com/connect/oauth2/authorize?redirect_uri=" + redirect_uri + "&appid="
+ appid + "&response_type=code&scope=snsapi_base&state=1#wechat_redirect";
try {
response.sendRedirect(code_url);
} catch (IOException e) {
e.printStackTrace();
}
}
/**
* Description: 通过code获取用户openid
*
* @author dsn
* @date 2018年9月21日 下午12:02:18
* @param appid
* @param appsecret
* @param code
* @return
* @version 1.0
*/
public static String getOpenid(String appid, String appsecret, String code) {
String openid_url = "https://api.weixin.qq.com/sns/oauth2/access_token?appid=" + appid + "&secret=" + appsecret
+ "&code=" + code + "&grant_type=authorization_code";
JSONObject result = PayCommonUtil.httpsRequest(openid_url, "POST");
if (result.containsKey("errcode")) {
return "";
} else {
return result.getString("openid");
}
}
/**
* Description: 根据微信openid 获取用户是否关注公众号
*
* @author dsn
* @date 2018年9月21日 上午9:56:10
* @param openId
* @return
* @version 1.0
*/
public static Integer subscribeState(String openId) {
String tmpurl = "https://api.weixin.qq.com/cgi-bin/user/info?access_token="
+ getAccessToken("wxcd1feb93b3454edc", "0a05181ac3bc53c44206a58e97d99b26") + "&openid=" + openId;
JSONObject result = PayCommonUtil.httpsRequest(tmpurl, "GET");
JSONObject resultJson = new JSONObject(result);
String errmsg = (String) resultJson.get("errmsg");
if (errmsg == null) {
//用户是否订阅该公众号标识(0代表此用户没有关注该公众号 1表示关注了该公众号)。
Integer subscribe = (Integer) resultJson.get("subscribe");
return subscribe;
}
return -1;
}
/**
* Description: 推送消息给用户
*
* @author dsn
* @date 2018年9月21日 上午11:44:09
* @param openId
* @param title
* @param carname
* @version 1.0
*/
public static void sendMsg(String openId, String title, String carname) {
final String appid = "";
final String sercet = "";
//用户是否订阅该公众号标识 (0代表此用户没有关注该公众号 1表示关注了该公众号)
Integer state = WeiXinUtils.subscribeState(openId);
// 绑定了微信并且关注了服务号的用户 , 注册成功-推送注册短信
if (state == 1) {
String date = DateTime.now().toString("yyyy-MM-dd HH:mm:ss");
Map param = new HashMap<>();
param.put("first", new TemplateData(title, "#FF0000"));
param.put("keyword1", new TemplateData(carname, "#696969"));
param.put("keyword2", new TemplateData(date, "#696969"));
param.put("remark", new TemplateData("", "#696969"));
//注册的微信-模板Id
// String regTempId = WX_TemplateMsgUtil
// .getWXTemplateMsgId(WeiXinEnum.WX_TEMPLATE_MSG_NUMBER.CAR_REMIND.getMsgNumber(), appid, sercet);
//调用发送微信消息给用户的接口
WX_TemplateMsgUtil.sendWechatMsgToUser(openId, "模板ID", "", "#000000",
WX_TemplateMsgUtil.packJsonmsg(param), appid, sercet);
}
}
}
发送post请求封装的httpsRqquest方法也贴出来了:
/**
* 发送https请求
*
* @param requestUrl
* 请求地址
* @param requestMethod
* 请求方式(GET、POST)
* @param outputStr
* 提交的数据
* @return JSONObject(通过JSONObject.get(key)的方式获取json对象的属性值)
*/
public static JSONObject httpsRequest(String requestUrl, String requestMethod) {
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);
conn.setConnectTimeout(3000);
// 设置请求方式(GET/POST)
conn.setRequestMethod(requestMethod);
// conn.setRequestProperty("content-type",
// "application/x-www-form-urlencoded");
// 当outputStr不为null时向输出流写数据
// 从输入流读取返回内容
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.parseObject(buffer.toString());
} catch (ConnectException ce) {
logger.error("", ce);
} catch (Exception e) {
logger.error("", e);
}
return jsonObject;
}
还需要一个模板消息封装类WX_TemplateMsgUtil:
import java.util.Map;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import com.alibaba.fastjson.JSONObject;
public class WX_TemplateMsgUtil {
private static Logger log = LoggerFactory.getLogger(WX_TemplateMsgUtil.class);
/**
* 封装模板详细信息
*
* @return
*/
public static JSONObject packJsonmsg(Map param) {
JSONObject json = new JSONObject();
for (Map.Entry entry : param.entrySet()) {
JSONObject keyJson = new JSONObject();
TemplateData dta = entry.getValue();
keyJson.put("value", dta.getValue());
keyJson.put("color", dta.getColor());
json.put(entry.getKey(), keyJson);
}
return json;
}
/**
* 根据模板的编号 新增并获取模板ID
*
* @param templateSerialNumber
* 模板库中模板的 "编号"
* @return 模板ID
*/
public static String getWXTemplateMsgId(String templateSerialNumber, String appid, String secret) {
String tmpurl = "https://api.weixin.qq.com/cgi-bin/template/api_add_template?access_token="
+ WeiXinUtils.getAccessToken(appid, secret);
JSONObject json = new JSONObject();
json.put("template_id_short", templateSerialNumber);
JSONObject result = PayCommonUtil.HttpsRequest(tmpurl, "POST", json.toString());
JSONObject resultJson = new JSONObject(result);
String errmsg = (String) resultJson.get("errmsg");
// log.info("获取模板编号返回信息:" + errmsg);
if (!"ok".equals(errmsg)) {
return "error";
}
String templateId = (String) resultJson.get("template_id");
return templateId;
}
/**
* 根据模板ID 删除模板消息
*
* @param templateId
* 模板ID
* @return
*/
public static String deleteWXTemplateMsgById(String templateId, String appid, String secret) {
String tmpurl = "https://api.weixin.qq.com/cgi-bin/template/del_private_template?access_token="
+ WeiXinUtils.getAccessToken(appid, secret);
JSONObject json = new JSONObject();
json.put("template_id", templateId);
try {
JSONObject result = PayCommonUtil.HttpsRequest(tmpurl, "POST", json.toString());
JSONObject resultJson = new JSONObject(result);
// log.info("删除" + templateId + "模板消息,返回CODE:" + resultJson.get("errcode"));
String errmsg = (String) resultJson.get("errmsg");
if (!"ok".equals(errmsg)) {
return "error";
}
} catch (Exception e) {
e.printStackTrace();
}
return "success";
}
/**
* 发送微信消息(模板消息)
*
* @param touser
* 用户 OpenID
* @param templatId
* 模板消息ID
* @param clickurl
* URL置空,则在发送后,点击模板消息会进入一个空白页面(ios),或无法点击(android)。
* @param topcolor
* 标题颜色
* @param data
* 详细内容
* @return
*/
public static String sendWechatMsgToUser(String touser, String templatId, String clickurl, String topcolor,
JSONObject data, String appid, String secret) {
String tmpurl = "https://api.weixin.qq.com/cgi-bin/message/template/send?access_token="
+ WeiXinUtils.getAccessToken(appid, secret);
JSONObject json = new JSONObject();
json.put("touser", touser);
json.put("template_id", templatId);
json.put("url", clickurl);
json.put("topcolor", topcolor);
json.put("data", data);
try {
JSONObject result = PayCommonUtil.HttpsRequest(tmpurl, "POST", json.toString());
JSONObject resultJson = new JSONObject(result);
// log.info("发送微信消息返回信息:" + resultJson.get("errcode"));
String errmsg = (String) resultJson.get("errmsg");
if (!"ok".equals(errmsg)) { //如果为errmsg为ok,则代表发送成功,公众号推送信息给用户了。
return "error";
}
} catch (Exception e) {
e.printStackTrace();
return "error";
} finally {
if (templatId != null) {
//删除新增的 微信模板
// deleteWXTemplateMsgById(templatId, appid, secret);
}
}
return "success";
}
}
TemplateData类:
public class TemplateData {
public String getValue() {
return value;
}
public void setValue(String value) {
this.value = value;
}
public String getColor() {
return color;
}
public void setColor(String color) {
this.color = color;
}
private String value;
private String color;
public TemplateData(String value, String color) {
this.value = value;
this.color = color;
}
}
到这里就是一个完整的流程了,其实是很简单的,只要理解了流程。
当然还有一种用户授权登录,会弹出一个需要用户点击的确定,就可以获取用户的昵称,性别之类的信息。
https://blog.csdn.net/dsn727455218/article/details/65630151 找个地址就是授权登录的
需要注意的是a标签里面的scope为snsapi_userinfo 才能调起授权页面。
如有需要可以加我Q群【308742428】大家一起讨论技术。
后面会不定时为大家更新文章,敬请期待。