[037] 微信公众帐号开发教程第13篇-图文消息全攻略

引言及内容概要

已经有几位读者抱怨“柳峰只用到文本消息作为示例,从来不提图文消息,都不知道图文消息该如何使用”,好吧,我错了,原本以为把基础API封装完、框架搭建好,再给出一个文本消息的使用示例,大家就能够照猫画虎的,或许是因为我的绘画功底太差,画出的那只猫本来就不像猫吧……

本篇主要介绍微信公众帐号开发中图文消息的使用,以及图文消息的几种表现形式。标题取名为“图文消息全攻略”,这绝对不是标题党,是想借此机会把大家对图文消息相关的问题、疑虑、障碍全部清除掉。

图文消息的主要参数说明

通过微信官方的消息接口指南,可以看到对图文消息的参数介绍,如下图所示:

从图中可以了解到:

1)图文消息的个数限制为10,也就是图中ArticleCount的值(图文消息的个数,限制在10条以内);

2)对于多图文消息,第一条图文的图片显示为大图,其他图文的图片显示为小图;

3)第一条图文的图片大小建议为640*320,其他图文的图片大小建议为80*80;

好了,了解这些,再结合第4篇文章所讲的消息及消息处理工具的封装,想要回复图文消息给用户也就不是什么难事了。

图文消息的多种表现形式

下面直接通过代码演示图文消息最主要的五种表现形式的用法,源代码如下:

