一 access_token基本概念

  定义:access_token是公众号的全局唯一接口调用凭据,公众号调用各接口时都需使用access_token。开发者需要进行妥善保存。

  时效性:access_token的有效期目前为2个小时(7200秒),需定时刷新,重复获取将导致上次获取的access_token失效。

  针对时效性问题的处理方法:

    1 让系统每隔2个小时去刷新一次票据,这样无论合适我们内部调用接口,这个票据始终是最新的;

    2 为了方便频繁调用,我们需要把票据存储在一个地方,并且是唯一的一个地方,这个地方要被所有的子文件都能访问到,一定不能存在内存中。

  公众平台的API调用所需的access_token的使用及生成方式说明:

    1、建议公众号开发者使用中控服务器统一获取和刷新Access_token,其他业务逻辑服务器所使用的access_token均来自于该中控服务器,不应该各自去刷新,否则容易造成冲突,导致access_token覆盖而影响业务;

    2、目前Access_token的有效期通过返回的expire_in来传达,目前是秒之内的值。中控服务器需要根据这个有效时间提前去刷新新access_token。在刷新过程中,中控服务器对外输出的依然是老access_token,此时公众平台后台会保证在刷新短时间内,新老access_token都可用,这保证了第三方业务的平滑过渡;

    3、Access_token的有效时间可能会在未来有调整,所以中控服务器不仅需要内部定时主动刷新,还需要提供被动刷新access_token的接口,这样便于业务服务器在API调用获知access_token已超时的情况下,可以触发access_token的刷新流程。

    公众号可以使用AppID和AppSecret调用本接口来获取access_token。AppID和AppSecret可在“微信公众平台-开发-基本配置”页中获得(需要已经成为开发者,且帐号没有异常状态)。调用接口时,请登录“微信公众平台-开发-基本配置”提前将服务器IP地址添加到IP白名单中,点击查看设置方法,否则将无法调用成功。

  接口调用请求说明:

    https请求方式: GET
    https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid=APPID&secret=APPSECRET

  获取票据次数上限:每日2000次

