前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >Java微信公众平台开发(五)--文本及图文消息回复的实现

Java微信公众平台开发(五)--文本及图文消息回复的实现

作者头像
用户2417870
发布于 2019-09-18 03:35:38
发布于 2019-09-18 03:35:38
86700
代码可运行
举报
文章被收录于专栏:g歌德ag歌德a
运行总次数:0
代码可运行

上篇我们说到回复消息可以根据是否需要上传文件到微信服务器可划分为【普通消息】和【多媒体消息】,这里我们来讲述普通消息的回复实现,在消息回复中存在一个关键字段【openid】,它是微信用户对于公众号的唯一标识,这里不做过多解释后面将给出时间专门来讲解微信生态中的关键字!

(一)回复文本消息

在前面我们已经完成了对消息的分类和回复消息实体的建立,这里回复文本消息需要用到的就是我们的TextMessage,我们把回复文本消息在【文本消息】类型中给出回复!在我们做消息回复的时候需要设置消息的接收人ToUserName(openid)、消息的发送方FromUserName、消息类型MsgType、创建时间CreateTime以及消息体Content,由于我们我们的消息回复格式是需要为xml,所以最终我们需要将其装换成xml再做返回输出!

首先我们在工具类MessageUtil的代码做出部分修改和添加,实现最后版本为:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
  1 package com.gede.wechat.util;
  2 import java.io.InputStream;
  3 import java.io.Writer;
  4 import java.util.HashMap;
  5 import java.util.List;
  6 import java.util.Map;
  7 import javax.servlet.http.HttpServletRequest;
  8 import org.dom4j.Document;
  9 import org.dom4j.Element;
 10 import org.dom4j.io.SAXReader;
 11 import com.gede.wechat.response.*;
 12 import com.thoughtworks.xstream.XStream;
 13 import com.thoughtworks.xstream.core.util.QuickWriter;
 14 import com.thoughtworks.xstream.io.HierarchicalStreamWriter;
 15 import com.thoughtworks.xstream.io.xml.DomDriver;
 16 import com.thoughtworks.xstream.io.xml.PrettyPrintWriter;
 17 import com.thoughtworks.xstream.io.xml.XppDriver;
 18 
 19 /**
 20 * @author gede
 21 * @version date:2019年5月22日 下午3:44:27
 22 * @description :
 23 */
 24 public class MessageUtil {
 25 
 26     /**
 27      * 返回消息类型:文本
 28      */
 29     public static final String RESP_MESSAGE_TYPE_TEXT = "text";
 30 
 31     /**
 32      * 返回消息类型:音乐
 33      */
 34     public static final String RESP_MESSAGE_TYPE_MUSIC = "music";
 35 
 36     /**
 37      * 返回消息类型:图文
 38      */
 39     public static final String RESP_MESSAGE_TYPE_NEWS = "news";
 40 
 41     /**
 42      * 返回消息类型:图片
 43      */
 44     public static final String RESP_MESSAGE_TYPE_Image = "image";
 45 
 46     /**
 47      * 返回消息类型:语音
 48      */
 49     public static final String RESP_MESSAGE_TYPE_Voice = "voice";
 50 
 51     /**
 52      * 返回消息类型:视频
 53      */
 54     public static final String RESP_MESSAGE_TYPE_Video = "video";
 55 
 56     /**
 57      * 请求消息类型:文本
 58      */
 59     public static final String REQ_MESSAGE_TYPE_TEXT = "text";
 60 
 61     /**
 62      * 请求消息类型:图片
 63      */
 64     public static final String REQ_MESSAGE_TYPE_IMAGE = "image";
 65 
 66     /**
 67      * 请求消息类型:链接
 68      */
 69     public static final String REQ_MESSAGE_TYPE_LINK = "link";
 70 
 71     /**
 72      * 请求消息类型:地理位置
 73      */
 74     public static final String REQ_MESSAGE_TYPE_LOCATION = "location";
 75 
 76     /**
 77      * 请求消息类型:音频
 78      */
 79     public static final String REQ_MESSAGE_TYPE_VOICE = "voice";
 80 
 81     /**
 82      * 请求消息类型:视频
 83      */
 84     public static final String REQ_MESSAGE_TYPE_VIDEO = "video";
 85 
 86     /**
 87      * 请求消息类型:推送
 88      */
 89     public static final String REQ_MESSAGE_TYPE_EVENT = "event";
 90 
 91     /**
 92      * 事件类型:subscribe(订阅)
 93      */
 94     public static final String EVENT_TYPE_SUBSCRIBE = "subscribe";
 95 
 96     /**
 97      * 事件类型:unsubscribe(取消订阅)
 98      */
 99     public static final String EVENT_TYPE_UNSUBSCRIBE = "unsubscribe";
100 
101     /**
102      * 事件类型:CLICK(自定义菜单点击事件)
103      */
104     public static final String EVENT_TYPE_CLICK = "CLICK";
105 
106     /**
107      * 事件类型:VIEW(自定义菜单 URl 视图)
108      */
109     public static final String EVENT_TYPE_VIEW = "VIEW";
110 
111     /**
112      * 事件类型:LOCATION(上报地理位置事件)
113      */
114     public static final String EVENT_TYPE_LOCATION = "LOCATION";
115 
116     /**
117      * 事件类型:LOCATION(上报地理位置事件)
118      */
119     public static final String EVENT_TYPE_SCAN = "SCAN";
120 
121     @SuppressWarnings("unchecked")
122     public static Map<String, String> parseXml(HttpServletRequest request)
123             throws Exception {
124         // 将解析结果存储在 HashMap 中
125         Map<String, String> map = new HashMap<String, String>();
126         // 从 request 中取得输入流
127         InputStream inputStream = request.getInputStream();
128         // 读取输入流
129         SAXReader reader = new SAXReader();
130         Document document = reader.read(inputStream);
131         // 得到 xml 根元素
132         Element root = document.getRootElement();
133         // 得到根元素的所有子节点
134         List<Element> elementList = root.elements();
135 
136         // 遍历所有子节点
137         for (Element e : elementList)
138             map.put(e.getName(), e.getText());
139 
140         // 释放资源
141         inputStream.close();
142         inputStream = null;
143 
144         return map;
145     }
146 
147     public static String textMessageToXml(TextMessage textMessage) {
148         XStream xstream = new XStream(new DomDriver("utf-8"));
149         xstream.alias("xml", textMessage.getClass());
150         return xstream.toXML(textMessage);
151     }
152     public static String newsMessageToXml(NewsMessage newsMessage) {
153         xstream.alias("xml", newsMessage.getClass());
154         xstream.alias("item", new Article().getClass());
155         return xstream.toXML(newsMessage);
156     }
157 
158     public static String imageMessageToXml(ImageMessage imageMessage) {
159         xstream.alias("xml", imageMessage.getClass());
160         return xstream.toXML(imageMessage);
161     }
162 
163     public static String voiceMessageToXml(VoiceMessage voiceMessage) {
164         xstream.alias("xml", voiceMessage.getClass());
165         return xstream.toXML(voiceMessage);
166     }
167 
168     public static String videoMessageToXml(VideoMessage videoMessage) {
169         xstream.alias("xml", videoMessage.getClass());
170         return xstream.toXML(videoMessage);
171     }
172 
173     public static String musicMessageToXml(MusicMessage musicMessage) {
174         xstream.alias("xml", musicMessage.getClass());
175         return xstream.toXML(musicMessage);
176     }
177 
178     /**
179      * 对象到 xml 的处理
180      */
181     private static XStream xstream = new XStream(new XppDriver() {
182         public HierarchicalStreamWriter createWriter(Writer out) {
183             return new PrettyPrintWriter(out) {
184                 // 对所有 xml 节点的转换都增加 CDATA 标记
185                 boolean cdata = true;
186 
187                 @SuppressWarnings("rawtypes")
188                 public void startNode(String name, Class clazz) {
189                     super.startNode(name, clazz);
190                 }
191 
192                 protected void writeText(QuickWriter writer, String text) {
193                     if (cdata) {
194                         writer.write("<![CDATA[");
195                         writer.write(text);
196                         writer.write("]]>");
197                     } else {
198                         writer.write(text);
199                     }
200                 }
201             };
202         }
203     });
204 }