[java] view plaincopy

  1. package org.liufeng.course.service;
  2. import java.util.ArrayList;
  3. import java.util.Date;
  4. import java.util.List;
  5. import java.util.Map;
  6. import javax.servlet.http.HttpServletRequest;
  7. import org.liufeng.course.message.resp.Article;
  8. import org.liufeng.course.message.resp.NewsMessage;
  9. import org.liufeng.course.message.resp.TextMessage;
  10. import org.liufeng.course.util.MessageUtil;
  11. /**
  12. * 核心服务类
  13. *
  14. * @author liufeng
  15. * @date 2013-07-25
  16. */
  17. public class CoreService {
  18. /**
  19. * 处理微信发来的请求
  20. *
  21. * @param request
  22. * @return
  23. */
  24. public static String processRequest(HttpServletRequest request) {
  25. String respMessage = null;
  26. try {
  27. // xml请求解析
  28. Map<String, String> requestMap = MessageUtil.parseXml(request);
  29. // 发送方帐号(open_id)
  30. String fromUserName = requestMap.get("FromUserName");
  31. // 公众帐号
  32. String toUserName = requestMap.get("ToUserName");
  33. // 消息类型
  34. String msgType = requestMap.get("MsgType");
  35. // 默认回复此文本消息
  36. TextMessage textMessage = new TextMessage();
  37. textMessage.setToUserName(fromUserName);
  38. textMessage.setFromUserName(toUserName);
  39. textMessage.setCreateTime(new Date().getTime());
  40. textMessage.setMsgType(MessageUtil.RESP_MESSAGE_TYPE_TEXT);
  41. textMessage.setFuncFlag(0);
  42. // 由于href属性值必须用双引号引起,这与字符串本身的双引号冲突,所以要转义
  43. textMessage.setContent("欢迎访问<a href=\"http://blog.csdn.net/lyq8479\">柳峰的博客</a>!");
  44. // 将文本消息对象转换成xml字符串
  45. respMessage = MessageUtil.textMessageToXml(textMessage);
  46. // 文本消息
  47. if (msgType.equals(MessageUtil.REQ_MESSAGE_TYPE_TEXT)) {
  48. // 接收用户发送的文本消息内容
  49. String content = requestMap.get("Content");
  50. // 创建图文消息
  51. NewsMessage newsMessage = new NewsMessage();
  52. newsMessage.setToUserName(fromUserName);
  53. newsMessage.setFromUserName(toUserName);
  54. newsMessage.setCreateTime(new Date().getTime());
  55. newsMessage.setMsgType(MessageUtil.RESP_MESSAGE_TYPE_NEWS);
  56. newsMessage.setFuncFlag(0);
  57. List<Article> articleList = new ArrayList<Article>();
  58. // 单图文消息
  59. if ("1".equals(content)) {
  60. Article article = new Article();
  61. article.setTitle("微信公众帐号开发教程Java版");
  62. article.setDescription("柳峰,80后,微信公众帐号开发经验4个月。为帮助初学者入门,特推出此系列教程,也希望借此机会认识更多同行!");
  63. article.setPicUrl("http://0.xiaoqrobot.duapp.com/images/avatar_liufeng.jpg");
  64. article.setUrl("http://blog.csdn.net/lyq8479");
  65. articleList.add(article);
  66. // 设置图文消息个数
  67. newsMessage.setArticleCount(articleList.size());
  68. // 设置图文消息包含的图文集合
  69. newsMessage.setArticles(articleList);
  70. // 将图文消息对象转换成xml字符串
  71. respMessage = MessageUtil.newsMessageToXml(newsMessage);
  72. }
  73. // 单图文消息---不含图片
  74. else if ("2".equals(content)) {
  75. Article article = new Article();
  76. article.setTitle("微信公众帐号开发教程Java版");
  77. // 图文消息中可以使用QQ表情、符号表情
  78. article.setDescription("柳峰,80后," + emoji(0x1F6B9)
  79. + ",微信公众帐号开发经验4个月。为帮助初学者入门,特推出此系列连载教程,也希望借此机会认识更多同行!\n\n目前已推出教程共12篇,包括接口配置、消息封装、框架搭建、QQ表情发送、符号表情发送等。\n\n后期还计划推出一些实用功能的开发讲解,例如:天气预报、周边搜索、聊天功能等。");
  80. // 将图片置为空
  81. article.setPicUrl("");
  82. article.setUrl("http://blog.csdn.net/lyq8479");
  83. articleList.add(article);
  84. newsMessage.setArticleCount(articleList.size());
  85. newsMessage.setArticles(articleList);
  86. respMessage = MessageUtil.newsMessageToXml(newsMessage);
  87. }
  88. // 多图文消息
  89. else if ("3".equals(content)) {
  90. Article article1 = new Article();
  91. article1.setTitle("微信公众帐号开发教程\n引言");
  92. article1.setDescription("");
  93. article1.setPicUrl("http://0.xiaoqrobot.duapp.com/images/avatar_liufeng.jpg");
  94. article1.setUrl("http://blog.csdn.net/lyq8479/article/details/8937622");
  95. Article article2 = new Article();
  96. article2.setTitle("第2篇\n微信公众帐号的类型");
  97. article2.setDescription("");
  98. article2.setPicUrl("http://avatar.csdn.net/1/4/A/1_lyq8479.jpg");
  99. article2.setUrl("http://blog.csdn.net/lyq8479/article/details/8941577");
  100. Article article3 = new Article();
  101. article3.setTitle("第3篇\n开发模式启用及接口配置");
  102. article3.setDescription("");
  103. article3.setPicUrl("http://avatar.csdn.net/1/4/A/1_lyq8479.jpg");
  104. article3.setUrl("http://blog.csdn.net/lyq8479/article/details/8944988");
  105. articleList.add(article1);
  106. articleList.add(article2);
  107. articleList.add(article3);
  108. newsMessage.setArticleCount(articleList.size());
  109. newsMessage.setArticles(articleList);
  110. respMessage = MessageUtil.newsMessageToXml(newsMessage);
  111. }
  112. // 多图文消息---首条消息不含图片
  113. else if ("4".equals(content)) {
  114. Article article1 = new Article();
  115. article1.setTitle("微信公众帐号开发教程Java版");
  116. article1.setDescription("");
  117. // 将图片置为空
  118. article1.setPicUrl("");
  119. article1.setUrl("http://blog.csdn.net/lyq8479");
  120. Article article2 = new Article();
  121. article2.setTitle("第4篇\n消息及消息处理工具的封装");
  122. article2.setDescription("");
  123. article2.setPicUrl("http://avatar.csdn.net/1/4/A/1_lyq8479.jpg");
  124. article2.setUrl("http://blog.csdn.net/lyq8479/article/details/8949088");
  125. Article article3 = new Article();
  126. article3.setTitle("第5篇\n各种消息的接收与响应");
  127. article3.setDescription("");
  128. article3.setPicUrl("http://avatar.csdn.net/1/4/A/1_lyq8479.jpg");
  129. article3.setUrl("http://blog.csdn.net/lyq8479/article/details/8952173");
  130. Article article4 = new Article();
  131. article4.setTitle("第6篇\n文本消息的内容长度限制揭秘");
  132. article4.setDescription("");
  133. article4.setPicUrl("http://avatar.csdn.net/1/4/A/1_lyq8479.jpg");
  134. article4.setUrl("http://blog.csdn.net/lyq8479/article/details/8967824");
  135. articleList.add(article1);
  136. articleList.add(article2);
  137. articleList.add(article3);
  138. articleList.add(article4);
  139. newsMessage.setArticleCount(articleList.size());
  140. newsMessage.setArticles(articleList);
  141. respMessage = MessageUtil.newsMessageToXml(newsMessage);
  142. }
  143. // 多图文消息---最后一条消息不含图片
  144. else if ("5".equals(content)) {
  145. Article article1 = new Article();
  146. article1.setTitle("第7篇\n文本消息中换行符的使用");
  147. article1.setDescription("");
  148. article1.setPicUrl("http://0.xiaoqrobot.duapp.com/images/avatar_liufeng.jpg");
  149. article1.setUrl("http://blog.csdn.net/lyq8479/article/details/9141467");
  150. Article article2 = new Article();
  151. article2.setTitle("第8篇\n文本消息中使用网页超链接");
  152. article2.setDescription("");
  153. article2.setPicUrl("http://avatar.csdn.net/1/4/A/1_lyq8479.jpg");
  154. article2.setUrl("http://blog.csdn.net/lyq8479/article/details/9157455");
  155. Article article3 = new Article();
  156. article3.setTitle("如果觉得文章对你有所帮助,请通过博客留言或关注微信公众帐号xiaoqrobot来支持柳峰!");
  157. article3.setDescription("");
  158. // 将图片置为空
  159. article3.setPicUrl("");
  160. article3.setUrl("http://blog.csdn.net/lyq8479");
  161. articleList.add(article1);
  162. articleList.add(article2);
  163. articleList.add(article3);
  164. newsMessage.setArticleCount(articleList.size());
  165. newsMessage.setArticles(articleList);
  166. respMessage = MessageUtil.newsMessageToXml(newsMessage);
  167. }
  168. }
  169. } catch (Exception e) {
  170. e.printStackTrace();
  171. }
  172. return respMessage;
  173. }
  174. /**
  175. * emoji表情转换(hex -> utf-16)
  176. *
  177. * @param hexEmoji
  178. * @return
  179. */
  180. public static String emoji(int hexEmoji) {
  181. return String.valueOf(Character.toChars(hexEmoji));
  182. }
  183. }

