JAVA—咖啡馆

——欢迎访问rogerfan的博客,常来《JAVA——咖啡馆》坐坐,喝杯浓香的咖啡,彼此探讨一下JAVA技术,交流工作经验,分享JAVA带来的快乐!本网站部分转载文章,如果有版权问题请与我联系。

BlogJava 首页 新随笔 联系 聚合 管理
  447 Posts :: 145 Stories :: 368 Comments :: 0 Trackbacks

[-]

  1. 模版消息
    1. 具体调用方法
      1. 事件推送
      2. 返回码说明
  2. 一开发模板消息SDK
  3. 二构造模板消息并发送
    1. 模版消息

由于柳峰老师的新书还没有出来,网上也没有过多介绍基于Java语言的开发微信公众平台模板消息的例子,因此有了本文。由于个人表达能力和编程能力有限,请多多包涵。本文仅介绍拥有模板消息权限的微信公众账号开发。

本文分为以下两部分:

1.开发模板消息SDK

2.构造模板消息并发送

首先看一下模板消息接口文档:

模版消息

为了保证用户不受到骚扰,在开发者出现需要主动提醒、通知用户时,才允许开发者在公众平台网站中模板消息库中选择模板,选择后获得模板ID,再根据模板ID向用户主动推送提醒、通知消息。


模板消息调用时主要需要模板ID和模板中各参数的赋值内容。请注意:

1.模板中参数内容必须以".DATA"结尾,否则视为保留字;

2.模板保留符号"{{ }}"

具体调用方法

第一步:获取模板ID

通过在模板消息功能的模板库中使用需要的模板,可以获得模板ID。

第二步:请求接口

请注意,URL置空,则在发送后,点击模板消息会进入一个空白页面(ios),或无法点击(android)。

POST请求

https://api.weixin.qq.com/cgi-bin/message/template/send?access_token=ACCESS_TOKEN

请求包为一个json:

{
"touser":"OPENID",
"template_id":"ngqIpbwh8bUfcSsECmogfXcV14J0tQlEpBO27izEYtY",
"url":"http://weixin.qq.com/download",
"topcolor":"#FF0000",
"data":{
"User": {
"value":"黄先生",
"color":"#173177"
},
"Date":{
"value":"06月07日 19时24分",
"color":"#173177"
},
"CardNumber":{
"value":"0426",
"color":"#173177"
},
"Type":{
"value":"消费",
"color":"#173177"
},
"Money":{
"value":"人民币260.00元",
"color":"#173177"
},
"DeadTime":{
"value":"06月07日19时24分",
"color":"#173177"
},
"Left":{
"value":"6504.09",
"color":"#173177"
}
}
}

发送效果图:

 


事件推送

在模版消息发送任务完成后,微信服务器会将是否送达成功作为通知,发送到开发者中心中填写的服务器配置地址中。

1、送达成功时,推送的XML如下:

<xml>
<ToUserName><![CDATA[gh_7f083739789a]]></ToUserName>
<FromUserName><![CDATA[oia2TjuEGTNoeX76QEjQNrcURxG8]]&g;</FromUserName>
<CreateTime>1395658920</CreateTime>
<MsgType><![CDATA[event
]]></MsgType>
<Event><![CDATA[TEMPLATESENDJOBFINISH]]></Event>
<MsgID>200163836</MsgID>
<Status><![CDATA[success]]></Status>
</xml>

2、送达由于用户拒收(用户设置拒绝接收公众号消息)而失败时,推送的XML如下:

<xml>
<ToUserName><![CDATA[gh_7f083739789a]]></ToUserName>
<FromUserName><![CDATA[oia2TjuEGTNoeX76QEjQNrcURxG8]]></FromUserName>
<CreateTime>1395658984</CreateTime>
<MsgType><![CDATA[event]]></MsgType>
<Event><![CDATA[TEMPLATESENDJOBFINISH]]></Event>
<MsgID>200163840</MsgID>
<Status><![CDATA[failed:user block]]></Status>
</xml>

3、送达由于其他原因失败时,推送的XML如下:

<xml>
<ToUserName><![CDATA[gh_7f083739789a]]></ToUserName>
<FromUserName><![CDATA[oia2TjuEGTNoeX76QEjQNrcURxG8]]></FromUserName>
<CreateTime>1395658984</CreateTime>
<MsgType><![CDATA[event]]></MsgType>
<Event><![CDATA[TEMPLATESENDJOBFINISH]]></Event>
<MsgID>200163840</MsgID>
<Status><![CDATA[failed: system failed]]></Status>
</xml>

返回码说明