二 获取access_token逻辑代码示例

  新建项目文件wechat,项目目录结构如下:

  app.js文件代码:

  1. 'use strict';
  2. // 引入模块
  3. var Koa = require('koa');
  4. var path = require('path');
  5. //引入中间件
  6. var generator = require('./model/generator');
  7. var util = require('./libs/util');
  8. //引入文本文件
  9. var wechat_file = path.join(__dirname, './config/wechat.txt');
  10. // 声明对象字面量config 用于存储配置信息、读取写入票据的方法
  11. var config = {
  12. wechat: {
  13. appID: 'wx044125d0a173dd15',
  14. appSecret: '21295f049b49fe324d7302186c294fe7',
  15. token: 'beijingjiangweiwechatlearntokenvalueabcdefg',
  16. getAccessToken:function(){
  17. return util.readFileAsync(wechat_file);
  18. },
  19. saveAccessToken:function(data){
  20. data = JSON.stringify(data);
  21. return util.writeFileAsync(wechat_file, data);
  22. }
  23. }
  24. }
  25.  
  26. // 实例化Koa的web服务器
  27. var app = new Koa();
  28. //传入配置参数
  29. app.use(generator(config.wechat));
  30. //监听3100端口
  31. app.listen(3100);
  32. //console.log('listening:3100');

  generator.js文件代码:

  1. 'use strict';
  2. // 引入模块
  3. var sha1 = require('sha1');
  4. var Promise = require('bluebird');
  5. var request = Promise.promisify(require('request'));
  6.  
  7. //增加url配置项
  8. var prefix = 'https://api.weixin.qq.com/cgi-bin/';
  9. var api = {
  10. accessToken: prefix + 'token?grant_type=client_credential'
  11. }
  12.  
  13. //利用构造函数生成实例 完成票据存储逻辑
  14. function weChat(opts) {
  15. var that = this;
  16. this.appID = opts.appID;
  17. this.appSecret = opts.appSecret;
  18. this.getAccessToken = opts.getAccessToken;
  19. this.saveAccessToken = opts.saveAccessToken;
  20. //获取票据的方法
  21. this.getAccessToken()
  22. .then(function(data) {
  23. //从静态文件获取票据,JSON化数据,如果有异常,则尝试更新票据
  24. try {
  25. data = JSON.parse(data);
  26. } catch (e) {
  27. return that.updateAccessToken();
  28. }
  29. //判断票据是否在有效期内,如果合法,向下传递票据,如果不合法,更新票据
  30. if (that.isValidAccessToken(data)) {
  31. Promise.resolve(data);
  32. } else {
  33. return that.updateAccessToken();
  34. }
  35. })
  36. //将拿到的票据信息和有效期信息存储起来
  37. .then(function(data) {
  38. console.log(data);
  39. that.access_token = data.access_token;
  40. that.expires_in = data.expires_in;
  41.  
  42. that.saveAccessToken(data);
  43. })
  44. };
  45.  
  46. //在weChat的原型链上增加验证有效期的方法
  47. weChat.prototype.isValidAccessToken = function(data) {
  48. //进行判断,如果票据不合法,返回false
  49. if (!data || !data.access_token || !data.expires_in) {
  50. return false;
  51. }
  52. //拿到票据和过期时间的数据
  53. var access_token = data.access_token;
  54. var expires_in = data.expires_in;
  55. //获取当前时间
  56. var now = (new Date().getTime());
  57. //如果当前时间小于票据过期时间,返回true,否则返回false
  58. if (now < expires_in) {
  59. return true;
  60. } else {
  61. return false;
  62. };
  63. };
  64.  
  65. //在weChat的原型链上增加更新票据的方法
  66. weChat.prototype.updateAccessToken = function() {
  67. var appID = this.appID;
  68. var appSecret = this.appSecret;
  69. var url = api.accessToken + '&appid=' + appID + '&secret=' + appSecret;
  70.  
  71. return new Promise(function(resolve, reject) {
  72. //使用request发起请求
  73. request({
  74. url: url,
  75. json: true
  76. }).then(function(response) {
  77. var data = response.body;
  78. var now = (new Date().getTime());
  79. var expires_in = now + (data.expires_in - 20) * 1000;
  80. //把新票据的有效时间赋值给data
  81. data.expires_in = expires_in;
  82. resolve(data);
  83. })
  84. })
  85. };
  86.  
  87. // 建立中间件函数并暴露出去
  88. module.exports = function(opts) {
  89. //实例化weChat()函数
  90. var wechat = new weChat(opts);
  91. return function*(next) {
  92. console.log(this.query);
  93. var token = opts.token;
  94. var signature = this.query.signature;
  95. var nonce = this.query.nonce;
  96. var timestamp = this.query.timestamp;
  97. var echostr = this.query.echostr;
  98. // 进行字典排序
  99. var str = [token, timestamp, nonce].sort().join('');
  100. // 进行加密
  101. var sha = sha1(str);
  102. // 判断加密后的值是否等于签名值
  103. if (sha === signature) {
  104. this.body = echostr + '';
  105. } else {
  106. this.body = 'wrong';
  107. }
  108. }
  109. };

  util.js文件代码:

  1. 'use strict';
  2. //引入模块
  3. var fs = require('fs');
  4. var Promise = require('bluebird');
  5. //读取票据信息
  6. exports.readFileAsync = function(fpath, encoding) {
  7. return new Promise(function(resolve, reject) {
  8. fs.readFile(fpath, encoding, function(err, content) {
  9. if (err) {
  10. reject(err);
  11. }else{
  12. resolve(content);
  13. };
  14. });
  15. });
  16. };
  17. //写入票据信息
  18. exports.writeFileAsync = function(fpath, content) {
  19. return new Promise(function(resolve, reject) {
  20. fs.writeFile(fpath, content, function(err) {
  21. if (err) {
  22. reject(err);
  23. }else{
  24. resolve();
  25. };
  26. });
  27. });
  28. };

  实现获取微信票据的全部代码,我都贴了出来,现在来对照代码捋顺一遍逻辑,在app.js文件中,我们声明了对象字面量config对象,把微信的appID、appSecret、token以及获取(getAccessToken)和更新(saveAccessToken)票据的方法,然后把config对象传给了generator即generator.js文件weChat()函数,注意generator.js文件中下方暴露出来的函数,首先是实例化了weChat()函数,首先把config对象传递过来的配置信息放在自身的属性上,然后调用自身获取票据的方法getAccessToken()去读取文本文件里的token值,如果值是合法的,就向下传递,如果不合法就更新票据,更新时调用updateAccessToken()方法,请求微信服务的url地址,从地址里拿到返回的JSON数据,把数据中的票据时间缩短20秒,然后赋给数据本身,然后再把数据向下传递,最终会拿到一个有效的票据结果,然后把票据的两个字段access_token、expires_in赋值给自身,并调用saveAccessToken()方法,把票据结果存储到文本文件中,票据获取的过程就完成了。

  在终端启动app.js文件,启动魔法隧道,微信测试号接口配置也没有问题的话,我们可以看到在wechat.txt文件中,已经写入票据信息了。

  备注:实现获取票据的过程,我只贴了代码示例,里面的各个依赖模块,需要手动安装(使用npm install)。

注意:因为官方文档是处于更新状态的,所以后面关于微信公众号的知识点,可能跟最新的文档有一定的差异,所以开发的时候还是要以最新的文档为准。