我们回复文本消息的简单实现:修改MsgDispatcher,在消息分类为【文本消息】中加入如下代码:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
 1 package com.gede.wechat.dispatcher;
 2 
 3 import java.util.Date;
 4 import java.util.Map;
 5 
 6 import com.gede.wechat.response.TextMessage;
 7 import com.gede.wechat.util.MessageUtil;
 8 
 9 /**
10 * @author gede
11 * @version date:2019年5月23日 下午6:49:11
12 * @description :
13 */
14 public class MsgDispatcher {
15     public static String processMessage(Map<String, String> map) {
16         String openid=map.get("FromUserName"); //用户openid
17         String mpid=map.get("ToUserName");   //公众号原始ID
18         //普通文本消息
19         TextMessage txtmsg=new TextMessage();
20         txtmsg.setToUserName(openid);
21         txtmsg.setFromUserName(mpid);
22         txtmsg.setCreateTime(new Date().getTime());
23         txtmsg.setMsgType(MessageUtil.RESP_MESSAGE_TYPE_TEXT);
24         if (map.get("MsgType").equals(MessageUtil.REQ_MESSAGE_TYPE_TEXT)) { // 文本消息
25             txtmsg.setContent("你好,欢迎您的关注!");
26             return MessageUtil.textMessageToXml(txtmsg);
27         }
28         if (map.get("MsgType").equals(MessageUtil.REQ_MESSAGE_TYPE_IMAGE)) { // 图片消息
29             System.out.println("==============这是图片消息!");
30         }
31         if (map.get("MsgType").equals(MessageUtil.REQ_MESSAGE_TYPE_LINK)) { // 链接消息
32             System.out.println("==============这是链接消息!");
33         }
34         if (map.get("MsgType").equals(MessageUtil.REQ_MESSAGE_TYPE_LOCATION)) { // 位置消息
35             System.out.println("==============这是位置消息!");
36         }
37         if (map.get("MsgType").equals(MessageUtil.REQ_MESSAGE_TYPE_VIDEO)) { // 视频消息
38             System.out.println("==============这是视频消息!");
39         }
40         if (map.get("MsgType").equals(MessageUtil.REQ_MESSAGE_TYPE_VOICE)) { // 语音消息
41             System.out.println("==============这是语音消息!");
42         }
43  
44         return null;
45     }
46 }

