Java获取微信公众号新增用户事件

一、新建项目工程

  1. 新建一个spring项目


    image
  1. 填写 Group 和 Artifact 信息


    image
  2. 这步可以直接跳过,后面再按需导入


    image
  1. 选择工程地址


    在这里插入图片描述

二、配置

pom.xml

<dependencies>
    <!-- spring相关包 -->
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter</artifactId>
    </dependency>
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-web</artifactId>
    </dependency>
    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-test</artifactId>
        <scope>test</scope>
    </dependency>
    <!-- http请求 -->
    <dependency>
        <groupId>org.apache.httpcomponents</groupId>
        <artifactId>httpclient</artifactId>
        <version>4.5.6</version>
    </dependency>
    <!-- 工具包 -->
    <dependency>
        <groupId>cn.hutool</groupId>
        <artifactId>hutool-all</artifactId>
        <version>5.5.8</version>
    </dependency>
    <!-- xml解析工具包 -->
    <dependency>
        <groupId>dom4j</groupId>
        <artifactId>dom4j</artifactId>
        <version>1.6.1</version>
    </dependency>
</dependencies>

application.yml

spring:
  application:
    name: wechat

server:
  port: 8900

appid: # 微信开发者appid
secret: # 微信开发者appsecret
token: # 服务器校验token

三、相关类

在这里插入图片描述

AccessToken / AccessTokenInfo

/**
 * AccessToken 实体类
 * @author unidentifiable
 * @date 2021-02-28
 */
public class AccessToken {
    /**
     * 获取到的凭证
     */
    private String tokenName;
    /**
     * 凭证有效时间  单位:秒
     */
    private int expireSecond;

    /**
     * get/set ...
     */
}

/**
 * AccessToken 实体类
 * @author unidentifiable
 * @date 2021-02-28
 */
public class AccessTokenInfo {
    /**
     * accessToken:像微信端发起请求需携带该accessToken
     */
    public static AccessToken accessToken = null;
}

MessageUtil

/**
 * 信息工具类
 * @author unidentifiable
 * @date 2021-02-28
 */
public class MessageUtil {
    /**
     * 解析微信发来的请求(XML)
     *
     * @param request 请求
     * @return map
     * @throws Exception 异常
     */
    public static Map<String, String> parseXml(HttpServletRequest request) {
        // 将解析结果存储在HashMap中
        Map<String, String> map = new HashMap<>(16);
        // 从request中取得输入流
        try (InputStream inputStream = request.getInputStream()) {
            System.out.println("获取输入流");
            // 读取输入流
            SAXReader reader = new SAXReader();
            Document document = reader.read(inputStream);
            // 得到xml根元素
            Element root = document.getRootElement();
            // 得到根元素的所有子节点
            List<Element> elementList = root.elements();

            // 遍历所有子节点
            for (Element e : elementList) {
                System.out.println(e.getName() + " | " + e.getText());
                map.put(e.getName(), e.getText());
            }
        } catch (Exception e) {
            e.printStackTrace();
        }

        return map;
    }

    /**
     * 获取用户详情
     * @param openId 微信公众号用户ID
     * @return 用户详情
     */
    public static String getUserInfo(String openId) {
        // 拼接 url,发起 http.get 请求
        String url = "https://api.weixin.qq.com/cgi-bin/user/info?access_token=" + AccessTokenInfo.accessToken.getTokenName() + "&openid=" + openId + "&lang=zh_CN";
        return HttpUtil.get(url);
    }
}

AccessTokenConfig

/**
 * AccessToken 配置类
 *
 * @author unidentifiable
 * @date 2021-02-28
 */
public class AccessTokenConfig {

