java 解决微信配置服务器时出现的token验证失败问题

配置微信服务器需要我们填写的有两个值,一个是URL,一个是Token,接下来我们就来看看这两个值该如何填写


1.URL

URL必须是一个线上地址,外网能访问的一个80端口地址。这个问题我们可以利用花生壳来映射一个80端口的外网地址

花生壳的使用配置:点击打开链接


2.Token的填写

这里的token需要以java代码中的token一致,你点击提交时,微信会根据你填写的URL地址进入你的代码进行验证,如何你没写java代码

就会出现Token验证失败

java代码如何,编写一个微信工具类(WeiXinUtils)和一个验证类(SignUtil)


WinXinUtils类代码如下


package com.utils.wx;


import com.mysql.jdbc.StringUtils;
import org.apache.log4j.Logger;
import org.apache.poi.util.StringUtil;

import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;

/**
 * @Author SPF
 * @Date 2017/6/6
 */
public class WeiXinUtils extends HttpServlet {
    private static final Logger logger = Logger.getLogger(WeiXinUtils.class);
    @Override
    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {
        //微信加密签名,signature结合了开发者填写的token参数和请求中的timestamp,nonce参数
        String signature = request.getParameter("signature");
        //时间戳
        String timestamp = request.getParameter("timestamp");
        //随机数
        String nonce = request.getParameter("nonce");
        //随机字符串
        String echostr = request.getParameter("echostr");

        if (SignUtil.checkSignature(signature, timestamp, nonce)) {
            logger.info("[signature: "+signature + "]<-->[timestamp: "+ timestamp+"]<-->[nonce: "+nonce+"]<-->[echostr: "+echostr+"]");
            response.getOutputStream().println(echostr);
        }
    }

    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        doGet(req,resp);
    }
}


SignUtil类代码如下


package com.utils.wx;

import java.security.MessageDigest;
import java.security.NoSuchAlgorithmException;
import java.util.Arrays;

/**
 * @Author SPF
 * @Date 2017/6/6
 */
public class SignUtil {
    private static String token = "WinXinSPF";

    /**
     * 校验签名
     * @param signature 签名
     * @param timestamp 时间戳
     * @param nonce 随机数
     * @return 布尔值
     */
    public static boolean checkSignature(String signature,String timestamp,String nonce){
        String checktext = null;
        if (null != signature) {
            //对ToKen,timestamp,nonce 按字典排序
            String[] paramArr = new String[]{token,timestamp,nonce};
            Arrays.sort(paramArr);
            //将排序后的结果拼成一个字符串
            String content = paramArr[0].concat(paramArr[1]).concat(paramArr[2]);

            try {
                MessageDigest md = MessageDigest.getInstance("SHA-1");
                //对接后的字符串进行sha1加密
                byte[] digest = md.digest(content.toString().getBytes());
                checktext = byteToStr(digest);
            } catch (NoSuchAlgorithmException e){
                e.printStackTrace();
            }
        }
        //将加密后的字符串与signature进行对比
        return checktext !=null ? checktext.equals(signature.toUpperCase()) : false;
    }

    /**
     * 将字节数组转化我16进制字符串
     * @param byteArrays 字符数组
     * @return 字符串
     */
    private static String byteToStr(byte[] byteArrays){
        String str = "";
        for (int i = 0; i < byteArrays.length; i++) {
            str += byteToHexStr(byteArrays[i]);
        }
        return str;
    }

    /**
     *  将字节转化为十六进制字符串
     * @param myByte 字节
     * @return 字符串
     */
    private static String byteToHexStr(byte myByte) {
        char[] Digit = {'0','1','2','3','4','5','6','7','8','9','A','B','C','D','E','F'};
        char[] tampArr = new char[2];
        tampArr[0] = Digit[(myByte >>> 4) & 0X0F];
        tampArr[1] = Digit[myByte & 0X0F];
        String str = new String(tampArr);
        return str;
    }

}


配置web.xml

<servlet>
  <servlet-name>weixin</servlet-name>
  <servlet-class>com.utils.wx.WeiXinUtils</servlet-class>
</servlet>
<servlet-mapping>
  <servlet-name>weixin</servlet-name>
  <url-pattern>/wx</url-pattern>
</servlet-mapping>

然后启动项目,点击提交,如图

java 解决微信配置服务器时出现的token验证失败问题

成功!

注意:URL路径=外网地址+微信工具类访问路径

   Token=java中的Token