此时从逻辑上来说,代码已完成,但是从完整的微信响应来看,我们只是完成了回复内容的编辑,并没有去响应微信服务器让服务器去回复消息,所以我们还需要修改WechatSecurity这个控制类,修改的时候我们还要主要本地服务器和微信服务器编码的问题,为了避免麻烦我们统一设置成utf-8。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
 1 package com.gede.wechat.controller;
 2 
 3 import java.io.PrintWriter;
 4 import java.util.Map;
 5 
 6 import javax.servlet.http.HttpServletRequest;
 7 import javax.servlet.http.HttpServletResponse;
 8 
 9 import org.apache.log4j.Logger;
10 import org.springframework.stereotype.Controller;
11 import org.springframework.web.bind.annotation.RequestMapping;
12 import org.springframework.web.bind.annotation.RequestMethod;
13 import org.springframework.web.bind.annotation.RequestParam;
14 
15 import com.gede.wechat.dispatcher.EventDispatcher;
16 import com.gede.wechat.dispatcher.MsgDispatcher;
17 import com.gede.wechat.util.MessageUtil;
18 import com.gede.wechat.util.SignUtil;
19 
20 /**
21 * @author gede
22 * @version date:2019年5月22日 下午2:53:46
23 * @description :
24 */
25 @Controller
26 @RequestMapping("/wechat")
27 public class WechatSecurity {
28     private static Logger logger = Logger.getLogger(WechatSecurity.class);
29  
30     @RequestMapping(value = "security", method = RequestMethod.GET)
31     public void doGet(
32             HttpServletRequest request,
33             HttpServletResponse response,
34             @RequestParam(value = "signature", required = true) String signature,
35             @RequestParam(value = "timestamp", required = true) String timestamp,
36             @RequestParam(value = "nonce", required = true) String nonce,
37             @RequestParam(value = "echostr", required = true) String echostr) {
38         try {
39             if (SignUtil.checkSignature(signature, timestamp, nonce)) {
40                 PrintWriter out = response.getWriter();
41                 out.print(echostr);
42                 out.close();
43             } else {
44                 logger.info("这里存在非法请求!");
45             }
46         } catch (Exception e) {
47             logger.error(e, e);
48         }
49     }
50  
51     /**
52      * @Description: 接收微信端消息处理并做分发
53      * @param @param request
54      * @param @param response   
55      * @author dapengniao
56      * @date 2016年3月7日 下午4:06:47
57      */
58     @RequestMapping(value = "security", method = RequestMethod.POST)
59     public void DoPost(HttpServletRequest request,HttpServletResponse response) {
60         try{
61              Map<String, String> map=MessageUtil.parseXml(request);
62              String msgtype=map.get("MsgType");
63                if(MessageUtil.REQ_MESSAGE_TYPE_EVENT.equals(msgtype)){
64                     request.setCharacterEncoding("UTF-8");
65                     response.setCharacterEncoding("UTF-8");
66                     String msgrsp=EventDispatcher.processEvent(map); //进入事件处理
67                     PrintWriter out = response.getWriter();
68                     out.print(msgrsp);
69                     out.close();
70                 }else{
71                     request.setCharacterEncoding("UTF-8");
72                     response.setCharacterEncoding("UTF-8");
73                     String msgrsp =MsgDispatcher.processMessage(map); //进入消息处理
74                     PrintWriter out = response.getWriter();
75                     out.print(msgrsp);
76                     out.close();
77                 }
78         }catch(Exception e){
79             logger.error(e,e);
80         }
81     }
82 }

启动项目,当我们发送任何文本消息后我们可以看到我们的回复内容,如图:

(二)图文消息回复

图文消息的回复和文本消息的实现模式是一样的,只不过对应消息体的字段有所区别而已,这里为了和文本消息能有所区分我在【图片消息】实现图文消息的回复,修改MsgDispatcher:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
 1 NewsMessage newmsg=new NewsMessage();
 2         newmsg.setToUserName(openid);
 3         newmsg.setFromUserName(mpid);
 4         newmsg.setCreateTime(new Date().getTime());
 5         newmsg.setMsgType(MessageUtil.RESP_MESSAGE_TYPE_NEWS);
 6         if (map.get("MsgType").equals(MessageUtil.REQ_MESSAGE_TYPE_IMAGE)) { // 图片消息
 7             System.out.println("==============这是图片消息!");
 8             Article article=new Article();
 9             article.setDescription("这是图文消息1"); //图文消息的描述
10             article.setPicUrl("https://i.loli.net/2019/05/26/5cea3d137aa1469348.jpg"); //图文消息图片地址
11             article.setTitle("图文消息1");  //图文消息标题
12             article.setUrl("https://www.cnblogs.com/gede");  //图文url链接
13             List<Article> list=new ArrayList<Article>();
14             list.add(article);     //这里发送的是单图文,如果需要发送多图文则在这里list中加入多个Article即可!
15             newmsg.setArticleCount(list.size());
16             newmsg.setArticles(list);
17             return MessageUtil.newsMessageToXml(newmsg);
18         }      

启动项目,当我们发送任何图片消息后我们可以看到我们的回复内容,如图:

最后在这里分享一下自己一直使用的免费图床网站。如果图省事,直接进入这个网址,上传图片就行了,只不过服务器不在国内,有点慢。地址:https://sm.ms/

可以自己折腾自己服务器的话,就用我这个,附件下载,直接丢在自己的服务器上就可以。这个的服务器是新浪的。