5 微信票据 access_token--开发微信的第二道坎儿的更多相关文章

  1. 微信公众号开发--微信JS-SDK分享到朋友圈和分享给朋友

    之前写过一篇使用微信JS-SDK来实现扫一扫功能的博客 微信公众号开发–微信JS-SDK扫一扫功能 在该博客里介绍了微信JS-SDK的基本用法,其中包括以下几个步骤 还详细介绍了通过config接口注 ...

  2. C#开发微信公众平台开发-微信海报介绍和开发流程

    “让客户发展客户”,微信海报才是微信公众平台最高明的吸粉手段,海报上有粉丝的专属二维码,有粉丝的头像及商户宣传的广告等.新粉丝扫描这个专属二维码会关注公众号,同时分享海报的粉丝会增加积分换取礼品或者优 ...

  3. 微信公众平台开发——微信授权登录(OAuth2.0)

    1.OAuth2.0简介 OAuth(开放授权)是一个开放标准,允许用户让第三方应用访问该用户在某一网站上存储的私密的资源(如照片,视频,联系人列表),而无需将用户名和密码提供给第三方应用. 允许用户 ...

  4. 微信公众平台开发-微信服务器IP接口实例(含源码)

    微信公众平台开发-access_token获取及应用(含源码)作者: 孟祥磊-<微信公众平台开发实例教程> 学习了access_token的获取及应用后,正式的使用access_token ...

  5. 微信公众平台开发 微信JSSDK开发

    根据微信开发文档步骤如下: 1.先登录微信公众平台进入“公众号设置”的“功能设置”里填写“JS接口安全域名”. JS接口安全域名设置 mi.com(前面不用带www/http,域名必须备案过) 2.引 ...

  6. [5] 微信公众号开发 - 微信支付功能开发(网页JSAPI调用)

    1.微信支付的流程 如下三张手机截图,我们在微信网页端看到的支付,表面上看到的是 "点击支付按钮 - 弹出支付框 - 支付成功后出现提示页面",实际上的核心处理过程是: 点击支付按 ...

  7. C#微信公众号开发--微信事件交互

    前言 一切准备工作就绪时就先实现一个关注公众号后向客户端推送一条消息.关注后推送消息需要一个get请求.一个post请求,get请求主要是为了向微信服务器验证,post请求主要就是处理微信消息了. 调 ...

  8. 微信小程序开发----微信开发者工具使用

    新建项目选择小程序项目,选择代码存放的硬盘路径,填入刚刚申请到的小程序的 AppID,给你的项目起一个好听的名字,最后,勾选 "创建 QuickStart 项目" (注意: 你要选 ...

  9. 微信小程序 开发 微信开发者工具 快捷键

    微信小程序已经跑起来了.快捷键设置找了好久没找到,完全凭感觉.图贴出来.大家看看. 我现在用的是0.10.101100的版本,后续版本更新快捷键也应该不会有什么变化. 现在貌似不能修改.如果有同学找到 ...

随机推荐

  1. CS academy Growing Trees【模板】DP求树的直径

    [题意概述] 给出一棵树,树上的边有两个值a和b,你可以在[0,limit]范围内选择一个整数delta,树上的边的权值为a+b*delta,现在问当delta为多少的时候树的直径最小.最小直径是多少 ...

  2. [第一波模拟\day3\T2]{独立集}(bubble.cpp)

    [问题描述] 有一天,一个名叫顺旺基的程序员从石头里诞生了.又有一天,他学会了冒泡排序和独立集.在一个图里,独立集就是一个点集,满足任意两个点之间没有边.于是他就想把这两个东西结合在一起.众所周知,独 ...

  3. 【Codeforces 1006D】Two Strings Swaps

    [链接] 我是链接,点我呀:) [题意] 题意 [题解] 注意只能改变a不能改变b 然后只要让a[i],b[i],a[n-i-1],b[n-i-1]这4个字符能凑成两对.全都一样就可以了 分类讨论下就 ...

  4. Backspace doesn't delete inner html tags of a contenteditable DIV in Firefox

    https://bugzilla.mozilla.org/show_bug.cgi?id=439808 backspace键 在ff下不能使用  div contenteditable=true时

  5. 谁才是最强战舰!-From 南京理工大学第八届程序设计大赛(校外镜像),博弈~~

    谁才是最强战舰! Time Limit: 1000MS Memory Limit: 65536KB Description 依阿华来到镇守府的第一件事情,就是找大和solo!然而这并不是什么好消息,说 ...

  6. 【带权并查集】HDU 3047 Zjnu Stadium

    http://acm.hdu.edu.cn/showproblem.php?pid=3047 [题意] http://blog.csdn.net/hj1107402232/article/detail ...

  7. SSH日志位置

    # Redhat or Fedora Core: /var/log/secure # Mandrake, FreeBSD or OpenBSD: /var/log/auth.log # SuSE: / ...

  8. CERC 2014 (动态树+主席树)

    CERC 2014 Pork barrel Problem : n个点m条边有边权的无向图,有q个询问,每次询问权值在[L,R]内的边组成的最小生成树的权值和,强制在线. n <= 1000, ...

  9. vs2010 静态使用 opencv 2.46 库

    下载opencv2.46的库,假设解压到OpenCV246,设置如下: 在工程的c++的include目录下添加:OpenCV246\opencv\build\include 在工程的c++的lib目 ...

  10. iOS - 设置系统类似的方法弃用警告的方式

    在开发过程中,调用系统方法时,经常可以看xCode 提示 该方法已弃用,如下图: 觉得特别炫,查一下资料,如果自己也想实现如下的效果,只需要采用系统的如下几个关键字加在方法名后面就可以了: NS_DE ...