如果不明白CoreService类放在什么位置,该如何使用,请查看本系列教程的第5篇。上面代码实现的功能是当用户发送数字1-5时,分别回复五种不同表现形式的图文消息给用户,如下:

a)用户发送1,回复单图文消息。参考代码68~81行,运行效果如下:

b)用户发送2,回复单图文消息---不含图片。参考代码82~96行,运行效果如下:

说明:图文消息的标题、描述是可以包含QQ表情、符号表情的。

c)用户发送3,回复多图文消息。参考代码97~123行,运行效果如下:

说明:对于多图文消息,描述不会被显示,可以在标题使用换行符,使得显示更加美观。

d)用户发送4,回复多图文消息---首条消息不含图片。参考代码124~158行,运行效果如下:

e)用户发送5,回复多图文消息---最后一条消息不含图片。参考代码159~186行,运行效果如下:

可以看出,图文消息有着丰富的内容及多样化的表现形式,希望大家能够根据各自的特点及实际使用需要,合理地运用。

最后,根据实践经验,我对图文消息做一个使用总结:

1)一定要给图文消息的Url属性赋值。不管是单图文,还是多图文,或者是不含图片的图文,都有可能会被用户点击。如果Url为空,用户点击后将会打开一个空白页面,这给用户的体验是非常差的;