    static {
        Properties prop = new Properties();
        String appid = null;
        String secret = null;
        try {
            // 读取 application.yml 获取微信开发者 appid 和 appsecret
            InputStream in = MessageUtil.class.getClassLoader().getResourceAsStream("application.yml");
            prop.load(in);
            appid = prop.getProperty("appid");
            secret = prop.getProperty("secret");
        } catch (Exception e) {
            e.printStackTrace();
        }

        // 接口地址为https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid=APPID&secret=APPSECRET,其中grant_type固定写为client_credential即可。
        String url = String.format("https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid=%s&secret=%s", appid, secret);
        // 此请求为https的get请求,返回的数据格式为{"access_token":"ACCESS_TOKEN","expires_in":7200}
        String result = HttpUtil.get(url);

        System.out.println("获取到的access_token=" + result);

        //使用FastJson将Json字符串解析成Json对象
        JSONObject json = new JSONObject(result);
        AccessToken token = new AccessToken();
        token.setTokenName(json.get("access_token", String.class));
        token.setExpireSecond(json.get("expires_in", Integer.class));

        // 开启一个线程,循环更新 AccessToken,该值有效期 2小时,需手动刷新
        new Thread(() -> {
            while (true) {
                try {
                    // 获取accessToken
                    AccessTokenInfo.accessToken = token;
                    // 获取成功
                    if (AccessTokenInfo.accessToken != null) {
                        // 获取到access_token 休眠7000秒,大约2个小时左右
                        Thread.sleep(7000 * 1000);
                    } else {
                        // 获取的access_token为空 休眠3秒
                        Thread.sleep(3000);
                    }
                } catch (Exception e) {
                    System.out.println("发生异常:" + e.getMessage());
                    e.printStackTrace();
                    try {
                        // 发生异常休眠1秒
                        Thread.sleep(1000);
                    } catch (Exception e1) {
                        e.printStackTrace();
                    }
                }
            }
        }).start();
    }
}

SubscribeService

package com.unidentifiable.wechat.service;

import com.unidentifiable.wechat.util.MessageUtil;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Service;

import javax.servlet.http.HttpServletRequest;
import java.security.MessageDigest;
import java.security.NoSuchAlgorithmException;
import java.util.Arrays;
import java.util.Map;

/**
 * Service
 * @author unidentifiable
 * @date 2021-02-28
 */
@Service("subscribeService")
public class SubscribeService {
    @Value("${subscriptionKey}")
    public String subscriptionKey;
    @Value("${uriBase}")
    public String uriBase;
    @Value("${token}")
    public String token;

    /**
     * 公众号服务器校验
     *
     * @param req 请求
     * @return 校验结果
     */
    public String verification(HttpServletRequest req) {
        // 接收微信服务器发送请求时传递过来的参数
        String signature = req.getParameter("signature");
        String timestamp = req.getParameter("timestamp");
        String nonce = req.getParameter("nonce");
        String echostr = req.getParameter("echostr");

        // 将token、timestamp、nonce三个参数进行字典序排序,并拼接为一个字符串
        String sortStr = sort(token, timestamp, nonce);
        // 字符串进行shal加密
        String mySignature = shal(sortStr);
        // 校验微信服务器传递过来的签名 和  加密后的字符串是否一致, 若一致则签名通过
        if (!"".equals(signature) && !"".equals(mySignature) && signature.equals(mySignature)) {
            System.out.println("-----签名校验通过-----");
            return echostr;
        } else {
            System.out.println("-----校验签名失败-----");
            return "";
        }
    }

    /**
     * 参数排序
     *
     * @param token     自定义token
     * @param timestamp timestamp
     * @param nonce     nonce
     * @return 排序后拼接字符串
     */
    public String sort(String token, String timestamp, String nonce) {
        String[] strArray = {token, timestamp, nonce};
        Arrays.sort(strArray);
        StringBuilder sb = new StringBuilder();
        for (String str : strArray) {
            sb.append(str);
        }
        return sb.toString();
    }

    /**
     * 字符串进行shal加密
     *
     * @param str 字符串
     * @return 密文
     */
    public String shal(String str) {
        try {
            MessageDigest digest = MessageDigest.getInstance("SHA-1");
            digest.update(str.getBytes());
            byte messageDigest[] = digest.digest();

            StringBuilder hexString = new StringBuilder();
            // 字节数组转换为 十六进制 数
            for (int i = 0; i < messageDigest.length; i++) {
                String shaHex = Integer.toHexString(messageDigest[i] & 0xFF);
                if (shaHex.length() < 2) {
                    hexString.append(0);
                }
                hexString.append(shaHex);
            }
            return hexString.toString();

        } catch (NoSuchAlgorithmException e) {
            e.printStackTrace();
        }
        return "";
    }

    /**
     * 关注请求
     *
     * @param req 请求
     * @return 关注事件结果
     */
    public String subscribeEvent(HttpServletRequest req) {
        String result = null;
        Map<String, String> map = null;

        try {
            /*
                解析请求,得到相关信息
                FromUserName| openid 用户ID
                CreateTime|1614499703 时间
                MsgType|event 消息类型
                Event|subscribe 事件类型
                EventKey|
             */
            map = MessageUtil.parseXml(req);
        } catch (Exception e) {
            e.printStackTrace();
        }

        // 新开一个线程防止当前现在延时导致微信重复发起请求
        final Map<String, String> m = map;
        final String[] str = new String[1];
        new Thread(() -> {
            // 获取判断是否为关注事件
            if (null != m && "event".equals(m.get("MsgType")) && "subscribe".equals(m.get("Event"))) {
                str[0] = MessageUtil.getUserInfo(m.get("FromUserName"));
                System.out.println(str[0]);
                
                // 这里可以添加邮件通知等功能,也可以判断别的事件,做出对应操作
            }
        }).start();

        return result;
    }

}

SubscribeController

package com.unidentifiable.wechat.controller;

import com.unidentifiable.wechat.service.SubscribeService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PostMapping;

import javax.servlet.http.HttpServletRequest;

/**
 * Controller
 * @author unidentifiable
 * @date 2021-02-28
 */
public class SubscribeController {
    @Autowired
    private SubscribeService subscribeService;

    /**
     * 服务器校验
     * @param req 请求
     * @return 校验结果
     */
    @GetMapping("/")
    public String verification(HttpServletRequest req) {
        return subscribeService.verification(req);
    }

    /**
     * 消息事件
     * @param req 请求
     * @return 结果
     */
    @PostMapping("/")
    public boolean subscribeEvent(HttpServletRequest req) {
        String s = subscribeService.subscribeEvent(req);
        return null != s;
    }
}

四、内网穿透(有域名的这一步可以省略)

这里使用的是 natapp 这一款工具,下载好对应版本工具

https://natapp.cn/

image

新建一个免费的隧道,配置好需要映射的地址跟端口,然后复制 authtoken

image

image

修改客户端配置文件中的 authtoken(我这里是win版本,linux版本没有该文件,建议新增一份,不然后台启动可能会连接超时)

config.ini

#将本文件放置于natapp同级目录 程序将读取 [default] 段
#在命令行参数模式如 natapp -authtoken=xxx 等相同参数将会覆盖掉此配置
#命令行参数 -config= 可以指定任意config.ini文件
[default]
authtoken=                      #对应一条隧道的authtoken
clienttoken=                    #对应客户端的clienttoken,将会忽略authtoken,若无请留空,
log=none                        #log 日志文件,可指定本地文件, none=不做记录,stdout=直接屏幕输出 ,默认为none
loglevel=DEBUG                  #日志等级 DEBUG, INFO, WARNING, ERROR 默认为 DEBUG
http_proxy=                     #代理设置 如 http://10.123.10.10:3128 非代理上网用户请务必留空

配置完成直接双击运行即可,linux版后台启动可用 (./natapp &) 命令运行,注意是有括号的。出现如下界面即为成功。

image

五、微信公众平台接口测试账号

https://mp.weixin.qq.com/debug/cgi-bin/sandbox?t=sandbox/login

拿到 appID 跟 appsecret 填入 appliation.yml 配置文件中,启动项目,启动 natapp ,填入域名以及自定义的token 进行接口配置校验。


image

绑定完成服务器信息之后,就可以扫描下面的二维码进行测试了。

image

教程到这里就结束了,大家有想实现什么自定义功能的话可以查看微信官方文档,看里面有提供什么功能,然后自己再修改下对应的判断即可。


image

项目地址: https://gitee.com/unidentifiable/wechatpublicaccount

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 162,547评论 4 374
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 68,787评论 2 308
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 112,175评论 0 254
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,752评论 0 223
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 53,169评论 3 297
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 41,056评论 1 226
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 32,195评论 2 321
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,980评论 0 214
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,724评论 1 250
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,872评论 2 254
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,344评论 1 265
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,667评论 3 264
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,379评论 3 245
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,202评论 0 9
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,992评论 0 201
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 36,189评论 2 286
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,987评论 2 279

推荐阅读更多精彩内容