在调用模板消息接口后,会返回JSON数据包。正常时的返回JSON数据包示例:

{
"errcode":0,
"errmsg":"ok",
"msgid":200228332
}

错误时的返回JSON数据,形式类似,错误码请见本页下方返回码说明。


返回码 说明
-1 系统繁忙
0 请求成功
40001 验证失败
40002 不合法的凭证类型
40003 不合法的OpenID
40004 不合法的媒体文件类型
40005 不合法的文件类型
40006 不合法的文件大小
40007 不合法的媒体文件id
40008 不合法的消息类型
40009 不合法的图片文件大小
40010 不合法的语音文件大小
40011 不合法的视频文件大小
40012 不合法的缩略图文件大小
40013 不合法的APPID
41001 缺少access_token参数
41002 缺少appid参数
41003 缺少refresh_token参数
41004 缺少secret参数
41005 缺少多媒体文件数据
41006 access_token超时
42001 需要GET请求
43002 需要POST请求
43003 需要HTTPS请求
44001 多媒体文件为空
44002 POST的数据包为空
44003 图文消息内容为空
45001 多媒体文件大小超过限制
45002 消息内容超过限制
45003 标题字段超过限制
45004 描述字段超过限制
45005 链接字段超过限制
45006 图片链接字段超过限制
45007 语音播放时间超过限制
45008 图文消息超过限制
45009 接口调用超过限制
46001 不存在媒体数据
47001 解析JSON/XML内容错误

一、开发模板消息SDK

模板消息的定义如下:

模板消息也是使用access_token作为授权来发送。但是请大家注意:这里的access_token与网页授权的access_token完全不是一回事。可不要拿网页授权的access_token当作参数传递。

获取模板消息access_token:

String appId = "xxxxxxxxxxxxx"; //公众账号的唯一标识

String appSecret="xxxxxxxxxxx"; //公众账号的密钥

Token token = CommonUtil.getToken(appId, appSecret);

String access_token = token.getAccessToken();

#CommonUtil工具类具体代码:

 

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;

/**
 * 通用工具类
 * 
 * 
@author liufeng
 
*/

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;
    }

    
}


#Token实体类具体代码:


/**
 * 凭证实体类
 * 
 * 
@author liufeng
 
*/

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;
    }

    
}

 


至此,我们获得了模板消息的access_token。将它作为参数传递到模板消息接口文档中POST请求的ACCESS_TOKEN。

//得到构造好的模板消息请求地址

String requestURL = "https://api.weixin.qq.com/cgi-bin/message/template/send?access_token="+access_token;


二、构造模板消息并发送

