全局说明

* 详细说明请参考前两篇文章。

本文说明

*本文分为五部分:* 工具类AccessTokenUtils的封装* 自定义菜单和个性化菜单文档的阅读解析* 菜单JSON的分析以及构建对应bean* 自定义菜单的实现* 个性化菜单的实现
*  微信自定义菜单所有类型菜单都给出演示
* 本文结束会给出包括本文前四篇文章的所有演示源码

工具类AccessTokenUtils的封装

  • 在上文中关于AccessToken的获取和定时保存已经详细介绍过,此处直接给出处理过之后封装的AccessTokenUtils,实现原理以及文档阅读不再给出。
  • AccessTokenUtils.java

    package com.gist.utils;import java.io.File;
    import java.io.FileInputStream;
    import java.io.FileOutputStream;
    import java.io.IOException;
    import java.io.InputStreamReader;
    import java.net.URL;import javax.net.ssl.HttpsURLConnection;import com.gist.bean.Access_token;
    import com.google.gson.Gson;/*** @author 高远</n> 邮箱:wgyscsf@163.com</n> 博客 http://blog.csdn.net/wgyscsf</n>*         编写时期 2016-4-7 下午5:44:33*/
    public class AccessTokenUtils {private static final long MAX_TIME = 7200 * 1000;// 微信允许最长Access_token有效时间(ms)private static final String TAG = "WeixinApiTest";// TAGprivate static final String APPID = "wx889b020b3666b0b8";// APPIDprivate static final String SECERT = "6da7676bf394f0a9f15fbf06027856bb";// 秘钥/** 该方法实现获取Access_token、保存并且只保存2小时Access_token。如果超过两个小时重新获取;如果没有超过两个小时,直接获取。该方法依赖* :public static String getAccessToken();* * 思路:将获取到的Access_token和当前时间存储到file里,* 取出时判断当前时间和存储里面的记录的时间的时间差,如果大于MAX_TIME,重新获取,并且将获取到的存储到file替换原来的内容* ,如果小于MAX_TIME,直接获取。*/// 为了调用不抛异常,这里全部捕捉异常,代码有点长public static String getSavedAccess_token() {Gson gson = new Gson();// 第三方jar,处理json和bean的转换String mAccess_token = null;// 需要获取的Access_token;FileOutputStream fos = null;// 输出流FileInputStream fis = null;// 输入流File file = new File("temp_access_token.temp");// Access_token保存的位置try {// 如果文件不存在,创建if (!file.exists()) {file.createNewFile();}} catch (Exception e1) {e1.printStackTrace();}// 如果文件大小等于0,说明第一次使用,存入Access_tokenif (file.length() == 0) {try {mAccess_token = getAccessToken();// 获取AccessTokenAccess_token at = new Access_token();at.setAccess_token(mAccess_token);at.setExpires_in(System.currentTimeMillis() + "");// 设置存入时间String json = gson.toJson(at);fos = new FileOutputStream(file, false);// 不允许追加fos.write((json).getBytes());// 将AccessToken和当前时间存入文件fos.close();return mAccess_token;} catch (Exception e) {e.printStackTrace();}} else {// 读取文件内容byte[] b = new byte[2048];int len = 0;try {fis = new FileInputStream(file);len = fis.read(b);} catch (IOException e1) {// TODO Auto-generated catch blocke1.printStackTrace();}String mJsonAccess_token = new String(b, 0, len);// 读取到的文件内容Access_token access_token = gson.fromJson(mJsonAccess_token,new Access_token().getClass());if (access_token.getExpires_in() != null) {long saveTime = Long.parseLong(access_token.getExpires_in());long nowTime = System.currentTimeMillis();long remianTime = nowTime - saveTime;// System.out.println(TAG + "时间差:" + remianTime + "ms");if (remianTime < MAX_TIME) {Access_token at = gson.fromJson(mJsonAccess_token,new Access_token().getClass());mAccess_token = at.getAccess_token();return mAccess_token;} else {mAccess_token = getAccessToken();Access_token at = new Access_token();at.setAccess_token(mAccess_token);at.setExpires_in(System.currentTimeMillis() + "");String json = gson.toJson(at);try {fos = new FileOutputStream(file, false);// 不允许追加fos.write((json).getBytes());fos.close();} catch (IOException e) {// TODO Auto-generated catch blocke.printStackTrace();}return mAccess_token;}} else {return null;}}return mAccess_token;}/** 获取微信服务器AccessToken。该部分和getAccess_token() 一致,不再加注释*/public static String getAccessToken() {String urlString = "https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid="+ APPID + "&secret=" + SECERT;String reslut = null;try {URL reqURL = new URL(urlString);HttpsURLConnection httpsConn = (HttpsURLConnection) reqURL.openConnection();InputStreamReader isr = new InputStreamReader(httpsConn.getInputStream());char[] chars = new char[1024];reslut = "";int len;while ((len = isr.read(chars)) != -1) {reslut += new String(chars, 0, len);}isr.close();} catch (IOException e) {e.printStackTrace();}Gson gson = new Gson();Access_token access_token = gson.fromJson(reslut,new Access_token().getClass());if (access_token.getAccess_token() != null) {return access_token.getAccess_token();} else {return null;}}
    }
    

自定义菜单和个性化菜单文档的阅读解析

  • 自定义菜单

    • 自定义菜单创建接口
    • 自定义菜单查询接口
    • 自定义菜单删除接口
    • 自定义菜单事件推送
    • 个性化菜单接口
    • 获取公众号的菜单配置
  • 文档地址:http://mp.weixin.qq.com/wiki/10/0234e39a2025342c17a7d23595c6b40a.html
  • 官网文档给出这样解释:

    * 自定义菜单接口可实现多种类型按钮,如下:1、click:点击事件...;2、view:跳转事件...;3、...(关于自定义菜单)
    * 接口调用请求说明 http请求方式:POST(请使用https协议) https://api.weixin.qq.com/cgi-bin/menu/create?access_token=ACCESS_TOKEN(关于自定义菜单)
    * click和view的请求示例 {"button":[...]}  (关于自定义菜单)
    * 参数说明...(关于自定义菜单)
    * 创建个性化菜单http请求方式:POST(请使用https协议)https://api.weixin.qq.com/cgi-bin/menu/addconditional?access_token=ACCESS_TOKEN(关于个性化菜单)
    * 请求示例: {"button":[...],"matchrule":{...}}(关于个性化菜单)
    * 参数说明...(关于个性化菜单)
    * 开发者可以通过以下条件来设置用户看到的菜单(关于个性化菜单):1、用户分组(开发者的业务需求可以借助用户分组来完成)2、性别3、手机操作系统4、地区(用户在微信客户端设置的地区)5、语言(用户在微信客户端设置的语言)
    
  • 理解:

    • 又是熟悉的POST请求,但是,关于调用貌似说的含糊其辞,不太明白。只是知道我们需要使用“?access_token=ACCESS_TOKEN”这个参数,这个参数我们在上篇文章已经获取到了。假如我们将微信文档给的那个请求地址中“ACCESS_TOKEN”换成我们获取到的自己的ACCESS_TOKEN,访问该网址,会看到“{“errcode”:44002,”errmsg”:”empty post data hint: [Gdveda0984vr23]”}”。大概意思是,空的post请求数据。所以,我们要通过POST请求的形式传递参数给微信服务器,在文档下面还给出了参数的格式:{“button”:[…]},所以,我们要按照该格式给微信服务器进行传递参数。
    • 关于参数说明,我们可以看到在自定义菜单创建中有七个参数。在个性化菜单接口中除去这七个参数之外,另外多个八个参数。简单查看此部分文档,我们可以了解到这个八个参数是为了个性化菜单做匹配筛选用的。
    • 现在,我们需要按照微信文档的要求构造json通过post的请求向微信服务器发送这一串json数据,json里面就包括我们创建的各种类型的按钮事件。

菜单JSON的分析以及构建对应bean

  • 自定义菜单json分析(不包括个性化菜单)。下面这段代码是微信文档给的示例。

    click和view的请求示例{"button":[{  "type":"click","name":"今日歌曲","key":"V1001_TODAY_MUSIC"},{"name":"菜单","sub_button":[{    "type":"view","name":"搜索","url":"http://www.soso.com/"},{"type":"view","name":"视频","url":"http://v.qq.com/"},{"type":"click","name":"赞一下我们","key":"V1001_GOOD"}]}]}
    
  • 经过分析我们可以看到这串json数据分为三层:“”button”:[{…},{…}]”、“[{…},{{“name”:菜单,”sub_button”:[{},{}]}]”、“{“type”:”view”,”name:”:”视频”,”url”:”…”},{},{}”,可能看起来比较晕。
  • 但是,如果我们能够联想起来现实中看到的微信菜单,就会好理解一点:一级:菜单(一个菜单),下包括一到三个父按钮;二级:父按钮(1~3个父按钮),下包括一到五个子按钮;三级:子按钮(1~5个子按钮)。
  • 现在,我们可以看到json和我们理解的“菜单”可以一一对应起来了。现在重点是如何确认每一级的“级名”,在java中也就是对应的javabean对象。
  • 同时,因为一级菜单下会有多个父按钮,所以是一个List<父菜单>的形式。父按钮下可能有多个子菜单,也是一个 List<子菜单>;但是,父按钮也有可能也是一个单独的可以响应的按钮。是一个单独的父按钮对象。子按钮就是一个单独的子按钮对象。
  • 查看关于自定义菜单的参数说明,我们可以看到按钮分为一级按钮(“button”)和二级按钮(“sub_button”)。还有一些公用的数据类型,例如:菜单响应类型(“type”)、菜单标题(“name”)、click类型的参数(“key”)、view类型的参数(“url”)、media_id类型和view_limited类型的参数(“media_id”)。
  • 数据抽象(没有写setter,getter):

    //按钮基类
    public class BaseButton {private String type;private String name;private String key;private String url;private String media_id;
    }
    //子按钮
    public class SonButton extends BaseButton {private String sub_button;
    }
    //父按钮
    public class FatherButton extends BaseButton {
    private String button;//可能直接一个父按钮做响应
    @SerializedName("sub_button")//为了保证Gson解析后子按钮的名字是“sub_button”,具体用法请搜索
    private List<SonButton> sonButtons;//可能有多个子按钮
    }public class Menu {
    @SerializedName("button")
    private List<FatherButton> fatherButtons;
    }
    
  • 以上是完整的自定义菜单的分析以及对应javabean的构建。

  • 对于个性化菜单,如果查看该部分的文档,会发现和自定义菜单大致相同,只是多个一个“配置”的json,格式是这样的:{“button”:[…],”matchrule”:{…}}。
  • 我们发现,“匹配”这段json和“button”是同级的,分析和实现和上面基本等同,直接给出实现的javabean。

    //匹配的json对应的json
    public class MatchRule {
    private String group_id;
    private String sex;
    private String client_platform_type;
    private String country;
    private String province;
    private String city;
    private String language;
    }//修改Menu.java
    public class Menu {
    @SerializedName("button")
    private List<FatherButton> fatherButtons;
    private MatchRule matchrule;
    }
    

自定义菜单的实现

  • 任务,我们实现所有微信按钮响应类型:

     任务(注释:“m-0”表示父按钮;“m-n”表示第m个父按钮,第n个子按钮(m,n≠0)):1-0:名字:click,响应点击事件:点击推事件。2-0:名字:父按钮2。2-1:名字:view,响应事件:跳转网页;2-2:名字:scancode_push,响应事件:扫码推事件;2-3:名字:scancode_waitmsg,响应事件:扫码推事件且弹出“消息接收中”提示框;2-4:名字:pic_sysphoto,响应事件:弹出系统拍照发图。2-5:名字:pic_photo_or_album,响应事件:弹出拍照或者相册发图。3-0:名字:父按钮3。3-1:名字:pic_weixin,响应事件:弹出微信相册发图器;3-2:名字:location_select,响应事件:弹出地理位置选择器;3-3:名字:media_id,响应事件:下发消息(除文本消息);3-4:名字:view_limited,响应事件:跳转图文消息url。
    

*实现源码(引用的AccessTokenUtils.java在第一部分:工具类AccessTokenUtils的封装)

    /** 创建自定义菜单。*/@Testpublic void createCommMenu() {String ACCESS_TOKEN = AccessTokenUtils.getAccessToken();// 获取AccessToken,AccessTokenUtils是封装好的类// 拼接api要求的httpsurl链接String urlString = "https://api.weixin.qq.com/cgi-bin/menu/create?access_token="+ ACCESS_TOKEN;try {// 创建一个urlURL reqURL = new URL(urlString);// 拿取链接HttpsURLConnection httpsConn = (HttpsURLConnection) reqURL.openConnection();httpsConn.setDoOutput(true);// 取得该连接的输出流,以读取响应内容OutputStreamWriter osr = new OutputStreamWriter(httpsConn.getOutputStream());osr.write(getMenuJson());// 使用本类外部方法getMenuJson()osr.close();// 返回结果InputStreamReader isr = new InputStreamReader(httpsConn.getInputStream());// 读取服务器的响应内容并显示char[] chars = new char[1024];String reslut = "";int len;while ((len = isr.read(chars)) != -1) {reslut += new String(chars, 0, len);}System.out.println("返回结果:" + reslut);isr.close();} catch (IOException e) {e.printStackTrace();}}public String getMenuJson() {Gson gson = new Gson();// json处理工具Menu menu = new Menu();// 菜单类List<FatherButton> fatherButtons = new ArrayList<FatherButton>();// 菜单中的父按钮集合// -----------// 父按钮1FatherButton fb1 = new FatherButton();fb1.setName("click");fb1.setType("click");fb1.setKey("10");// -------------// 父按钮2FatherButton fb2 = new FatherButton();fb2.setName("父按钮2");List<SonButton> sonButtons2 = new ArrayList<SonButton>();// 子按钮的集合// 子按钮2-1SonButton sb21 = new SonButton();sb21.setName("view");sb21.setUrl("http://www.baidu.com");sb21.setType("view");// 子按钮2-2SonButton sb22 = new SonButton();sb22.setName("scancode_push");sb22.setType("scancode_push");sb22.setKey("22");// 子按钮2-3SonButton sb23 = new SonButton();sb23.setName("scancode_waitmsg");sb23.setType("scancode_waitmsg");sb23.setKey("23");// 子按钮2-4SonButton sb24 = new SonButton();sb24.setName("pic_sysphoto");sb24.setType("pic_sysphoto");sb24.setKey("24");// 子按钮2-5SonButton sb25 = new SonButton();sb25.setName("pic_photo_or_album");sb25.setType("pic_photo_or_album");sb25.setKey("25");// 添加子按钮到子按钮集合sonButtons2.add(sb21);sonButtons2.add(sb22);sonButtons2.add(sb23);sonButtons2.add(sb24);sonButtons2.add(sb25);// 将子按钮放到2-0父按钮集合fb2.setSonButtons(sonButtons2);// ------------------// 父按钮3FatherButton fb3 = new FatherButton();fb3.setName("父按钮3");List<SonButton> sonButtons3 = new ArrayList<SonButton>();// 子按钮3-1SonButton sb31 = new SonButton();sb31.setName("pic_weixin");sb31.setType("pic_weixin");sb31.setKey("31");// 子按钮3-2SonButton sb32 = new SonButton();sb32.setName("locatselect");sb32.setType("location_select");sb32.setKey("32");// // 子按钮3-3-->测试不了,因为要media_id。这需要调用素材id.// SonButton sb33 = new SonButton();// sb33.setName("media_id");// sb33.setType("media_id");// sb33.setMedia_id("???");// // 子按钮3-4-->测试不了,因为要media_id。这需要调用素材id.// SonButton sb34 = new SonButton();// sb34.setName("view_limited");// sb34.setType("view_limited");// sb34.setMedia_id("???");// 添加子按钮到子按钮队列sonButtons3.add(sb31);sonButtons3.add(sb32);// sonButtons3.add(sb33);// sonButtons3.add(sb34);// 将子按钮放到3-0父按钮队列fb3.setSonButtons(sonButtons3);// ---------------------// 将父按钮加入到父按钮集合fatherButtons.add(fb1);fatherButtons.add(fb2);fatherButtons.add(fb3);// 将父按钮队列加入到菜单栏menu.setFatherButtons(fatherButtons);String json = gson.toJson(menu);System.out.println(json);// 测试输出return json;}

个性化菜单的实现

  • 任务:根据性别展示不同的按钮显示(可以根据性别、地区、分组手机操作系统等)
  • 修改代码一,因为是不同的微信后台实现,所以接口也不一样,不过还是POST请求,代码不用改,只要替换原来urlString即可。

    // 拼接api要求的httpsurl链接
    String urlString = "https://api.weixin.qq.com/cgi-bin/menu/addconditional?access_token="+ ACCESS_TOKEN;
    
  • 修改代码二,只要创建一个MatchRule,设置匹配规则,然后将matchrule加入到menu便可以完成匹配规则。

    // -----
    // 从此处开始设置个性菜单
    MatchRule matchrule = new MatchRule();
    matchrule.setSex("2");// 男生
    menu.setMatchrule(matchrule);
    // ----
    

测试号二维码,大家可以扫描查看效果(只保留最新效果,因为测试,可能效果会变化):


这是平时写的一些技术文章的微信公众号,欢迎关注:


源码:http://download.csdn.net/detail/wgyscsf/9485817

java微信开发API解析(四)-自定义菜单以及个性化菜单实现相关推荐

  1. java微信开发API解析(二)-获取消息和回复消息

    java微信开发API解析(二)-获取消息和回复消息 说明 * 本演示样例依据微信开发文档:http://mp.weixin.qq.com/wiki/home/index.html最新版(4/3/20 ...

  2. 微信个性化菜单 java_java微信开发API第四步 微信自定义个性化菜单实现

    微信如何实现自定义个性化菜单,下面为大家介绍 一.全局说明 详细说明请参考前两篇文章. 二.本文说明 本文分为五部分: * 工具类AccessTokenUtils的封装 * 自定义菜单和个性化菜单文档 ...

  3. java 微信 菜单_java微信开发API第四步 微信自定义个性化菜单实现

    微信如何实现自定义个性化菜单,下面为大家介绍 一.全局说明 详细说明请参考前两篇文章. 二.本文说明 本文分为五部分: * 工具类AccessTokenUtils的封装 * 自定义菜单和个性化菜单文档 ...

  4. java获取微信的通讯录,java微信开发API第三步 微信获取以及保存接口调用凭证

    微信如何获取以及保存接口调用凭证,下面就为大家进行介绍 一.说明 *详细说明请参考前两篇文章. *本文分为三部分: 接口调用凭证access_token的作用以及解释 如何获取接口调用凭证access ...

  5. 微信小程序生态13-微信公众号自定义菜单、个性化菜单配置

    文章导航 微信小程序生态1-初识小程序 微信小程序生态2-创建一个微信小程序 微信小程序生态3-微信小程序登录流程设计 微信小程序生态4-扫普通二维码进入小程序.打开短链接进入小程序 微信小程序生态5 ...

  6. Java微信开发_00_资源汇总贴

    1.微信公众平台技术文档(https://mp.weixin.qq.com/wiki?t=resource/res_main&id=mp1445241432) 2.微信企业号开发接口文档(ht ...

  7. Java微信开发入门

    前言: 1.先简单介绍一下微信公众号的开发者模式和编辑模式,玩过微信公众号的同志都知道,在微信公众号后台功能栏里面可以编辑消息自动回复,关键字回复,自定义菜单等功能,这些都是在视图页面下编辑的,叫编辑 ...

  8. java微信开发-消息接收和自动回复

    0.前提条件 1.需要一个公众平台帐号(测试帐号也可以) 2.平台需要被人关注 1.导入jar commons-beanutils-1.8.0.jar commons-collections-3.1. ...

  9. 限时团购,6.9折:《微信开发深度解析:公众号、小程序高效开发秘籍》推荐序

    全书由目 Senparc.Weixin SDK 作者苏震巍历时 2 年完成,涵盖了开发微信公众号及小程序需要用的的各项后端开发技能.技巧.避坑提示,以及 Senparc.Weixin SDK 微信公众 ...

最新文章

  1. Genome Biology | 基于RNA-seq的孟德尔疾病变异分析
  2. 人人都是产品经理读书笔记(四)
  3. 如何让网站被百度谷歌快速收录
  4. AnimalTFDB 3.0:动物转录因子注释和预测的综合资源库
  5. prometheus连续查询_Grafana + Prometheus快速搭建监控平台
  6. 逆天的IE7中,诡异的横向滚动条
  7. java 获取oracle表结构_获取Oracle中所有表的列表?
  8. IoC、Spring 环境搭建、Spring 创建对象的三种方式、DI
  9. Windroy Lets Android run on Windows systems-- 国外androids 虚拟系统分享
  10. 下载最新adb(platform-tools)
  11. JQuery文档信息
  12. AI前言技术,一文带你深入了解语音识别的前世今生
  13. 计算机二级ms office有题库吗,计算机二级office题库-吃惊!二级MS Office高级应用通过率仅有21.07%!...
  14. 值得收藏的资源网站大全
  15. 以太坊开发入门,完整入门
  16. 必须正视TD-SCDMA可能存在的重大技术错误(ZT)
  17. java罗马数字_Java实现的求解经典罗马数字和阿拉伯数字相互转换问题示例
  18. 七天学习Go语言-从0-0.05
  19. 一款非常方便简单的Mac个人理财软件:Moneydance
  20. WordPress关注微信公众号回复可见和阅读更多的方法

热门文章

  1. 相约鹏城|11.15-19高交会X慕尼黑华南电子展X华南电路板博览会X半导体显示展齐聚深圳,共襄盛举~
  2. 如何把类数组对象转为数组对象
  3. 末日要塞找不到服务器,末日要塞攻略 末日要塞流程攻略
  4. Java - 日志(进阶篇)
  5. word2vec python实现_word2vec及其python实现
  6. 招聘旺季,外包公司招聘任务艰巨,程序员为什么不喜欢去外包公司工作?
  7. 关于setAttribute覆盖的问题
  8. 2019马甲包提审近况
  9. 建设银行信用卡电子账单读取
  10. 如何在Visio中画精准的圆弧