2)只有单图文的描述才会显示,多图文的描述不会被显示

3)图文消息的标题、描述中可以使用QQ表情和符号表情。合理地运用表情符号,会使得消息更加生动;

4)图文消息的标题、描述中可以使用换行符。合理地使用换行符,会使得内容结构更加清晰;

5)图文消息的标题、描述中不支持超文本链接(html的<a>标签)。不只是技术上实现不了,就连逻辑上也说不通,因为一条图文消息的任何位置被点击,都将调用微信内置的浏览器打开Url,如果标题、描述里再放几个超链接,不知道点击该打开哪个页面。真搞不懂为什么有好几个同学都在问这个问题,难道设计成多图文不好吗?

6)图文消息的链接、图片链接可以使用外部域名下的资源,如本例中:柳峰的头像、博文的链接,都是指向CSDN网站的资源。在网上,甚至是微信官方交流群里,认为图文消息的Url、PicUrl不可以使用外链的大有人在,不知道这谣言从哪开始的,实践是检验真理的唯一标准!

7)使用指定大小的图片。第一条图文的图片大小建议为640*320,其他图文的图片大小建议为80*80。如果使用的图片太大,加载慢,而且耗流量;如果使用的图片太小,显示后会被拉伸,失真了很难看。

8)每条图文消息的图文建议控制在1-4条。这样在绝大多数终端上一屏能够显示完,用户扫一眼就能大概了解消息的主要内容,这样最有可能促使用户去点击并阅读。

如果觉得文章对你有所帮助,请通过留言或关注微信公众帐号xiaoqrobot来支持柳峰!

时间: 2024-10-11 13:47:47

[037] 微信公众帐号开发教程第13篇-图文消息全攻略的相关文章

[029] 微信公众帐号开发教程第5篇-各种消息的接收与响应

前一篇文章里我们已经把微信公众平台接口中消息及相关操作都进行了封装,本章节将主要介绍如何接收微信服务器发送的消息并做出响应. 明确在哪接收消息 从微信公众平台接口消息指南中可以了解到,当用户向公众帐号发消息时,微信服务器会将消息通过POST方式提交给我们在接口配置信息中填写的URL,而我们就需要在URL所指向的请求处理类CoreServlet的doPost方法中接收消息.处理消息和响应消息.   接收.处理.响应消息 下面先来看我已经写好的CoreServlet的完整代码: [java] vie

微信公众帐号开发教程第14篇-自定义菜单的创建及菜单事件响应

微信公众帐号开发教程第14篇-自定义菜单的创建及菜单事件响应 参考文章:http://blog.csdn.net/lyq8479/article/details/9841371

微信公众帐号开发教程第3篇-开发模式启用及接口配置

编辑模式与开发模式 微信公众帐号申请成功后,要想接收处理用户的请求,就必须要在"高级功能"里进行配置,点击"高级功能",将看到如下界面: 从上图中可以看到,高级功能包含两种模式:编辑模式和开发模式,并且这两种模式是互斥关系,即两种模式不能同时开启.那两种模式有什么区别呢?作为开发人员到底要开启哪一种呢?编辑模式:主要针对非编程人员及信息发布类公众帐号使用.开启该模式后,可以方便地通过界面配置"自定义菜单"和"自动回复的消息".

微信公众帐号开发教程第12篇-符号表情的发送(下)

