鍍金池/ 教程/ Java/ Java 微信公眾平臺開發(fā)(六)--微信開發(fā)中的 token 獲取
Java 微信公眾平臺開發(fā)(八)--多媒體消息回復
Java 微信公眾平臺開發(fā)(四)--回復消息的分類及實體的創(chuàng)建
Mybatis 工具 Generator
Java 微信公眾平臺開發(fā)(十一)--開發(fā)中微信公眾平臺/開放平臺/商戶平臺的關聯(lián)
微信開發(fā)準備(二)--springmvc+mybatis 項目結構的搭建
 Java 微信公眾平臺開發(fā)(十二)--微信用戶信息的獲取
Java 微信公眾平臺開發(fā)(十五)--微信 JSSDK 的使用
微信開發(fā)準備(三)--框架以及工具的基本使用
Java 微信公眾平臺開發(fā)(十三)--微信 JSSDK 中 Config 配置
Java 微信公眾平臺開發(fā)(一)--接入微信公眾平臺
Java 微信公眾平臺開發(fā)(十四)【番外篇】--微信 web 開發(fā)者工具使用
Java 微信公眾平臺開發(fā)【番外篇】(七)--公眾平臺測試帳號的申請
微信開發(fā)準備(一)--Maven 倉庫管理新建 WEB 項目
Java 微信公眾平臺開發(fā)(三)--接收消息的分類及實體的創(chuàng)建
Java 微信公眾平臺開發(fā)(九)--關鍵字回復以及客服接口實現(xiàn)(該公眾號暫時無法提供服務解決方案)
微信開發(fā)準備(四)--nat123 內網地址公網映射實現(xiàn)
Java 微信公眾平臺開發(fā)(五)--文本及圖文消息回復的實現(xiàn)
Java 微信公眾平臺開發(fā)(十)--微信自定義菜單的創(chuàng)建實現(xiàn)
Java 微信公眾平臺開發(fā)(六)--微信開發(fā)中的 token 獲取
Java 微信公眾平臺開發(fā)(二)--微信服務器 post 消息體的接收

Java 微信公眾平臺開發(fā)(六)--微信開發(fā)中的 token 獲取

(一)token 的介紹

引用:access_token 是公眾號的全局唯一票據(jù),公眾號調用各接口時都需使用 access_token。開發(fā)者需要進行妥善保存。access_token 的存儲至少要保留 512 個字符空間。access_token 的有效期目前為 2 個小時,需定時刷新,重復獲取將導致上次獲取的 access_token 失效!

(二)token 的獲取參考文檔

獲取的流程我們完全可以參考微信官方文檔:http://mp.weixin.qq.com/wiki/14/9f9c82c1af308e3b14ba9b973f99a8ba.html 如圖:

http://wiki.jikexueyuan.com/project/java-wechat/images/25.png" alt="" />

(三)token 獲取流程分析

從公眾平臺獲取賬號的 AppID 和 AppSecret; token 獲取并解析存儲執(zhí)行體; 采用任務調度每隔兩小時執(zhí)行一次 token 獲取執(zhí)行體;

(四)token 的獲取流程的具體實現(xiàn)

1.獲取 appid 和 appsecret

在微信公眾平臺【開發(fā)】——>【基本配置】中可以查看到我們需要的兩個參數(shù):

http://wiki.jikexueyuan.com/project/java-wechat/images/26.png" alt="" />

這里我們將他們定義到我們的配置文件【wechat.properties】中,大致代碼為:

#獲取到的 appid
appid=wx7e32765bc24XXXX 
#獲取到的 AppSecret
AppSecret=d58051564fe9d86093f9XXXXX

2.token 獲取并解析存儲執(zhí)行體的代碼編寫

由于在這里我們需要通過 http 的 get 請求向微信服務器獲取時效性為 7200 秒的 token,所以我在這里寫了一個 http 請求的工具類,以方便我們的使用,如下:

package com.cuiyongzhi.wechat.util;

import java.io.BufferedInputStream;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStreamWriter;
import java.net.MalformedURLException;
import java.net.URI;
import java.net.URL;
import java.net.URLConnection;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.zip.GZIPInputStream;

import org.apache.http.HttpResponse;
import org.apache.http.NameValuePair;
import org.apache.http.client.ClientProtocolException;
import org.apache.http.client.HttpClient;
import org.apache.http.client.entity.UrlEncodedFormEntity;
import org.apache.http.client.methods.HttpGet;
import org.apache.http.client.methods.HttpPost;
import org.apache.http.entity.StringEntity;
import org.apache.http.impl.client.DefaultHttpClient;
import org.apache.http.message.BasicNameValuePair;
import org.apache.http.protocol.HTTP;
import org.apache.http.util.EntityUtils;

/**
 * ClassName: HttpUtils
 * 
 * @Description: http 請求工具類
 * @author dapengniao
 * @date 2016 年 3 月 10 日 下午 3:57:14
 */
@SuppressWarnings("deprecation")
public class HttpUtils {

    /**
     * @Description: http get 請求共用方法
     * @param @param reqUrl
     * @param @param params
     * @param @return
     * @param @throws Exception
     * @author dapengniao
     * @date 2016 年 3 月 10 日 下午 3:57:39
     */
    @SuppressWarnings("resource")
    public static String sendGet(String reqUrl, Map<String, String> params)
            throws Exception {
        InputStream inputStream = null;
        HttpGet request = new HttpGet();
        try {
            String url = buildUrl(reqUrl, params);
            HttpClient client = new DefaultHttpClient();

            request.setHeader("Accept-Encoding", "gzip");
            request.setURI(new URI(url));

            HttpResponse response = client.execute(request);

            inputStream = response.getEntity().getContent();
            String result = getJsonStringFromGZIP(inputStream);
            return result;
        } finally {
            if (inputStream != null) {
                inputStream.close();
            }
            request.releaseConnection();
        }

    }

    /**
     * @Description: http post 請求共用方法
     * @param @param reqUrl
     * @param @param params
     * @param @return
     * @param @throws Exception
     * @author dapengniao
     * @date 2016 年 3 月 10 日 下午 3:57:53
     */
    @SuppressWarnings("resource")
    public static String sendPost(String reqUrl, Map<String, String> params)
            throws Exception {
        try {
            Set<String> set = params.keySet();
            List<NameValuePair> list = new ArrayList<NameValuePair>();
            for (String key : set) {
                list.add(new BasicNameValuePair(key, params.get(key)));
            }
            if (list.size() > 0) {
                try {
                    HttpClient client = new DefaultHttpClient();
                    HttpPost request = new HttpPost(reqUrl);

                    request.setHeader("Accept-Encoding", "gzip");
                    request.setEntity(new UrlEncodedFormEntity(list, HTTP.UTF_8));

                    HttpResponse response = client.execute(request);

                    InputStream inputStream = response.getEntity().getContent();
                    try {
                        String result = getJsonStringFromGZIP(inputStream);

                        return result;
                    } finally {
                        inputStream.close();
                    }
                } catch (Exception ex) {
                    ex.printStackTrace();
                    throw new Exception("網絡連接失敗,請連接網絡后再試");
                }
            } else {
                throw new Exception("參數(shù)不全,請稍后重試");
            }
        } catch (Exception ex) {
            ex.printStackTrace();
            throw new Exception("發(fā)送未知異常");
        }
    }

    /**
     * @Description: http post 請求 json 數(shù)據(jù)
     * @param @param urls
     * @param @param params
     * @param @return
     * @param @throws ClientProtocolException
     * @param @throws IOException
     * @author dapengniao
     * @date 2016 年 3 月 10 日 下午 3:58:15
     */
    public static String sendPostBuffer(String urls, String params)
            throws ClientProtocolException, IOException {
        HttpPost request = new HttpPost(urls);

        StringEntity se = new StringEntity(params, HTTP.UTF_8);
        request.setEntity(se);
        // 發(fā)送請求
        @SuppressWarnings("resource")
        HttpResponse httpResponse = new DefaultHttpClient().execute(request);
        // 得到應答的字符串,這也是一個 JSON 格式保存的數(shù)據(jù)
        String retSrc = EntityUtils.toString(httpResponse.getEntity());
        request.releaseConnection();
        return retSrc;

    }

    /**
     * @Description: http 請求發(fā)送 xml 內容
     * @param @param urlStr
     * @param @param xmlInfo
     * @param @return
     * @author dapengniao
     * @date 2016 年 3 月 10 日 下午 3:58:32
     */
    public static String sendXmlPost(String urlStr, String xmlInfo) {
        // xmlInfo xml 具體字符串

        try {
            URL url = new URL(urlStr);
            URLConnection con = url.openConnection();
            con.setDoOutput(true);
            con.setRequestProperty("Pragma:", "no-cache");
            con.setRequestProperty("Cache-Control", "no-cache");
            con.setRequestProperty("Content-Type", "text/xml");
            OutputStreamWriter out = new OutputStreamWriter(
                    con.getOutputStream());
            out.write(new String(xmlInfo.getBytes("utf-8")));
            out.flush();
            out.close();
            BufferedReader br = new BufferedReader(new InputStreamReader(
                    con.getInputStream()));
            String lines = "";
            for (String line = br.readLine(); line != null; line = br
                    .readLine()) {
                lines = lines + line;
            }
            return lines; // 返回請求結果
        } catch (MalformedURLException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        }
        return "fail";
    }

    private static String getJsonStringFromGZIP(InputStream is) {
        String jsonString = null;
        try {
            BufferedInputStream bis = new BufferedInputStream(is);
            bis.mark(2);
            // 取前兩個字節(jié)
            byte[] header = new byte[2];
            int result = bis.read(header);
            // reset 輸入流到開始位置
            bis.reset();
            // 判斷是否是 GZIP 格式
            int headerData = getShort(header);
            // Gzip 流 的前兩個字節(jié)是 0x1f8b
            if (result != -1 && headerData == 0x1f8b) {
                // LogUtil.i("HttpTask", " use GZIPInputStream  ");
                is = new GZIPInputStream(bis);
            } else {
                // LogUtil.d("HttpTask", " not use GZIPInputStream");
                is = bis;
            }
            InputStreamReader reader = new InputStreamReader(is, "utf-8");
            char[] data = new char[100];
            int readSize;
            StringBuffer sb = new StringBuffer();
            while ((readSize = reader.read(data)) > 0) {
                sb.append(data, 0, readSize);
            }
            jsonString = sb.toString();
            bis.close();
            reader.close();
        } catch (Exception e) {
            e.printStackTrace();
        }

        return jsonString;
    }

    private static int getShort(byte[] data) {
        return (data[0] << 8) | data[1] & 0xFF;
    }

    /**
     * 構建 get 方式的 url
     * 
     * @param reqUrl
     *            基礎的 url 地址
     * @param params
     *            查詢參數(shù)
     * @return 構建好的 url
     */
    public static String buildUrl(String reqUrl, Map<String, String> params) {
        StringBuilder query = new StringBuilder();
        Set<String> set = params.keySet();
        for (String key : set) {
            query.append(String.format("%s=%s&", key, params.get(key)));
        }
        return reqUrl + "?" + query.toString();
    }

}

我們在做 http 請求的時候需要目標服務器的 url,這里在項目中為了方便對 url 的管理我們在資源目錄下建立了 interface_url.properties 用于存放目標 url,這里我們將請求 token 的 url 存入:

#獲取 token 的 url
tokenUrl=https://api.weixin.qq.com/cgi-bin/token

我們需要將我們配置的配置文件在項目初始化后能得到啟動,所以我在這里加入一個項目初始化的代碼實現(xiàn),用于項目啟動初始化 interface_url.properties 和 wechat.properties 中的配置:

package com.cuiyongzhi.web.start;

import javax.servlet.ServletConfig;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;

/**
 * ClassName: InterfaceUrlIntiServlet
 * @Description: 項目啟動初始化 servlet
 * @author dapengniao
 * @date 2016 年 3 月 10 日 下午 4:08:43
 */
public class InterfaceUrlIntiServlet extends HttpServlet {

    private static final long serialVersionUID = 1L;

    @Override
    public void init(ServletConfig config) throws ServletException {
        InterfaceUrlInti.init();
    }

}

初始化的具體實現(xiàn),將初始化過后的方法都存入到 GlobalConstants 中方便項目中隨意調用,如下:

package com.cuiyongzhi.web.start;

import java.io.IOException;
import java.io.InputStream;
import java.util.Properties;

import com.cuiyongzhi.web.util.GlobalConstants;

/**
 * ClassName: InterfaceUrlInti
 * @Description: 項目啟動初始化方法
 * @author dapengniao
 * @date 2016 年 3 月 10 日 下午 4:08:21
 */
public class InterfaceUrlInti {

    public synchronized static void init(){
        ClassLoader cl = Thread.currentThread().getContextClassLoader();
        Properties props = new Properties();
        if(GlobalConstants.interfaceUrlProperties==null){
            GlobalConstants.interfaceUrlProperties = new Properties();
        }
        InputStream in = null;
        try {
            in = cl.getResourceAsStream("interface_url.properties");
            props.load(in);
            for(Object key : props.keySet()){
                GlobalConstants.interfaceUrlProperties.put(key, props.get(key));
            }

            props = new Properties();
            in = cl.getResourceAsStream("wechat.properties");
            props.load(in);
            for(Object key : props.keySet()){
                GlobalConstants.interfaceUrlProperties.put(key, props.get(key));
            }

        } catch (IOException e) {
            e.printStackTrace();
        }finally{
            if(in!=null){
                try {
                    in.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
        return;
    }

}

當我們把所有的準備工作都做好了之后我們可以開始真正的去獲取 token 了,這里我們將獲取到的 token 解析之后依然存儲到 GlobalConstants 中方便使用,簡單代碼如下:

package com.cuiyongzhi.wechat.common;

import java.util.HashMap;
import java.util.Map;

import net.sf.json.JSONObject;

import com.cuiyongzhi.web.util.GlobalConstants;
import com.cuiyongzhi.wechat.util.HttpUtils;

/**
 * ClassName: WeChatTask
 * @Description: 微信兩小時定時任務體
 * @author dapengniao
 * @date 2016 年 3 月 10 日 下午 1:42:29
 */
public class WeChatTask {
    /**
     * @Description: 任務執(zhí)行體
     * @param @throws Exception
     * @author dapengniao
     * @date 2016 年 3 月 10 日 下午 2:04:37
     */
    public void getToken_getTicket() throws Exception {
        Map<String, String> params = new HashMap<String, String>();
        params.put("grant_type", "client_credential");
        params.put("appid", GlobalConstants.getInterfaceUrl("appid"));
        params.put("secret", GlobalConstants.getInterfaceUrl("AppSecret"));
        String jstoken = HttpUtils.sendGet(
                GlobalConstants.getInterfaceUrl("tokenUrl"), params);
        String access_token = JSONObject.fromObject(jstoken).getString(
                "access_token"); // 獲取到 token 并賦值保存
        GlobalConstants.interfaceUrlProperties.put("access_token", access_token);
                System.out.println(new SimpleDateFormat("yyyy-MM-dd HH:mm:ss").format(new Date())+"token 為=============================="+access_token);
    }

}

(三)采用任務調度每隔兩小時執(zhí)行一次 token 獲取執(zhí)行體

我們閱讀過微信的文檔會發(fā)現(xiàn)我們的 token 獲取的接口每天是有調用次數(shù)限制的,為了防止我們業(yè)務量比較大的情況下 token 的直接調用的接口次數(shù)不夠用,所以我們需要根據(jù) token 的時效性(7200s)在自己的業(yè)務服務器上做到 token 的緩存并定時獲取,我這里用到的任務調度的方式是采用 quartz,有關 quartz 的使用可以參考文章 http://cuiyongzhi.com/?tags=%E5%AE%9A%E6%97%B6%E4%BB%BB%E5%8A%A1 ,下面具體代碼的實現(xiàn):

package com.cuiyongzhi.wechat.quartz;

import org.apache.log4j.Logger;

import com.cuiyongzhi.wechat.common.WeChatTask;

public class QuartzJob{
    private static Logger logger = Logger.getLogger(QuartzJob.class);
    /**
     * @Description: 任務執(zhí)行獲取 token
     * @param    
     * @author dapengniao
     * @date 2016 年 3 月 10 日 下午 4:34:26
     */
    public void workForToken() {
        try {
            WeChatTask timer = new WeChatTask();
            timer.getToken_getTicket();
        } catch (Exception e) {
            logger.error(e, e);
        }
    }

}

這里新建配置文件 spring-quartz.xml 以方便 quartz 任務的管理和啟用,這里將我們需要用到的 workForToken()加入到執(zhí)行任務中:

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE beans PUBLIC "-//SPRING//DTD BEAN//EN" "http://www.springframework.org/dtd/spring-beans.dtd">

<beans>
    <!-- 要調用的工作類 -->
    <bean id="quartzJob" class="com.cuiyongzhi.wechat.quartz.QuartzJob"></bean>

    <!-- 定義調用對象和調用對象的方法 -->
    <bean id="jobtaskForToken"
        class="org.springframework.scheduling.quartz.MethodInvokingJobDetailFactoryBean">
        <!-- 調用的類 -->
        <property name="targetObject">
            <ref bean="quartzJob" />
        </property>
        <!-- 調用類中的方法 -->
        <property name="targetMethod">
            <value>workForToken</value>
        </property>

    </bean>
    <!-- 定義觸發(fā)時間 -->
    <bean id="doTimeForToken" class="org.springframework.scheduling.quartz.CronTriggerBean">
        <property name="jobDetail">
            <ref bean="jobtaskForToken" />
        </property>
        <!-- cron 表達式 -->
        <property name="cronExpression">
            <value>0 0/1 * * * ?</value>
        </property>
    </bean>

    <!-- 總管理類 如果將 lazy-init='false'那么容器啟動就會執(zhí)行調度程序 -->
    <bean id="startQuertz" lazy-init="false" autowire="no"
        class="org.springframework.scheduling.quartz.SchedulerFactoryBean">
        <property name="triggers">
            <list>
                <ref bean="doTimeForToken" />
            </list>
        </property>
    </bean>

</beans>

這里我為了測試將執(zhí)行間隔時間設置成了 1 分鐘一次,根據(jù)需要可以自行修改執(zhí)行時間;最后我們需要在我們的 web.xml 啟動項中開啟 quartz 的使用:

<context-param>
        <param-name>contextConfigLocation</param-name>
        <param-value>classpath:spring.xml,classpath:spring-mybatis.xml,classpath:spring-quartz.xml</param-value>
        <!-- ,classpath:spring-quartz.xml 用于做任務調度 任務定時都可以 -->
    </context-param>

當這一切都準備完畢之后我們啟動項目,會發(fā)現(xiàn)每間隔一分鐘就會有 token 獲取到,這里我是將其存儲在項目變量中,但是如果需要考慮到項目橫向擴展這里建議將 token 存儲到緩存中;運行結果如下:

http://wiki.jikexueyuan.com/project/java-wechat/images/27.png" alt="" />

那么到這里 token 的獲取和保存就基本講完了,下一篇將講述【多媒體消息的回復】,感謝你的翻閱,如果有需要源碼或有疑問可以留言!