附件:新浪图

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
面向对象设计的设计模式(七):外观模式
定义解读:通过这个高层接口,可以将客户端与子系统解耦:客户端可以不直接访问子系统,而是通过外观类间接地访问;同时也可以提高子系统的独立性和可移植性。
用户2932962
2019/03/18
9580
面向对象设计的设计模式(七):外观模式
设计模式(九)外观模式Facade(结构型)
在遇到以下情况使用facade模式: 1) 当你要为一个复杂子系统提供一个简单接口时。子系统往往因为不断演化而变得越来越复杂。大多数模式使用时都会产生更多更小的类。 这使得子系统更具可重用性,也更容易对子系统进行定制,但这也给那些不需要定制子系统的用户带来一些使用上的困难。facade可以提供一个简单的缺省视图, 这一视图对大多数用户来说已经足够,而那些需要更多的可定制性的用户可以越过facade层。 2) 客户程序与抽象类的实现部分之间存在着很大的依赖性。引入 facade将这个子系统与客户以及其他的子系统分离,可以提高子系统的独立性 和可移植性。 3) 当你需要构建一个层次结构的子系统时,使用 facade模式定义子系统中每层的入口点。如果子系统之间是相互依赖的,你可以让它们仅通过facade进行通讯,从而简化了它们之间的依赖关系。
黄规速
2022/04/14
4010
设计模式(九)外观模式Facade(结构型)
Java设计模式-外观模式
在以前,手机没有这么方便的时候,我们一旦需要去哪里哪里办个什么证,那真就的从这签个字从那签个字,签一路,才能办下那个证,
宁在春
2022/10/31
2600
Java设计模式-外观模式
C++设计模式——Facade外观模式
外观模式是一种结构型设计模式, 又称为门面模式,也是一种基于创建对象来实现的模式,为子系统中的各组接口的使用提供了统一的访问入口。
Coder-ZZ
2024/06/18
2120
C++设计模式——Facade外观模式
设计模式---外观模式
引入外观角色之后,用户只需要直接与外观角色交互,用户与子系统之间的复杂关系由外观角色来实现,从而降低了系统的耦合度。
大忽悠爱学习
2021/11/15
3090
【设计模式】第十篇:外观模式,开着小破车的快乐
不知道大家有没有这样开或者坐过这样一辆“小破车”,他能跑,但是内部娱乐或者说一些辅助的设备几乎可以忽略不计,条件虽然艰苦了一些,但是我们还是要自己给自己创造快乐 ,夏天太热了,先给自己安装一台空调,害,其实就是一台小电扇,接着就是我们的 360度音响体验了,其实也就是一个低音炮,来吧,最奢侈的一个设备来了,遮光板上接一个屏幕,还能连一个简单的 DVD 机器,好的吧,麻雀虽小,但是也算五脏俱全了,就像代码一样,毕竟有功能,能运行的程序就是 “好程序” 对吧哈哈哈~
BWH_Steven
2020/12/29
2500
【设计模式】第十篇:外观模式,开着小破车的快乐
设计之禅——外观模式
平时在我们生活中,我们常常会接触学习各种各样的新事物,而能够快速吸引留住大量客户的都有一个共性,就是简单易学好上手。比如,windows和linux系统,前者比后者更加普及的原因也就是不需要经过专业系统的学习就能轻松使用。同样的,这种思想在编程界有一个专业的术语——外观模式。
夜勿语
2020/09/07
3770
设计模式(十):结构型之外观模式
Java微观世界
2025/01/21
1010
设计模式(十):结构型之外观模式
Java设计模式(九)----外观模式
外观模式 一、定义 二、结构 三、案例 四、特点 一、定义 Facade(外观)模式为子系统中的各类(或结构与方法)提供一个简明一致的界面,隐藏子系统的复杂性,使子系统更加容易使用。 二、结构 门面(Facade)角色 :客户端可以调用这个角色的方法。此角色知晓相关的(一个或者多个)子系统的功能和责任。在正常情况下,本角色会将所有从客户端发来的请求委派到相应的子系统去。 子系统(SubSystem)角色 :可以同时有
汤高
2018/01/11
6390
外观模式详解
外观模式是一种结构型设计模式,为子系统中的一组接口提供一个一致的接口。外观模式定义了一个高层接口,使得子系统更易于使用。
码事漫谈
2024/12/20
1280
外观模式详解
设计模式之 外观模式
外观模式(又称门面模式),通过外观的包装,使应用程序只能看到外观对象,而不会看到具体的细节对象,这样无疑会降低应用程序的复杂度,并且提高了程序的可维护性。UML类图如下:
BUG弄潮儿
2021/02/03
2700
设计模式之 外观模式
设计模式--外观模式
外观模式是一种结构型设计模式,用于为一组复杂的子系统提供一个简单的接口。它隐藏了子系统的复杂性,并向客户端提供了一个简单的接口,使得客户端可以与子系统交互,而不用了解其内部的复杂逻辑。
软件架构师Michael
2023/06/29
1950
Java设计模式之外观模式
外观模式是一种常见的设计模式,它属于结构型模式。该模式提供了一个统一的接口,以隐藏系统的复杂性,并将其与客户端分离开来。在 Java 中,外观模式可以帮助我们简化复杂系统的使用,并提供更简洁的接口供客户端调用。
刺槐儿
2024/01/26
2690
详解设计模式:外观模式
外观模式(Facade Pattern),又称为门面模式,是 GoF 的 23 种设计模式中的一种结构型设计模式。
栗筝i
2022/12/02
4450
详解设计模式:外观模式
23种设计模式,外观模式实战
外观模式(Facade Pattern)是一种使用频率非常高的结构型设计模式,其核心思想是为子系统中的一组接口提供一个一致的界面。外观模式定义了一个高层接口,这个接口使得这一子系统更加容易使用。简而言之,外观模式就是客户端与复杂子系统之间的一个简单而统一的接口。
小马哥学JAVA
2024/04/06
1520
【Java设计模式】015-外观模式
外观(Facade)模式又叫作门面模式,是一种通过为多个复杂的子系统提供一个一致的接口,而使这些子系统更加容易被访问的模式。该模式对外有一个统一接口,外部应用程序不用关心内部子系统的具体细节,这样会大大降低应用程序的复杂度,提高了程序的可维护性。
訾博ZiBo
2025/01/06
800
图解Java设计模式之外观模式
组建一个家庭影院 : DVD 播放器、投影仪、自动屏幕、环绕立体声、爆米花机,要求完成使用家庭影院的功能,其过程为: 直接用遥控器:统筹各设备开关 开爆米花机 放下屏幕 开投影仪 开音响 开 DVD,选 dvd 去拿爆米花 调暗灯光 播放 观影结束后,关闭各种设备
海仔
2020/03/26
4870
设计模式之适配器模式与外观模式(二)
好了,通过上次的学习,我们已经知道适配器模式是如何将一个类的接口转换成另一个符合客户期望的接口。同时也知道在Java中要做到这一点,必须将一个不兼容接口的对象包装起来,变成兼容的对象。
程序员小跃
2019/12/26
3410
Java设计模式:外观模式之优雅门面(九)
在复杂的软件系统中,往往存在多个子系统或组件协同工作的情况。这些子系统可能由不同的团队开发,使用不同的技术栈,具有各自独特的接口和依赖关系。当客户端需要使用这些子系统时,可能需要了解并调用多个接口,处理复杂的依赖关系,这无疑增加了客户端的使用难度。
公众号:码到三十五
2024/04/10
1810
Java设计模式:外观模式之优雅门面(九)
Java二十三种设计模式-外观模式(9/23)
外观模式(Facade Pattern)是一种结构型设计模式,它为子系统中的一组接口提供一个统一的高层接口。外观模式定义了一个可以与复杂子系统交互的简化接口,使得子系统更加易于使用。
正在走向自律
2024/12/18
1280
Java二十三种设计模式-外观模式(9/23)
相关推荐
面向对象设计的设计模式(七):外观模式
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档