微信開發之獲取openid及推送模板訊息
阿新 • • 發佈:2018-12-29
有很多的朋友再問我怎麼獲取code,openid之類的問題,在這裡我就給大家分享一下。
在做微信支付是需要獲取openid的,推送模板訊息也是需要openid包括其他一些功能分享等也都是需要的,openid是什麼呢,這裡給大家解釋一下,是使用者在公眾號中的一個標識,就相當於你資料表中的ID一樣,一個身份標識。
不多說廢話了這裡直接上程式碼了:
這裡寫一個了a標籤來獲取code,需要注意的是redirect_uri=接收code的回撥地址(是帶http://格式)可以指向下面的register
appid:是公眾號的ID,appsecrect:是公眾號的金鑰,code:就是上面a標籤重定向的獲取到的
給一個封裝的WeiXinUtils類:
傳送post請求封裝的httpsRqquest方法也貼出來了:
還需要一個模板訊息封裝類WX_TemplateMsgUtil:
TemplateData類:
到這裡就是一個完整的流程了,其實是很簡單的,只要理解了流程。
當然還有一種使用者授權登入,會彈出一個需要使用者點選的確定,就可以獲取使用者的暱稱,性別之類的資訊。
https://blog.csdn.net/dsn727455218/article/details/65630151 找個地址就是授權登入的
需要注意的是a標籤裡面的scope為snsapi_userinfo 才能調起授權頁面。
在做微信支付是需要獲取openid的,推送模板訊息也是需要openid包括其他一些功能分享等也都是需要的,openid是什麼呢,這裡給大家解釋一下,是使用者在公眾號中的一個標識,就相當於你資料表中的ID一樣,一個身份標識。
不多說廢話了這裡直接上程式碼了:
這裡寫一個了a標籤來獲取code,需要注意的是redirect_uri=接收code的回撥地址(是帶http://格式)可以指向下面的register
<a style="color: #fff" href="https://open.weixin.qq.com/connect/oauth2/authorize?appid=你公眾號的appid&redirect_uri=接收code的回撥地址(是帶http://格式)&response_type=code&scope=snsapi_base&state=wxsq123#wechat_redirect">獲取openid</a>
appid:是公眾號的ID,appsecrect:是公眾號的金鑰,code:就是上面a標籤重定向的獲取到的
@RequestMapping(params = "method=register", method = RequestMethod.POST) public @ResponseBody Map<String, Object> 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<BR> * * @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<BR> * * @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 <BR> * * @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<BR> * * @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 獲取使用者是否關注公眾號<BR> * * @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: 推送訊息給使用者<BR> * * @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<String, TemplateData> 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<String, TemplateData> param) {
JSONObject json = new JSONObject();
for (Map.Entry<String, TemplateData> 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 才能調起授權頁面。