微信公众号开发之获取access_token[java版]

来源:互联网 发布:java 成员方法 编辑:程序博客网 时间:2024/05/16 14:30

在微信公众号开发的最基础的第一步就是获取access_token.

access_token的描述:

access_token是公众号的全局唯一票据,公众号调用各接口时都需使用access_token。正常情况下access_token有效期为7200秒,重复获取将导致上次获取的access_token失效。由于获取access_token的api调用次数非常有限,建议开发者全局存储与更新access_token,频繁刷新access_token会导致api调用受限,影响自身业务

 也就是说任何和微信服务器通讯的接口中都是以access_token来作为验证参数。

access_token的请求地址:

Html代码  收藏代码
  1. http请求方式: GET  
  2. https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid=APPID&secret=APPSECRET  

链接中有三个参数,分别是grant_type、appid和secret。根据图中的参数说明,grant_type传固定值client_credential,而appid和secret就是申请完自定义菜单后微信分配给我们的。

请求发送成功后,微信服务器会返回一个json串,包含access_token和expires_in两个元素。其中,access_token就是我们最终需要的凭证,而expires_in是凭证的有效期,单位是秒,7200秒也就是2个小时。这就意味着,不是每次访问特殊接口,都需要重新获取一次access_token,只要access_token还在有效期内,就一直可以使用。

 

通过get方式请求后微信服务器会返回json数据 如:

Java代码  收藏代码
  1. {"access_token":"ACCESS_TOKEN","expires_in":7200}  

 那么问题来了,怎么通过https呢?

下面来封装通用的请求方式[引用http://blog.csdn.net/lyq8479/article/details/9841371]

封装一个通用的请求方式,基本上需要符合以下几点:

1)支持HTTPS请求;

2)支持GET、POST两种方式;

3)支持有参数和无参。

对于https请求,我们还需要一个证书信任管理器,这个管理器类需要自己定义,需要实现X509TrustManager接口即可,代码如下

Java代码  收藏代码
  1. import java.security.cert.CertificateException;  
  2. import java.security.cert.X509Certificate;  
  3.   
  4. import javax.net.ssl.X509TrustManager;  
  5.   
  6. /** 
  7.  * 证书信任管理器(用于https请求) 
  8.  *  
  9.  */  
  10. public class MyX509TrustManager implements X509TrustManager {  
  11.   
  12.     public void checkClientTrusted(X509Certificate[] chain, String authType) throws CertificateException {  
  13.     }  
  14.   
  15.     public void checkServerTrusted(X509Certificate[] chain, String authType) throws CertificateException {  
  16.     }  
  17.   
  18.     public X509Certificate[] getAcceptedIssuers() {  
  19.         return null;  
  20.     }  
  21. }  

 

这个证书管理器的作用就是让它信任我们指定的证书,上面的代码意味着信任所有证书,不管是否权威机构颁发。

证书有了,通用的https请求方法就不难实现了,实现代码如下

Java代码  收藏代码
  1. import java.io.BufferedReader;  
  2. import java.io.InputStream;  
  3. import java.io.InputStreamReader;  
  4. import java.io.OutputStream;  
  5. import java.net.ConnectException;  
  6. import java.net.URL;  
  7.   
  8. import javax.net.ssl.HttpsURLConnection;  
  9. import javax.net.ssl.SSLContext;  
  10. import javax.net.ssl.SSLSocketFactory;  
  11. import javax.net.ssl.TrustManager;  
  12.   
  13. import net.sf.json.JSONObject;  
  14.   
  15. import org.slf4j.Logger;  
  16. import org.slf4j.LoggerFactory;  
  17.   
  18. /** 
  19.  * 公众平台通用接口工具类 
  20.  *  
  21.  * @date 2013-08-09 
  22.  */  
  23. public class WeixinUtil {  
  24.     private static Logger log = LoggerFactory.getLogger(WeixinUtil.class);  
  25.   
  26.     /** 
  27.      * 发起https请求并获取结果 
  28.      *  
  29.      * @param requestUrl 请求地址 
  30.      * @param requestMethod 请求方式(GET、POST) 
  31.      * @param outputStr 提交的数据 
  32.      * @return JSONObject(通过JSONObject.get(key)的方式获取json对象的属性值) 
  33.      */  
  34.     public static JSONObject httpRequest(String requestUrl, String requestMethod, String outputStr) {  
  35.         JSONObject jsonObject = null;  
  36.         StringBuffer buffer = new StringBuffer();  
  37.         try {  
  38.             // 创建SSLContext对象,并使用我们指定的信任管理器初始化  
  39.             TrustManager[] tm = { new MyX509TrustManager() };  
  40.             SSLContext sslContext = SSLContext.getInstance("SSL""SunJSSE");  
  41.             sslContext.init(null, tm, new java.security.SecureRandom());  
  42.             // 从上述SSLContext对象中得到SSLSocketFactory对象  
  43.             SSLSocketFactory ssf = sslContext.getSocketFactory();  
  44.   
  45.             URL url = new URL(requestUrl);  
  46.             HttpsURLConnection httpUrlConn = (HttpsURLConnection) url.openConnection();  
  47.             httpUrlConn.setSSLSocketFactory(ssf);  
  48.   
  49.             httpUrlConn.setDoOutput(true);  
  50.             httpUrlConn.setDoInput(true);  
  51.             httpUrlConn.setUseCaches(false);  
  52.             // 设置请求方式(GET/POST)  
  53.             httpUrlConn.setRequestMethod(requestMethod);  
  54.   
  55.             if ("GET".equalsIgnoreCase(requestMethod))  
  56.                 httpUrlConn.connect();  
  57.   
  58.             // 当有数据需要提交时  
  59.             if (null != outputStr) {  
  60.                 OutputStream outputStream = httpUrlConn.getOutputStream();  
  61.                 // 注意编码格式,防止中文乱码  
  62.                 outputStream.write(outputStr.getBytes("UTF-8"));  
  63.                 outputStream.close();  
  64.             }  
  65.   
  66.             // 将返回的输入流转换成字符串  
  67.             InputStream inputStream = httpUrlConn.getInputStream();  
  68.             InputStreamReader inputStreamReader = new InputStreamReader(inputStream, "utf-8");  
  69.             BufferedReader bufferedReader = new BufferedReader(inputStreamReader);  
  70.   
  71.             String str = null;  
  72.             while ((str = bufferedReader.readLine()) != null) {  
  73.                 buffer.append(str);  
  74.             }  
  75.             bufferedReader.close();  
  76.             inputStreamReader.close();  
  77.             // 释放资源  
  78.             inputStream.close();  
  79.             inputStream = null;  
  80.             httpUrlConn.disconnect();  
  81.             jsonObject = JSONObject.fromObject(buffer.toString());  
  82.         } catch (ConnectException ce) {  
  83.             log.error("Weixin server connection timed out.");  
  84.         } catch (Exception e) {  
  85.             log.error("https request error:{}", e);  
  86.         }  
  87.         return jsonObject;  
  88.     }  
  89. }  

 

在获取凭证创建菜单前,我们还需要封装一些pojo,这会让我们的代码更美观,有条理。

首先是调用获取凭证接口后,微信服务器会返回json格式的数据:{"access_token":"ACCESS_TOKEN","expires_in":7200},我们将其封装为一个AccessToken对象,对象有二个属性:token和expiresIn,代码如下:

Java代码  收藏代码
  1. /** 
  2.  * 微信通用接口凭证 
  3.  *  
  4.  * @date 2013-08-08 
  5.  */  
  6. public class AccessToken {  
  7.     // 获取到的凭证  
  8.     private String token;  
  9.     // 凭证有效时间,单位:秒  
  10.     private int expiresIn;  
  11.   
  12.     public String getToken() {  
  13.         return token;  
  14.     }  
  15.   
  16.     public void setToken(String token) {  
  17.         this.token = token;  
  18.     }  
  19.   
  20.     public int getExpiresIn() {  
  21.         return expiresIn;  
  22.     }  
  23.   
  24.     public void setExpiresIn(int expiresIn) {  
  25.         this.expiresIn = expiresIn;  
  26.     }  
  27. }  

 pojo类定义好之后,接下来就是获取access_token了

Java代码  收藏代码
  1. // 获取access_token的接口地址(GET) 限200(次/天)  
  2. public final static String access_token_url = "https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid=APPID&secret=APPSECRET";  
  3.   
  4. /** 
  5.  * 获取access_token 
  6.  *  
  7.  * @param appid 凭证 
  8.  * @param appsecret 密钥 
  9.  * @return 
  10.  */  
  11. public static AccessToken getAccessToken(String appid, String appsecret) {  
  12.     AccessToken accessToken = null;  
  13.   
  14.     String requestUrl = access_token_url.replace("APPID", appid).replace("APPSECRET", appsecret);  
  15.     JSONObject jsonObject = httpRequest(requestUrl, "GET"null);  
  16.     // 如果请求成功  
  17.     if (null != jsonObject) {  
  18.         try {  
  19.             accessToken = new AccessToken();  
  20.             accessToken.setToken(jsonObject.getString("access_token"));  
  21.             accessToken.setExpiresIn(jsonObject.getInt("expires_in"));  
  22.         } catch (JSONException e) {  
  23.             accessToken = null;  
  24.             // 获取token失败  
  25.             log.error("获取token失败 errcode:{} errmsg:{}", jsonObject.getInt("errcode"), jsonObject.getString("errmsg"));  
  26.         }  
  27.     }  
  28.     return accessToken;  
  29. }  

 

这样,就可以轻松得从返回的accessToken对象中获取accessToken了。

注意:

因为access_token每天得调用次数有限,建议不要每次的通讯都去获取新的access_token,可以将其用管理器的方式来进行管理,当失效的时候再重新获取。

1 0
原创粉丝点击