热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

java接入微信公众平台

一、注册公众号并认证二、准备一台服务器,必须用80端口三、定义一个接口,用于与微信服务器通信,配置开发者中心的服务器配置中的url1、服务器url下一个token,是自己定义的,但是要在你定义的这个接


一、注册公众号并认证

二、准备一台服务器,必须用80端口

三、定义一个接口,用于与微信服务器通信,配置开发者中心的服务器配置中的url

1、服务器url下一个token,是自己定义的,但是要在你定义的这个接口中也返回这个token,微信以此来验证这个token

2、微信服务器验证token时是使用get请求过来的,验证签名(接入指南中有参数、加密方式说明)后按要求返回

3、微信其他事件推送都是用该url,区别在于事件通知回调我们服务器都是post请求

4、微信公众平台中点提交,微信回调该url验证token,验证通过基本就ok了


四、上代码,我这里用springmvc写了个demo



package com.mx.controller;

import java.util.ArrayList;
import java.util.Collections;
import java.util.Comparator;
import java.util.List;

import javax.servlet.http.HttpServletRequest;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.ResponseBody;

import com.mx.coder.CiphertextUtil;
import com.mx.util.MyException;
import com.mx.util.StaticMessage;

@Controller
public class WeixinController extends BaseController {

private Logger logger = LoggerFactory.getLogger(WeixinController.class);

/**
* 微信回调
* @author xiangpeng
* @date 2015-12-21上午9:54:57
* @param request
* @return
* String
*/
@RequestMapping(value = "weiXinCallBack", method = {RequestMethod.GET, RequestMethod.POST})
public @ResponseBody String wxCallBack(HttpServletRequest request){
//是否为get请求
boolean methodIsGet = request.getMethod().toLowerCase().equals("get");
try {
if (methodIsGet){
String signature = request.getParameter("signature");//微信加密签名
String timestamp = request.getParameter("timestamp");//时间戳
String nOnce= request.getParameter("nonce");//随机数
String echostr = request.getParameter("echostr");//随机字符串(需回传)
//请求参数非空判断
if (super.isEmpty(signature) || super.isEmpty(timestamp)
|| super.isEmpty(nonce) || super.isEmpty(echostr)){
logger.info(StaticMessage.PARAM_NOT_ENOUGH);
return StaticMessage.PARAM_NOT_ENOUGH;
}
logger.info("======微信请求参数======");
logger.info("signature="+signature);
logger.info("timestamp="+timestamp);
logger.info("nOnce="+nonce);
logger.info("echostr="+echostr);
//校验签名
boolean signBool = checkSign(signature, timestamp, nonce);
if (signBool){
return echostr;
}
}else{

}
} catch (Exception e) {
MyException.printExceptionInfo(logger, e);
}
return null;
}

/**
* 校验签名
* @author xiangpeng
* @date 2015-12-21上午10:59:26
* @param signature
* @param timestamp
* @param nonce
* @return
* boolean
*/
private boolean checkSign(String signature, String timestamp, String nonce) throws Exception{
//生成签名
List signList = new ArrayList();
signList.add(StaticMessage.TOKEN);
signList.add(timestamp);
signList.add(nonce);
//1. 将token、timestamp、nonce三个参数进行字典序排序
Collections.sort(signList, new Comparator() {
@Override
public int compare(String o1, String o2) {
return o1.compareTo(o2);
}
});
//2. 将三个参数字符串拼接成一个字符串进行sha1加密
Object[] signArray = signList.toArray();
StringBuffer signSb = new StringBuffer();
signSb.append(signArray[0]);
signSb.append(signArray[1]);
signSb.append(signArray[2]);
String signStr = signSb.toString();
signStr = CiphertextUtil.passAlgorithmsCiphering(signStr, CiphertextUtil.SHA_1);
logger.info("signStr="+signStr);
//3. 开发者获得加密后的字符串可与signature对比,标识该请求来源于微信
if (!signature.equals(signStr)){
logger.info("======校验签名失败!======");
logger.info("signature="+signature);
logger.info("signStr="+signStr);
return false;
}
return true;
}
}
package com.mx.coder;import java.io.UnsupportedEncodingException;import java.security.MessageDigest;import java.security.NoSuchAlgorithmException;import java.util.ArrayList;import java.util.Collections;import java.util.List;import com.mx.util.StaticMessage;/** * MD5工具类 *  * @date 2014-12-26 上午10:02:25 *  * @author FCH *  */public class CiphertextUtil {public static final String SHA_1 = "SHA-1";public static final String SHA_256 = "SHA-256";private static final char[] CH_HEX = { '0', '1', '2', '3', '4', '5', '6','7', '8', '9', 'a', 'b', 'c', 'd', 'e', 'f' };/** * 64位MD5 *  * @param str * @return */public static String parseStrToMd5L64(String str) {String reStr = null;try {MessageDigest md5 = MessageDigest.getInstance("MD5");// reStr = base64en.encode(md5.digest(str.getBytes("UTF-8")));reStr = Base64Encoder.encode(md5.digest(str.getBytes("UTF-8")));} catch (NoSuchAlgorithmException e) {// TODO Auto-generated catch blocke.printStackTrace();} catch (UnsupportedEncodingException e) {// TODO Auto-generated catch blocke.printStackTrace();}return reStr;}/** * 32位小写MD5 *  * @param str * @return */public static String parseStrToMd5L32(String str) {String reStr = null;try {MessageDigest md5 = MessageDigest.getInstance("MD5");byte[] bytes = md5.digest(str.getBytes());StringBuffer stringBuffer = new StringBuffer();for (byte b : bytes) {int bt = b & 0xff;if (bt <16) {stringBuffer.append(0);}stringBuffer.append(Integer.toHexString(bt));}reStr = stringBuffer.toString();} catch (NoSuchAlgorithmException e) {// TODO Auto-generated catch blocke.printStackTrace();}return reStr;}/** * 32位大写MD5 *  * @param str * @return */public static String parseStrToMd5U32(String str) {String reStr = parseStrToMd5L32(str);if (reStr != null) {reStr = reStr.toUpperCase();}return reStr;}/** * 16位小写MD5 *  * @param str * @return */public static String parseStrToMd5L16(String str) {String reStr = parseStrToMd5L32(str);if (reStr != null) {reStr = reStr.substring(8, 24);}return reStr;}/** * 16位大写MD5 *  * @param str * @return */public static String parseStrToMd5U16(String str) {String reStr = parseStrToMd5U32(str);if (reStr != null) {reStr = reStr.substring(8, 24);}return reStr;}  /**     * 加密字符串     *      * @param sourceStr    需要加密目标字符串     * @param algorithmsName 算法名称(如:MD5,SHA-1,SHA-256)     * @return     */    public static String passAlgorithmsCiphering(String sourceStr,String algorithmsName){        String password = "";        MessageDigest md;        try {            md = MessageDigest.getInstance(algorithmsName);            // 使用指定byte[]更新摘要            md.update(sourceStr.getBytes());            // 完成计算,返回结果数组            byte[] b = md.digest();            password = byteArrayToHex(b);        } catch (NoSuchAlgorithmException e) {            e.printStackTrace();        }        return password;    }        /**     * 将字节数组转为十六进制字符串     *     * @param bytes     * @return 返回16进制字符串     */    private static String byteArrayToHex(byte[] bytes) {        // 一个字节占8位,一个十六进制字符占4位;十六进制字符数组的长度为字节数组长度的两倍        char[] chars = new char[bytes.length * 2];        int index = 0;        for (byte b : bytes) {            // 取字节的高4位            chars[index++] = CH_HEX[b >>> 4 & 0xf];            // 取字节的低4位            chars[index++] = CH_HEX[b & 0xf];        }        return new String(chars);    }public static void main(String[] args) {String str =passAlgorithmsCiphering("f", CiphertextUtil.SHA_1);System.out.println("加密后:" + str);}}





推荐阅读
  • 深入解析Java枚举及其高级特性
    本文详细介绍了Java枚举的概念、语法、使用规则和应用场景,并探讨了其在实际编程中的高级应用。所有相关内容已收录于GitHub仓库[JavaLearningmanual](https://github.com/Ziphtracks/JavaLearningmanual),欢迎Star并持续关注。 ... [详细]
  • 1.如何在运行状态查看源代码?查看函数的源代码,我们通常会使用IDE来完成。比如在PyCharm中,你可以Ctrl+鼠标点击进入函数的源代码。那如果没有IDE呢?当我们想使用一个函 ... [详细]
  • ASP.NET MVC中Area机制的实现与优化
    本文探讨了在ASP.NET MVC框架中,如何通过Area机制有效地组织和管理大规模应用程序的不同功能模块。通过合理的文件夹结构和命名规则,开发人员可以更高效地管理和扩展项目。 ... [详细]
  • Explore how Matterverse is redefining the metaverse experience, creating immersive and meaningful virtual environments that foster genuine connections and economic opportunities. ... [详细]
  • 技术分享:从动态网站提取站点密钥的解决方案
    本文探讨了如何从动态网站中提取站点密钥,特别是针对验证码(reCAPTCHA)的处理方法。通过结合Selenium和requests库,提供了详细的代码示例和优化建议。 ... [详细]
  • 本文介绍了Java并发库中的阻塞队列(BlockingQueue)及其典型应用场景。通过具体实例,展示了如何利用LinkedBlockingQueue实现线程间高效、安全的数据传递,并结合线程池和原子类优化性能。 ... [详细]
  • 毕业设计:基于机器学习与深度学习的垃圾邮件(短信)分类算法实现
    本文详细介绍了如何使用机器学习和深度学习技术对垃圾邮件和短信进行分类。内容涵盖从数据集介绍、预处理、特征提取到模型训练与评估的完整流程,并提供了具体的代码示例和实验结果。 ... [详细]
  • 深入解析 Spring Security 用户认证机制
    本文将详细介绍 Spring Security 中用户登录认证的核心流程,重点分析 AbstractAuthenticationProcessingFilter 和 AuthenticationManager 的工作原理。通过理解这些组件的实现,读者可以更好地掌握 Spring Security 的认证机制。 ... [详细]
  • 本文探讨了在 Vue 2.0 项目中使用 Axios 获取数据时可能出现的错误,并提供详细的解决方案和最佳实践。 ... [详细]
  • 本文作者分享了在阿里巴巴获得实习offer的经历,包括五轮面试的详细内容和经验总结。其中四轮为技术面试,一轮为HR面试,涵盖了大量的Java技术和项目实践经验。 ... [详细]
  • 本文详细介绍了优化DB2数据库性能的多种方法,涵盖统计信息更新、缓冲池调整、日志缓冲区配置、应用程序堆大小设置、排序堆参数调整、代理程序管理、锁机制优化、活动应用程序限制、页清除程序配置、I/O服务器数量设定以及编入组提交数调整等方面。通过这些技术手段,可以显著提升数据库的运行效率和响应速度。 ... [详细]
  • 本文介绍如何使用 Angular 6 的 HttpClient 模块来获取 HTTP 响应头,包括代码示例和常见问题的解决方案。 ... [详细]
  • 本文介绍了两种方法,用于检测 Android 设备是否开启了开发者模式。第一种方法通过检查 USB 调试模式的状态,第二种方法则直接判断开发者选项是否启用。这两种方法均提供了代码示例和详细解释。 ... [详细]
  • 在 Flutter 开发过程中,开发者经常会遇到 Widget 构造函数中的可选参数 Key。对于初学者来说,理解 Key 的作用和使用场景可能是一个挑战。本文将详细探讨 Key 的概念及其应用场景,并通过实例帮助你更好地掌握这一重要工具。 ... [详细]
  • Python 异步编程:ASGI 服务器与框架详解
    自 Python 3.5 引入 async/await 语法以来,异步编程迅速崛起,吸引了大量开发者的关注。本文将深入探讨 ASGI(异步服务器网关接口)及其在现代 Python Web 开发中的应用,介绍主流的 ASGI 服务器和框架。 ... [详细]
author-avatar
爱的甜蜜日记2010
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有