我们以一个餐饮行业的收到新订单通知的模板为例,模板详情如下:

  • 模板ID

    YtO2XATY0VtRbgE4jWWNl-Zdc992FDdguhMUbomNkA0
    开发者调用模板消息接口时需提供模板ID

  • 标题收到新订单通知
  • 行业餐饮 - 餐饮
  • 详细内容
    {{first.DATA}}
    
    日期:{{Day.DATA}} 
    订单编号:{{orderId.DATA}}
    订单类型:{{orderType.DATA}}
    联系人:{{customerName.DATA}} 
    联系电话:{{customerPhone.DATA}} 
    {{remark.DATA}}
    在发送时,需要将内容中的参数({{.DATA}}内为参数)赋值替换为需要的信息
  • 内容示例
    收到一个新的订单,确认接受请回复0,拒绝请回复1。
    
    日期:2014-10-10 
    订单编号:1002
    订单类型:订位 
    联系人:陈丑丑 
    联系电话:13222222222
    订单金额:100.00元
    付款状态:已微信支付  
    
    请及时处理您的订单。 
    
  • 按照上述条件,我们的消息体构造如下:
  • {"data":{"customerName":{"color":"#173177","value":"陈丑丑"},"customerPhone":{"color":"#173177","value":"13222222222"},"Day":{"color":"#173177","value":"15时06分"},"first":{"color":"#173177","value":"收到一个新的订单"},"orderId":{"color":"#173177","value":"1002"},"orderType":{"color":"#173177","value":"订位"},"remark":{"color":"#173177","value":"请及时处理您的订单"}},"template_id":"YtO2XATY0VtRbgE4jWWNl-Zdc992FDdguhMUbomNkA0","topcolor":"#173177","touser":"orsrOt9qBgO6dC-F3IL_MF52eplI","url":"http://weixin.qq.com/download"}
  • 该消息体为一个json。具体封装如下:

  1/**
  2 * 餐饮行业收到新订单通知模板消息实体类
  3 * @author xjw
  4 *
  5 */

  6public class NewOrdersTemplate {
  7    private String touser; //用户OpenID
  8    
  9    private String template_id; //模板消息ID
 10    
 11    private String url; //URL置空,则在发送后,点击模板消息会进入一个空白页面(ios),或无法点击(android)。
 12    
 13    private String topcolor; //标题颜色
 14    
 15    private Data data; //详细内容
 16
 17    public String getTouser() {
 18        return touser;
 19    }

 20
 21    public void setTouser(String touser) {
 22        this.touser = touser;
 23    }

 24
 25    public String getTemplate_id() {
 26        return template_id;
 27    }

 28
 29    public void setTemplate_id(String templateId) {
 30        template_id = templateId;
 31    }

 32
 33    public String getUrl() {
 34        return url;
 35    }

 36
 37    public void setUrl(String url) {
 38        this.url = url;
 39    }

 40
 41    public String getTopcolor() {
 42        return topcolor;
 43    }

 44
 45    public void setTopcolor(String topcolor) {
 46        this.topcolor = topcolor;
 47    }

 48
 49    public Data getData() {
 50        return data;
 51    }

 52
 53    public void setData(Data data) {
 54        this.data = data;
 55    }

 56}

 57
 58/**
 59 * "餐饮行业收到新订单通知"模板消息详细内容实体类
 60 * @author xjw
 61 *
 62 */

 63public class Data {
 64    private Data_first first;
 65    
 66    private Data_Day Day; //日期
 67    
 68    private Data_orderId orderId; //订单编号
 69    
 70    private Data_orderType orderType; //订单类型
 71    
 72    private Data_customerName customerName; //联系人
 73    
 74    private Data_customerPhone customerPhone; //联系电话
 75    
 76    private Data_remark remark;
 77
 78    public Data_first getFirst() {
 79        return first;
 80    }

 81
 82    public void setFirst(Data_first first) {
 83        this.first = first;
 84    }

 85
 86    
 87
 88    public Data_Day getDay() {
 89        return Day;
 90    }

 91
 92    public void setDay(Data_Day day) {
 93        Day = day;
 94    }

 95
 96    public Data_orderId getOrderId() {
 97        return orderId;
 98    }

 99
100    public void setOrderId(Data_orderId orderId) {
101        this.orderId = orderId;
102    }

103
104    public Data_orderType getOrderType() {
105        return orderType;
106    }

107
108    public void setOrderType(Data_orderType orderType) {
109        this.orderType = orderType;
110    }

111
112    public Data_customerName getCustomerName() {
113        return customerName;
114    }

115
116    public void setCustomerName(Data_customerName customerName) {
117        this.customerName = customerName;
118    }

119
120    public Data_customerPhone getCustomerPhone() {
121        return customerPhone;
122    }

123
124    public void setCustomerPhone(Data_customerPhone customerPhone) {
125        this.customerPhone = customerPhone;
126    }

127
128    public Data_remark getRemark() {
129        return remark;
130    }

131
132    public void setRemark(Data_remark remark) {
133        this.remark = remark;
134    }

135}

136/**
137 * first
138 * @author xjw
139 *
140 */

141public class Data_first {
142    private String value;
143    
144    private String color;
145
146    public String getValue() {
147        return value;
148    }

149
150    public void setValue(String value) {
151        this.value = value;
152    }

153
154    public String getColor() {
155        return color;
156    }

157
158    public void setColor(String color) {
159        this.color = color;
160    }

161    
162    
163}

164
165/**
166 * 日期
167 * @author xjw
168 *
169 */

170public class Data_Day {
171    
172    private String value;
173    
174    private String color;
175
176    public String getValue() {
177        return value;
178    }

179
180    public void setValue(String value) {
181        this.value = value;
182    }

183
184    public String getColor() {
185        return color;
186    }

187
188    public void setColor(String color) {
189        this.color = color;
190    }

191    
192    
193}

194
195
196
197
198/**
199 * 订单标号
200 * @author xjw
201 *
202 */

203public class Data_orderId {
204
205    private String value;
206    
207    private String color;
208
209    public String getValue() {
210        return value;
211    }

212
213    public void setValue(String value) {
214        this.value = value;
215    }

216
217    public String getColor() {
218        return color;
219    }

220
221    public void setColor(String color) {
222        this.color = color;
223    }

224  
225}

226
227/**
228 * 订单类型
229 * @author xjw
230 *
231 */

232public class Data_orderType {
233
234    private String value;
235    
236    private String color;
237
238    public String getValue() {
239        return value;
240    }

241
242    public void setValue(String value) {
243        this.value = value;
244    }

245
246    public String getColor() {
247        return color;
248    }

249
250    public void setColor(String color) {
251        this.color = color;
252    }

253    
254    
255}

256
257/**
258 * 联系人
259 * @author xjw
260 *
261 */

262public class Data_customerName {
263
264    private String value;
265    
266    private String color;
267
268
269    public String getValue() {
270        return value;
271    }

272
273    public void setValue(String value) {
274        this.value = value;
275    }

276
277    public String getColor() {
278        return color;
279    }

280
281    public void setColor(String color) {
282        this.color = color;
283    }

284    
285    
286}

287
288/**
289 * 联系电话
290 * @author xjw
291 *
292 */

293public class Data_customerPhone {
294
295    private String value;
296    
297    private String color;
298
299
300    public String getValue() {
301        return value;
302    }

303
304    public void setValue(String value) {
305        this.value = value;
306    }

307
308    public String getColor() {
309        return color;
310    }

311
312    public void setColor(String color) {
313        this.color = color;
314    }

315    
316    
317}

318
319/**
320 * remark
321 * @author xjw
322 *
323 */

324public class Data_remark {
325    
326    private String value;
327    
328    private String color;
329
330
331    public String getValue() {
332        return value;
333    }

334
335    public void setValue(String value) {
336        this.value = value;
337    }

338
339    public String getColor() {
340        return color;
341    }

342
343    public void setColor(String color) {
344        this.color = color;
345    }

346    
347    
348}

349
350/**
351     * 发送模板消息
352     * appId 公众账号的唯一标识
353     * appSecret 公众账号的密钥
354     * openId 用户标识
355     */

356    public void send_template_message(String appId, String appSecret, String openId) {
357
358
359        Token token = CommonUtil.getToken(appId, appSecret);
360        String access_token = token.getAccessToken();
361        String url = "https://api.weixin.qq.com/cgi-bin/message/template/send?access_token="+access_token;
362
363
364        NewOrdersTemplate temp = new NewOrdersTemplate();
365        Data data = new Data();
366        Data_first first = new Data_first();
367        Data_Day Day = new Data_Day();
368        Data_orderId orderId = new Data_orderId();
369        Data_orderType orderType = new Data_orderType();
370        Data_customerName customerName = new Data_customerName();
371        Data_customerPhone customerPhone = new Data_customerPhone();
372        Data_remark remark = new Data_remark();
373
374
375        first.setValue("收到一个新的订单");
376        first.setColor("#173177");
377        Day.setValue("14时56分");
378        Day.setColor("#173177");
379        orderId.setValue("1002");
380        orderId.setColor("#173177");
381        orderType.setValue("订位");
382        orderType.setColor("#173177");
383        customerName.setValue("陈丑丑");
384        customerName.setColor("#173177");
385        customerPhone.setValue("13222222222");
386        customerPhone.setColor("#173177");
387        remark.setValue("请及时处理您的订单");
388        remark.setColor("#173177");
389
390
391        data.setFirst(first);
392        data.setDay(Day);
393        data.setOrderId(orderId);
394        data.setOrderType(orderType);
395        data.setCustomerName(customerName);
396        data.setCustomerPhone(customerPhone);
397        data.setRemark(remark);
398        temp.setTouser(openId);
399        temp.setTemplate_id("YtO2XATY0VtRbgE4jWWNl-Zdc992FDdguhMUbomNkA0");
400        temp.setUrl("http://weixin.qq.com/download");
401        temp.setTopcolor("#173177");
402        temp.setData(data);
403        
404        String jsonString = JSONObject.fromObject(temp).toString().replace("day""Day");
405        JSONObject jsonObject = WeixinUtil.httpRequest(url, "POST", jsonString);
406        System.out.println(jsonObject);
407        int result = 0;
408        if (null != jsonObject) {  
409             if (0 != jsonObject.getInt("errcode")) {  
410                 result = jsonObject.getInt("errcode");  
411                  log.error("错误 errcode:{} errmsg:{}", jsonObject.getInt("errcode"), jsonObject.getString("errmsg"));  
412             }
  
413         }

414        log.info("模板消息发送结果:"+result);
415    }

416
417


WeixinUtil工具类在柳峰老师的博客里有源码。。。。。

实现效果如下:


如果小伙伴们觉得有什么问题可以提出,大家共同学习。

posted on 2016-09-23 17:02 rogerfan 阅读(902) 评论(0)  编辑  收藏 所属分类: 【开源技术】

只有注册用户登录后才能发表评论。


网站导航: