本文为原创,原始地址为http://www.cnblogs.com/fengzheng/p/5027630.html

上一篇说了微信开发的准备工作,准备工作完成之后,就要开始步入正题了。其实微信公众号开发,说白了,就是要构造和发送http或https的请求组成,并根据请求的返回数据做逻辑处理。

今天就来说一说微信开发第一步,公众号接入以及access_token的管理。

微信公众号接入

在微信公众号开发手册上,关于公众号接入这一节内容还是写的比较详细的,文档中说接入公众号需要3个步骤,分别是:

1、填写服务器配置
2、验证服务器地址的有效性
3、依据接口文档实现业务逻辑

其实,第3步已经不能算做公众号接入的步骤,而是接入之后,开发人员可以根据微信公众号提供的接口所能做的一些开发。

第1步中服务器配置包含服务器地址(URL)、Token和EncodingAESKey。

服务器地址即公众号后台提供业务逻辑的入口地址,目前只支持80端口,之后包括接入验证以及任何其它的操作的请求(例如消息的发送、菜单管理、素材管理等)都要从这个地址进入。接入验证和其它请求的区别就是,接入验证时是get请求,其它时候是post请求;

Token可由开发者可以任意填写,用作生成签名(该Token会和接口URL中包含的Token进行比对,从而验证安全性);

EncodingAESKey由开发者手动填写或随机生成,将用作消息体加解密密钥。本例中全部以未加密的明文消息方式,不涉及此配置项。

第2步,验证服务器地址的有效性,当点击“提交”按钮后,微信服务器将发送一个http的get请求到刚刚填写的服务器地址,并且携带四个参数:

接到请求后,我们需要做如下三步,若确认此次GET请求来自微信服务器,原样返回echostr参数内容,则接入生效,否则接入失败。

1. 将token、timestamp、nonce三个参数进行字典序排序
2. 将三个参数字符串拼接成一个字符串进行sha1加密
3. 开发者获得加密后的字符串可与signature对比,标识该请求来源于微信

代码会说话,以下是我定义的一个入口servlevt,在其中的doGet方法中定义校验方法:

  1. //token
  2. private final String token = "fengzheng";
  3.  
  4. protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
  5. System.out.println("开始签名校验");
  6. String signature = request.getParameter("signature");
  7. String timestamp = request.getParameter("timestamp");
  8. String nonce = request.getParameter("nonce");
  9. String echostr = request.getParameter("echostr");
  10.  
  11. ArrayList<String> array = new ArrayList<String>();
  12. array.add(signature);
  13. array.add(timestamp);
  14. array.add(nonce);
  15.  
  16. //排序
  17. String sortString = sort(token, timestamp, nonce);
  18. //加密
  19. String mytoken = Decript.SHA1(sortString);
  20. //校验签名
  21. if (mytoken != null && mytoken != "" && mytoken.equals(signature)) {
  22. System.out.println("签名校验通过。");
  23. response.getWriter().println(echostr); //如果检验成功输出echostr,微信服务器接收到此输出,才会确认检验完成。
  24. } else {
  25. System.out.println("签名校验失败。");
  26. }
  27. }
  28.  
  29. /**
  30. * 排序方法
  31. * @param token
  32. * @param timestamp
  33. * @param nonce
  34. * @return
  35. */
  36. public static String sort(String token, String timestamp, String nonce) {
  37. String[] strArray = { token, timestamp, nonce };
  38. Arrays.sort(strArray);
  39.  
  40. StringBuilder sbuilder = new StringBuilder();
  41. for (String str : strArray) {
  42. sbuilder.append(str);
  43. }
  44.  
  45. return sbuilder.toString();
  46. }

以下代码是加密的方法:

  1. public class Decript {
  2.  
  3. public static String SHA1(String decript) {
  4. try {
  5. MessageDigest digest = MessageDigest
  6. .getInstance("SHA-1");
  7. digest.update(decript.getBytes());
  8. byte messageDigest[] = digest.digest();
  9. // Create Hex String
  10. StringBuffer hexString = new StringBuffer();
  11. // 字节数组转换为 十六进制 数
  12. for (int i = 0; i < messageDigest.length; i++) {
  13. String shaHex = Integer.toHexString(messageDigest[i] & 0xFF);
  14. if (shaHex.length() < 2) {
  15. hexString.append(0);
  16. }
  17. hexString.append(shaHex);
  18. }
  19. return hexString.toString();
  20.  
  21. } catch (NoSuchAlgorithmException e) {
  22. e.printStackTrace();
  23. }
  24. return "";
  25. }
  26. }

servlet映射的xml如下:

  1. <servlet>
  2. <servlet-name>Start</servlet-name>
  3. <servlet-class>org.fengzheng.wechat.Start</servlet-class>
  4. </servlet>
  5. <servlet-mapping>
  6. <servlet-name>Start</servlet-name>
  7. <url-pattern>/wechat</url-pattern>
  8. </servlet-mapping>

  

我这里用的是IntelliJ IDEA+tomcat7.0开发,直接启动项目,然后用ngrok将本地8080端口映射到外网。进入微信测试公众号管理界面,在接口配置信息中填入映射的外网地址和token

点击提交按钮,页面会提示配置成功,

会到IDE,看到控制台中输出了信息

  

access_token管理

在将access_token之前,还有两个重要参数需要知晓,这两个参数分别是appID和appsecret,这是在申请公众号的时候自动分配给公众号的,相当于公众号的身份标示,在很多接口中需要这两个参数,接下来在请求access_token的时候就需要这两个参数。

公众号接入成功之后,接下来就要实现相应的逻辑了。在使用微信公众号接口中,发现有许多请求都需要access_token。access_token是公众号的全局唯一凭证,公众号调用各接口时都需使用access_token。开发者需要进行妥善保存。access_token的存储至少要保留512个字符空间。access_token的有效期目前为2个小时,需定时刷新,重复获取将导致上次获取的access_token失效。并且每天调用获取access_token接口的上限是2000次。

总结以上说明,access_token需要做到以下两点:

1.因为access_token有2个小时的时效性,要有一个机制保证最长2个小时重新获取一次;

2.因为接口调用上限每天2000次,所以不能调用太频繁;

就此,这里采用的方案是这样的,定义一个默认启动的servlet,在init方法中启动一个Thread,这个进程中定义一个无限循环的方法,用来获取access_token,当获取成功后,此进程休眠7000秒,否则休眠3秒钟继续获取。流程图如下:

下面正式开始在工程中实现以上思路,因为返回的数据都是json格式,这里会用到阿里的fastjson库,为构造请求和处理请求后的数据序列化和反序列化提供支持。后续的其它接口也会用到。

1.定义一个AccessToken实体

  1. public class AccessToken {
  2. public String getAccessToken() {
  3. return accessToken;
  4. }
  5.  
  6. public void setAccessToken(String accessToken) {
  7. this.accessToken = accessToken;
  8. }
  9.  
  10. public int getExpiresin() {
  11. return expiresin;
  12. }
  13.  
  14. public void setExpiresin(int expiresin) {
  15. this.expiresin = expiresin;
  16. }
  17.  
  18. private String accessToken;
  19.  
  20. private int expiresin;
  21. }

2.定义一个默认启动的servlet,在init方法中启动一个Thread,并在web.xml中将这个servlet设置为默认自启动的。

  1. import javax.servlet.ServletException;
  2. import javax.servlet.annotation.WebServlet;
  3. import javax.servlet.http.HttpServlet;
  4. import javax.servlet.http.HttpServletRequest;
  5. import javax.servlet.http.HttpServletResponse;
  6. import java.io.IOException;
  7.  
  8. /**
  9. * Created by huzhicheng on 2015/12/8.
  10. */
  11. @WebServlet(name = "AccessTokenServlet")
  12. public class AccessTokenServlet extends HttpServlet {
  13.  
  14. public void init() throws ServletException {
  15. TokenThread.appId = getInitParameter("appid"); //获取servlet初始参数appid和appsecret
  16. TokenThread.appSecret = getInitParameter("appsecret");
  17. System.out.println("appid:"+TokenThread.appId);
  18. System.out.println("appSecret:"+TokenThread.appSecret);
  19. new Thread(new TokenThread()).start(); //启动进程
  20. }
  21.  
  22. protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
  23.  
  24. }
  25.  
  26. protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
  27.  
  28. }
  29. }

在web.xml中设置servlet自启动,并设置初始化参数appid和appsecret

  1. <servlet>
  2. <servlet-name>initAccessTokenServlet</servlet-name>
  3. <servlet-class>
  4. org.fengzheng.wechat.accesstoken.AccessTokenServlet
  5. </servlet-class>
  6. <init-param>
  7. <param-name>appid</param-name>
  8. <param-value>your appid</param-value>
  9. </init-param>
  10. <init-param>
  11. <param-name>appsecret</param-name>
  12. <param-value>your appsecret</param-value>
  13. </init-param>
  14. <load-on-startup>0</load-on-startup>
  15. </servlet>

3.定义Thread类,在此类中调用access_token获取接口,并将得到的数据抽象到静态实体,以便在其它地方使用。接口地址为https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid=APPID&secret=APPSECRET,其中grant_type固定写为client_credential即可。此请求为https的get请求,返回的数据格式为{"access_token":"ACCESS_TOKEN","expires_in":7200}。

进程类实现如下:

  1. import com.alibaba.fastjson.JSON;
  2. import com.alibaba.fastjson.JSONObject;
  3. import org.fengzheng.wechat.common.NetWorkHelper;
  4.  
  5. /**
  6. * Created by huzhicheng on 2015/11/5.
  7. */
  8. public class TokenThread implements Runnable {
  9. public static String appId = "";
  10.  
  11. public static String appSecret= "";

  12.   //注意是静态的
  13. public static AccessToken accessToken = null;
  14.  
  15. public void run(){
  16. while (true){
  17. try{
  18. accessToken = this.getAccessToken();
  19. if(null!=accessToken){
  20. System.out.println(accessToken.getAccessToken());
  21. Thread.sleep(7000 * 1000); //获取到access_token 休眠7000秒
  22.  
  23. }else{
  24. Thread.sleep(1000*3); //获取的access_token为空 休眠3秒
  25. }
  26. }catch(Exception e){
  27. System.out.println("发生异常:"+e.getMessage());
  28. e.printStackTrace();
  29. try{
  30. Thread.sleep(1000*10); //发生异常休眠1秒
  31. }catch (Exception e1){
  32.  
  33. }
  34. }
  35. }
  36. }
  37.  
  38. /**
  39. * 获取access_token
  40. * @return
  41. */
  42. private AccessToken getAccessToken(){
  43. NetWorkHelper netHelper = new NetWorkHelper();
  44. String Url = String.format("https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid=%s&secret=%s",this.appId,this.appSecret);
  45. String result = netHelper.getHttpsResponse(Url,"");
  46. System.out.println(result);
  47. //response.getWriter().println(result);
  48. JSONObject json = JSON.parseObject(result);
  49. AccessToken token = new AccessToken();
  50. token.setAccessToken(json.getString("access_token"));
  51. token.setExpiresin(json.getInteger("expires_in"));
  52. return token;
  53. }
  54. }

其中NetWorkHelper中getHttpsResponse方法是请求一个https地址,参数requestMethod为字符串“GET”或者“POST”,传null或者“”默认为get方式。

实现如下:

  1. public String getHttpsResponse(String hsUrl,String requestMethod) {
  2. URL url;
  3. InputStream is = null;
  4. String resultData = "";
  5. try {
  6. url = new URL(hsUrl);
  7. HttpsURLConnection con = (HttpsURLConnection) url.openConnection();
  8. TrustManager[] tm = {xtm};
  9.  
  10. SSLContext ctx = SSLContext.getInstance("TLS");
  11. ctx.init(null, tm, null);
  12.  
  13. con.setSSLSocketFactory(ctx.getSocketFactory());
  14. con.setHostnameVerifier(new HostnameVerifier() {
  15. @Override
  16. public boolean verify(String arg0, SSLSession arg1) {
  17. return true;
  18. }
  19. });
  20.  
  21. con.setDoInput(true); //允许输入流,即允许下载
  22.  
  23. //在android中必须将此项设置为false
  24. con.setDoOutput(false); //允许输出流,即允许上传
  25. con.setUseCaches(false); //不使用缓冲
  26. if(null!=requestMethod && !requestMethod.equals("")) {
  27. con.setRequestMethod(requestMethod); //使用指定的方式
  28. }
  29. else{
  30. con.setRequestMethod("GET"); //使用get请求
  31. }
  32. is = con.getInputStream(); //获取输入流,此时才真正建立链接
  33. InputStreamReader isr = new InputStreamReader(is);
  34. BufferedReader bufferReader = new BufferedReader(isr);
  35. String inputLine = "";
  36. while ((inputLine = bufferReader.readLine()) != null) {
  37. resultData += inputLine + "\n";
  38. }
  39. System.out.println(resultData);
  40.  
  41. Certificate[] certs = con.getServerCertificates();
  42.  
  43. int certNum = 1;
  44.  
  45. for (Certificate cert : certs) {
  46. X509Certificate xcert = (X509Certificate) cert;
  47. }
  48.  
  49. } catch (Exception e) {
  50. e.printStackTrace();
  51. }
  52. return resultData;
  53. }
  54.  
  55. X509TrustManager xtm = new X509TrustManager() {
  56. @Override
  57. public X509Certificate[] getAcceptedIssuers() {
  58. // TODO Auto-generated method stub
  59. return null;
  60. }
  61.  
  62. @Override
  63. public void checkServerTrusted(X509Certificate[] arg0, String arg1)
  64. throws CertificateException {
  65. // TODO Auto-generated method stub
  66.  
  67. }
  68.  
  69. @Override
  70. public void checkClientTrusted(X509Certificate[] arg0, String arg1)
  71. throws CertificateException {
  72. // TODO Auto-generated method stub
  73.  
  74. }
  75. };

至此代码实现完毕,将项目部署,看到控制台输出如下:

  

为方面看效果,可以把休眠时间设置短一点,比如30秒获取一次,然后将access_token输出。下面做一个测试jsp页面,并把休眠时间设置为30秒,这样过30秒刷新页面,就可以看到变化,顺便演示一下在其它地方如何拿到access_token

  1. <%@ page contentType="text/html;charset=UTF-8" language="java" %>
  2. <%@ page import="org.fengzheng.wechat.accesstoken.TokenThread" %>
  3. <html>
  4. <head>
  5. <title></title>
  6. </head>
  7. <body>
  8. access_token为:<%=TokenThread.accessToken.getAccessToken()%>
  9. </body>
  10. </html>

这样在浏览器上浏览这个页面,显示效果如下:

30秒后刷新,这个值发生了变化:

  

代码已上传至github,仓库会随时更新,目前只有本篇所讲的代码。

欢迎关注公众号「gushidefengzheng」古时的风筝

还可以加入 Java 微信讨论群(如果二维码过期:请加微信:fengdezitai001 ,备注:cnblogs):

用java开发微信公众号:公众号接入和access_token管理(二)的更多相关文章

  1. Java开发微信公众号(五)---微信开发中如何获取access_token以及缓存access_token

    获取access_token是微信api最重要的一个部分,因为调用其他api很多都需要用到access_token.比如自定义菜单接口.客服接口.获取用户信息接口.用户分组接口.群发接口等在请求的时候 ...

  2. Java开发微信公众号(四)---微信服务器post消息体的接收及消息的处理

    在前几节文章中我们讲述了微信公众号环境的搭建.如何接入微信公众平台.以及微信服务器请求消息,响应消息,事件消息以及工具处理类的封装:接下来我们重点说一下-微信服务器post消息体的接收及消息的处理,这 ...

  3. Java开发微信公众号(三)---微信服务器请求消息,响应消息,事件消息以及工具处理类的封装

    在前面几篇文章我们讲了微信公众号环境的配置 和微信公众号服务的接入,接下来我们来说一下微信服务器请求消息,响应消息以及事件消息的相关内容,首先我们来分析一下消息类型和返回xml格式及实体类的封装. ( ...

  4. Java开发微信公众号(二)---开启开发者模式,接入微信公众平台开发

    接入微信公众平台开发,开发者需要按照如下步骤完成: 1.填写服务器配置 2.验证服务器地址的有效性 3.依据接口文档实现业务逻辑 资料准备: 1.一个可以访问的外网,即80的访问端口,因为微信公众号接 ...

  5. Java开发微信公众号(一)---初识微信公众号以及环境搭建

    ps:1.开发语言使用Java springMvc+Mybaits+spring maven实现 2.使用微信接口测试账号进行本地测试 https://mp.weixin.qq.com/debug/c ...

  6. 用java开发微信公众号:测试公众号与本地测试环境搭建(一)

    本文为原创,原始地址为:http://www.cnblogs.com/fengzheng/p/5023678.html 俗话说,工欲善其事,必先利其器.要做微信公众号开发,两样东西不可少,那就是要有一 ...

  7. JAVA开发微信支付-公众号支付/微信浏览器支付(JSAPI)

    写这篇文章的目的有2个,一是自己的项目刚开发完微信支付功能,趁热回个炉温习一下,二也是帮助像我这样对微信支付不熟悉,反复看了多天文档还是一知半解,原理都没摸清,更不要说实现了.本以为网上的微信开发教程 ...

  8. 使用Java开发微信公众平台(二)——消息的接收与响应

    上一篇文章(http://www.jerehedu.com/fenxiang/171807_for_detail.htm )中,我们学习了使用Java语言开发微信公众平台的第一部分——环境搭建与开发接 ...

  9. 使用Java开发微信公众平台(四)——消息的接收与响应

    上一篇文章(http://www.jerehedu.com/fenxiang/171807_for_detail.htm )中,我们学习了使用Java语言开发微信公众平台的第一部分——环境搭建与开发接 ...

随机推荐

  1. 十分钟介绍mobx与react

    原文地址:https://mobxjs.github.io/mobx/getting-started.html 写在前面:本人英语水平有限,主要是写给自己看的,若有哪位同学看到了有问题的地方,请为我指 ...

  2. Travis CI用来持续集成你的项目

    这里持续集成基于GitHub搭建的博客为项目 工具: zqz@ubuntu:~$ node --version v4.2.6 zqz@ubuntu:~$ git --version git versi ...

  3. ASP.NET Aries 入门开发教程1:框架下载与运行

    背景: 鉴于框架的使用者越来越多,文档太少,不少用户反映框架的入门门槛太高. 好吧,再辛苦下,抽时间写教程吧! 步骤1:下载框架源码 开源地址:https://github.com/cyq1162/A ...

  4. .NET Core系列 :4 测试

    2016.6.27 微软已经正式发布了.NET Core 1.0 RTM,但是工具链还是预览版,同样的大量的开源测试库也都是至少发布了Alpha测试版支持.NET Core, 这篇文章 The Sta ...

  5. tomcat开发远程调试端口以及利用eclipse进行远程调试

    一.tomcat开发远程调试端口 方法1 WIN系统 在catalina.bat里:  SET CATALINA_OPTS=-server -Xdebug -Xnoagent -Djava.compi ...

  6. 视频 - 在 VirtualBox 中部署 OpenStack

    大家新年好,CloudMan 今天给大家带来一件新年礼物. 一直以来大家都反馈 OpenStack 学习有两大障碍:1. 实验环境难搭2. 体系复杂,难道大今天我就先帮大家解决环境问题.前两天我抽空在 ...

  7. ubuntu如何安装nodejs最新版 本

    如何正确的安装nodejs? 我们可以先安装nvm, git clone https://github.com/creationix/nvm.git ~/.nvm 然后打开 ~/.bashrc ,   ...

  8. MJRefresh 源码解读 + 使用

    MJRefresh这个刷新控件是一款非常好用的框架,我们在使用一个框架的同时,最好能了解下它的实现原理,不管是根据业务要求在原有的基础上修改代码,还是其他的目的,弄明白作者的思路和代码风格,会受益匪浅 ...

  9. 菜鸟Python学习笔记第二天:关于Python黑客。

    2016年1月5日 星期四 天气:还好 一直不知道自己为什么要去学Python,其实Python能做到的Java都可以做到,Python有的有点Java也有,而且Java还是必修课,可是就是不愿意去学 ...

  10. Openfiler配置RAC共享存储

    将 Openfiler 用作 iSCSI 存储服务器,主要操作步骤如下: 1.设置 iSCSI 服务 2.配置网络访问 3.指定物理存储器并对其分区 4.创建新的卷组 5.创建所有逻辑卷 6.为每个逻 ...