第11篇文章给出了Unified版本的符号表情(emoji表情)代码表,并且介绍了如何在微信公众帐号开发模式下发送emoji表情,还在文章结尾出,卖了个关子:"小q机器人中使用的一些符号表情,在微信的符号表情选择栏里根本找不到,并且在上篇文章给出的符号表情代码表(Unified版)中也没有,那这些表情是如何发送的呢?"如下面两张图所示的符号表情"情侣"和"公共汽车".          本文主要介绍以下内容:1)如何在微信上使用更多的符号表情(即

[040] 微信公众帐号开发教程第16篇-应用实例之历史上的今天

内容概要 本篇文章主要讲解如何在微信公众帐号上实现“历史上的今天”功能.这个例子本身并不复杂,但希望通过对它的学习,读者能够对正则表达式有一个新的认识,能够学会运用现有的网络资源丰富自己的公众账号. 何谓历史上的今天 回顾历史的长河,历史是生活的一面镜子:以史为鉴,可以知兴衰:历史上的每一天,都是喜忧参半:可以了解历史的这一天发生的事件,借古可以鉴今,历史是不能忘记的.查看历史上每天发生的重大事情,增长知识,开拓眼界,提高人文素养. 寻找接口(数据源) 要实现查询“历史上的今天”,首先我们要找到

[038] 微信公众帐号开发教程第14篇-自定义菜单的创建及菜单事件响应

微信5.0发布 2013年8月5日,伴随着微信5.0 iPhone版的发布,公众平台也进行了重要的更新,主要包括: 1)运营主体为组织,可选择成为服务号或者订阅号: 2)服务号可以申请自定义菜单: 3)使用QQ登录的公众号,可以升级为邮箱登录: 4)使用邮箱登录的公众号,可以修改登录邮箱: 5)编辑图文消息可选填作者: 6)群发消息可以同步到腾讯微博. 其中,大家议论最多的当属前两条,就是关于帐号类型和自定义菜单的更新,我这里做几点补充说明: 1)目前公众号类型分为两种:服务号和订阅号,8月5日

[039] 微信公众帐号开发教程第15篇-自定义菜单的view类型(访问网页)

引言及内容概要 距离写上一篇文章<自定义菜单的创建及菜单事件响应>整整过了两个月的时间,那时公众平台还没有开放view类型的菜单.在不久前,微信公众平台悄悄开放了view类型的菜单,却没有在首页发布任何通知,貌似微信团队很喜欢这么干.一个偶然的机会,我留意到API文档的自定义菜单接口发生了变化,增加了对菜单view类型的说明: view(访问网页): 用户点击view类型按钮后,会直接跳转到开发者指定的url中. 于是我在第一时间更新了小q机器人(微信号:xiaoqrobot)的菜单,在一级菜

[028] 微信公众帐号开发教程第4篇-消息及消息处理工具的封装

工欲善其事必先利其器!本篇内容主要讲解如何将微信公众平台定义的消息及消息相关的操作封装成工具类,方面后期的使用.这里需要明确的是消息其实是由用户发给你的公众帐号的,消息先被微信平台接收到,然后微信平台会将该消息转给你在开发模式接口配置中指定的URL地址. 微信公众平台消息接口 要接收微信平台发送的消息,我们需要先熟悉微信公众平台API中消息接口部分,点此进入,点击后将进入到消息接口指南部分,如下图所示: 在上图左侧可以看到微信公众平台目前开放的接口有三种:消息接口.通用接口和自定义菜单接口.通用

微信公众帐号开发之一(java)

闲来没事,就记录一下微信公众平台的开发吧~ 其实微信公众平台开发没有想象中的那么困难,因为注册了微信公众平台帐号登录之后在开发者模式里有详细的文档,个人感觉介绍还是比较详细的. 微信公众平台订阅号和服务号的区别网上已经有N多的介绍了,这里就不再说了.从微信公众平台登录之后开始说起吧. 对于开发者来说,开发者模式是最好的选择了.因为在里面可以更自由更灵活的选择消息回复方式.编辑模式只能对特定的字符进行相应的消息回复.而且在开发者模式下可以灵活的创建自定义菜单(前提是你的帐号有提供相应的接口,需要微