程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> JAVA編程 >> 關於JAVA >> java微信開辟API第四步 微信自界說特性化菜單完成

java微信開辟API第四步 微信自界說特性化菜單完成

編輯:關於JAVA

java微信開辟API第四步 微信自界說特性化菜單完成。本站提示廣大學習愛好者:(java微信開辟API第四步 微信自界說特性化菜單完成)文章只能為提供參考,不一定能成為您想要的結果。以下是java微信開辟API第四步 微信自界說特性化菜單完成正文


微信若何完成自界說特性化菜單,上面為年夜家引見

1、全局解釋
具體解釋請參考前兩篇文章。

2、本文解釋
本文分為五部門:
    * 對象類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.谷歌.gson.Gson;

/**
 * @author 高遠</n> 郵箱:[email protected]</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";// TAG
 private static final String APPID = "wx889b020b3666b0b8";// APPID
 private 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_token
  if (file.length() == 0) {
   try {
    mAccess_token = getAccessToken();// 獲得AccessToken
    Access_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 block
    e1.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 block
      e.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的封裝)

 /*
  * 創立自界說菜單。
  */
 @Test
 public 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 {
   // 創立一個url
   URL 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>();// 菜單中的父按鈕聚集
  // -----------
  // 父按鈕1
  FatherButton fb1 = new FatherButton();
  fb1.setName("click");
  fb1.setType("click");
  fb1.setKey("10");
  // -------------
  // 父按鈕2
  FatherButton fb2 = new FatherButton();
  fb2.setName("父按鈕2");
  List<SonButton> sonButtons2 = new ArrayList<SonButton>();// 子按鈕的聚集

  // 子按鈕2-1
  SonButton sb21 = new SonButton();
  sb21.setName("view");
  sb21.setUrl("http://www.百度.com");
  sb21.setType("view");
  // 子按鈕2-2
  SonButton sb22 = new SonButton();
  sb22.setName("scancode_push");
  sb22.setType("scancode_push");
  sb22.setKey("22");
  // 子按鈕2-3
  SonButton sb23 = new SonButton();
  sb23.setName("scancode_waitmsg");
  sb23.setType("scancode_waitmsg");
  sb23.setKey("23");
  // 子按鈕2-4
  SonButton sb24 = new SonButton();
  sb24.setName("pic_sysphoto");
  sb24.setType("pic_sysphoto");
  sb24.setKey("24");
  // 子按鈕2-5
  SonButton 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);

  // ------------------
  // 父按鈕3
  FatherButton fb3 = new FatherButton();
  fb3.setName("父按鈕3");
  List<SonButton> sonButtons3 = new ArrayList<SonButton>();

  // 子按鈕3-1
  SonButton sb31 = new SonButton();
  sb31.setName("pic_weixin");
  sb31.setType("pic_weixin");
  sb31.setKey("31");
  // 子按鈕3-2
  SonButton 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://xiazai.jb51.net/201606/yuanma/WeixinApi(jb51.net).rar

以上就是本文的全體內容,願望對年夜家的進修有所贊助,也願望年夜家多多支撐。

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved