微信公众号发送模版消息 Java实现

微信公众号发送模版消息

背景:

当用户发布任务的时候,公众号会自动推送消息通知。例如我们都熟悉的场景:微信支付的时候,公众号会推送支付成功消息。
在这里插入图片描述

申请模版:

模版消息,顾名思义,就是有模版的消息,那么要模版干嘛呢?模版是从哪来呢?
发送消息需要有固定的格式,我们可以在微信公众号平台上配置模版。

微信公众号平台–>广告与服务–>模版消息–>我的模版
「我的模版」列表里的是已经申请的模版,如果里面的模版格式都不符合自己业务,可以到模版库里找,然后添加到「我的模版」。也可以按照自己的需求申请新的模版,一般第二个工作日会审核通过。

https://mp.weixin.qq.com/
在这里插入图片描述

打开模版详情,查看模版的格式,下图左边红框是消息最终展示的效果,
右边红框是需要传的参数。
在这里插入图片描述
有了模版之后,模版ID就是我们要放进代码里的,把模版ID复制出来。

发送模版消息接口文档:

消息模版准备好之后,暂时不要写代码奥,查看微信开发文档,看看发送模版都需要哪些参数。

微信开发文档–>基础消息能力–>模版消息接口–「发送模版消息」
https://developers.weixin.qq.com/doc/offiaccount/Message_Management/Template_Message_Interface.html
在这里插入图片描述

微信开发文档参数介绍

发送模版消息

http请求方式: POST 请求地址: https://api.weixin.qq.com/cgi-bin/message/template/send?access_token=ACCESS_TOKEN
这里是引用

注:
url和miniprogram都是非必填字段,若都不传则模板无跳转;若都传,会优先跳转至小程序。开发者可根据实际需要选择其中一种跳转方式即可。当用户的微信客户端版本不支持跳小程序时,将会跳转至url。

返回码说明:
在调用模板消息接口后,会返回JSON数据包。
返回JSON数据包示例如下
{
“errcode”:0,
“errmsg”:“ok”,
“msgid”:200228332
}

发送模版所需参数:
模版ID和openId是必须有的,剩下的就是和自己业务有关了。

上面的内容都搞定之后,就可以开始撸代码了

发送模版微信返回Dto

@Data
public class TemplateMsgResultDto extends ResultState {

    /**
     * 消息id(发送模板消息)
     */
    private String msgid;

}

发送模版微信返回状态

@Data
public class ResultState implements Serializable {

    /**
     * 状态码
     */
    private int errcode;

    /**
     * 信息
     */
    private String errmsg;

}

微信模版消息请求参数实体类

@Data
public class WxTemplateMsg {
    /**
     * 接收者openId
     */
    private String touser;
    /**
     * 模板ID
     */
    private String template_id;
    /**
     * 模板跳转链接
     */
    private String url;
    /**
     * 消息data
     */
    private TreeMap<String, TreeMap<String, String>> data;
    /**
     * 参数
     *
     * @param value 值
     * @param color 颜色
     * @return params
     */
    public static TreeMap<String, String> item(String value, String color) {
        TreeMap<String, String> params = new TreeMap<String, String>();
        params.put("value", value);
        params.put("color", color);
        return params;
    }
}

Java封装模版信息代码

 public TemplateMsgResultDto noticeTemplate(TemplateMsgVo templateMsgVo) {
        // 模版ID
        String templateId="XXX";
        TreeMap<String, TreeMap<String, String>> params = new TreeMap<>();
        //根据具体模板参数组装
        params.put("first", WxTemplateMsg.item("恭喜!您的需求已发布成功", "#000000"));
        params.put("keyword1", WxTemplateMsg.item(templateMsgVo.getTaskName(), "#000000"));
        params.put("keyword2", WxTemplateMsg.item("需求已发布", "#000000"));
        params.put("remark", WxTemplateMsg.item("请耐心等待审核", "#000000"));
        WxTemplateMsg wxTemplateMsg = new WxTemplateMsg();
        // 模版ID
        wxTemplateMsg.setTemplate_id(templateId);
        // openId
        wxTemplateMsg.setTouser(templateMsgVo.getOpenId());
        // 关键字赋值
        wxTemplateMsg.setData(params);
        String data = JsonUtils.ObjectToString(wxTemplateMsg);
        return handleSendMsgLog(data);
    }

发送模版代码

    private TemplateMsgResultDto handleSendMsgLog(String data) {
        TemplateMsgResultDto resultDto = new TemplateMsgResultDto();
        try {
            resultDto = sendTemplateMsg(data);
        } catch (Exception exception) {
            log.error("发送模版失败",  exception);
        }
        // TODO 可以记录一下发送记录的日志
        return resultDto;
    }


 public TemplateMsgResultDto sendTemplateMsg(String data) throws Exception {
        // 获取token
        String accessToken = getAccessToken();
        // 发送消息
        HttpResult httpResult = HttpUtils.stringPostJson(ConstantsPath.SEND_MESSAGE_TEMPLATE_URL + accessToken, data);
        return IMJsonUtils.getObject(httpResult.getBody(), TemplateMsgResultDto.class);
    }


 /**
     * 获取全局token
     */
    public String getAccessToken() {
        String key = ConstantsRedisKey.ADV_WX_ACCESS_TOKEN;
        // 从redis缓存中获取token
        if (redisCacheManager.get(key) != null) {
            return (String) redisCacheManager.get(key);
        }
        // 获取access_token
        String url = String.format(ConstantsPath.WX_ACCESS_TOKEN_URL, appid, secret);
        ResponseEntity<String> result = restTemplate.getForEntity(url, String.class);
        if (result.getStatusCode() == HttpStatus.OK) {
            JSONObject jsonObject = JSON.parseObject(result.getBody());
            String accessToken = jsonObject.getString("access_token");
            // Long expires_in = jsonObject.getLong("expires_in");
            redisCacheManager.set(key, accessToken, 1800);
            return accessToken;
        }
        return null;
    }

微信地址常量类

public class ConstantsPath {
    /**
     * 微信公众号获取全局token
     */
    public static final String WX_ACCESS_TOKEN_URL = "https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid=%s&secret=%s";
 
    /**
     * 微信发送模版消息
     */
    public static final String SEND_MESSAGE_TEMPLATE_URL = "https://api.weixin.qq.com/cgi-bin/message/template/send?access_token=";

}

Json工具类

package com.demo.advertiser.common.utils;

import com.fasterxml.jackson.databind.DeserializationFeature;
import com.fasterxml.jackson.databind.JavaType;
import com.fasterxml.jackson.databind.ObjectMapper;
import lombok.extern.slf4j.Slf4j;
import org.apache.commons.lang3.StringUtils;
import org.springframework.cglib.beans.BeanMap;

import java.beans.BeanInfo;
import java.beans.IntrospectionException;
import java.beans.Introspector;
import java.beans.PropertyDescriptor;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.text.SimpleDateFormat;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

@Slf4j
public class JsonUtils {
    private static ObjectMapper json;
    static {
        json = new ObjectMapper();
        json.setDateFormat(new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"));
        json.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false);
    }

    /**
     * 序列化为JSON字符串
     */
    public static String ObjectToString(Object object) {
        try {
            return (json.writeValueAsString(object));
        } catch (Exception e) {
            log.error("序列化为JSON字符串出错",e);
        }
        return null;
    }

    public static <T> T getObject(String jsonString, Class<T> clazz) {
        if (StringUtils.isEmpty(jsonString))
            return null;
        try {
            return json.readValue(jsonString, clazz);
        } catch (Exception e) {
            log.error("将JSON字符串转化为Map出错",e);
            return null;
        }
    }

}

Http工具类

package com.demo.advertiser.common.utils;

import com.google.common.base.Splitter;
import com.demo.advertiser.common.utils.component.HttpResult;
import lombok.extern.slf4j.Slf4j;
import org.apache.commons.lang.StringEscapeUtils;
import org.apache.commons.lang3.StringUtils;
import org.apache.http.HttpEntity;
import org.apache.http.NameValuePair;
import org.apache.http.client.config.RequestConfig;
import org.apache.http.client.entity.UrlEncodedFormEntity;
import org.apache.http.client.methods.CloseableHttpResponse;
import org.apache.http.client.methods.HttpGet;
import org.apache.http.client.methods.HttpPost;
import org.apache.http.entity.ContentType;
import org.apache.http.entity.StringEntity;
import org.apache.http.entity.mime.MultipartEntityBuilder;
import org.apache.http.entity.mime.content.FileBody;
import org.apache.http.entity.mime.content.StringBody;
import org.apache.http.impl.client.CloseableHttpClient;
import org.apache.http.impl.client.HttpClientBuilder;
import org.apache.http.impl.client.LaxRedirectStrategy;
import org.apache.http.message.BasicNameValuePair;
import org.apache.http.util.EntityUtils;
import org.springframework.stereotype.Component;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.*;
import java.net.URI;
import java.net.URL;
import java.net.URLDecoder;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

@Component
@Slf4j
public class HttpUtils {
    private static String sourcePath;

    public static HttpResult stringPostJson(String path, String content) throws Exception{
        return stringPost(path, null, content, "utf-8", "utf-8", "application/json");
    }

    public static HttpResult stringPost(String path, Map<String,String> headerMap, String content, String contentencode, String encode, String contentType) throws Exception{
        StringEntity entity = new StringEntity(content, contentencode);
        entity.setContentType(contentType);
        return post(path, headerMap, entity, encode);
    }

    private static HttpResult post(String path, Map<String,String> headerMap, HttpEntity entity, String encode){
        HttpResult httpResult = new HttpResult();
        CloseableHttpClient httpClient = null;
        CloseableHttpResponse response = null;
        try{
            HttpPost httpPost = new HttpPost(getURI(path));
            LaxRedirectStrategy redirectStrategy = new LaxRedirectStrategy();
            httpClient = HttpClientBuilder.create().setRedirectStrategy(redirectStrategy).build();
            RequestConfig requestConfig = RequestConfig.custom()
                    .setSocketTimeout(120000)
                    .setConnectTimeout(120000)
                    .setConnectionRequestTimeout(120000)
                    .setCircularRedirectsAllowed(true)
                    .setRedirectsEnabled(true)
                    .setMaxRedirects(5)
                    .build();
            httpPost.setConfig(requestConfig);

            httpPost.setHeader("User-Agent", header);

            if(headerMap != null && headerMap.size() > 0){
                for(String name:headerMap.keySet()) {
                    httpPost.addHeader(name, headerMap.get(name));
                }
            }
            httpPost.setEntity(entity);
            response = httpClient.execute(httpPost);
            httpResult.setStatus(response.getStatusLine().getStatusCode());
            if(httpResult.getStatus() == 200){
                HttpEntity resEntity = response.getEntity();
                httpResult.setBody(EntityUtils.toString(resEntity, encode));
            }
        }catch(Exception ex){
            log.error("post请求出错", ex);
        }finally{
            try{
                if(response != null){
                    response.close();
                }
                if(httpClient != null){
                    httpClient.close();
                }
            }catch(Exception ex) {
                log.error("post请求关闭资源出错", ex);
            }
        }
        return httpResult;
    }
}
package com.demo.advertiser.common.utils.component;

public class HttpResult {
	private Integer status = 601;
	private String body = "";
	
	public Integer getStatus() {
		return status;
	}
	
	public void setStatus(Integer status) {
		this.status = status;
	}
	
	public String getBody() {
		return body;
	}
	
	public void setBody(String body) {
		this.body = body;
	}
}

  • 1
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 7
    评论
### 回答1: Java微信公众号定时发送消息模板通常分为两部分: 第一部分是实现Wechat API与Java后端的通信。Wechat API是微信提供的接口,可以让我们通过Java代码与微信公众号进行交互。具体来说,我们需要在Java中通过HTTP或HTTPS协议发送请求,获取微信公众号的access_token,再通过access_token来进行后续的操作,例如:获取用户的消息发送消息、创建菜单等等。Java中可以使用Apache HttpClient或OkHttp等库来发送HTTP请求,拿到接口返回的JSON数据。 第二部分是实现定时发送消息Java中可以使用Quartz框架用于实现定时任务。Quartz是一个开源的作业调度框架,可以用于在指定时间执行任务。我们可以利用Quartz提供的API,创建一个定时任务,定期调用我们的后端Java代码发送微信公众号消息。在定时任务的实现中,我们需要考虑任务的频率、执行时间、失败策略等问题。 当以上两部分都实现以后,我们就可以在Java后端实现微信公众号定时发送消息的功能了。具体来说,我们需要将发送消息的代码放到定时任务中,指定要发送消息内容,以及要发送给的用户。然后启动定时任务即可,Java后端会自动按照设定的规则发送微信公众号消息。 值得注意的是,微信官方文档要求所有公众号对接的应用都必须是官方认证的,所以在实现这个功能之前,我们需要先将我们的应用向微信官方申请认证,获得相应的API调用权限。 ### 回答2: Java微信公众号定时发送消息模板是一种利用Java语言实现的微信公众平台的定时发送消息模板。该模板可以帮助微信公众号上的管理员在指定时间点自动发送指定的消息,从而提高工作效率和用户体验。 实现Java微信公众号定时发送消息模板的关键是使用了微信公众平台提供的接口。这些接口可以通过Java语言进行调用,从而实现微信公众号发送消息、设置菜单、自动回复、素材管理、用户管理等功能。在定时发送消息模板中,我们需要利用这些接口设置定时任务,并在指定时间点调用接口发送消息。 具体实现过程如下: 1. 首先需要在微信公众平台上申请开发者账号,并获取相应的开发者ID、开发者密码和Token。 2. 在Java中使用微信公众平台提供的SDK调用相关接口,实现发送消息、设置菜单、自动回复等功能。 3. 利用Java中的定时任务框架,如Quartz、Spring定时任务等,在指定的时间点调用相应的微信接口实现定时发送消息。 4. 利用Java中的模板功能,以标准化的格式组织需要发送消息内容,使消息具有可读性和易管理性。 需要注意的是,为了避免被微信公众平台封禁账号,我们需要遵循微信公众平台的相关规定,比如定时发送消息不能含有敏感词汇、不能频繁发送等。另外,定时发送的频次也需要控制在一定范围内,避免对用户造成困扰。 总之,Java微信公众号定时发送消息模板是一种非常实用的工具,可以帮助管理员提高工作效率和用户体验。但实现过程需要注意一些细节和规范,确保操作的合法性和安全性。 ### 回答3: Java微信公众号定时发送消息模板是指利用Java编程语言实现微信公众号定时发送消息功能,并且提供了消息模板方便开发者进行二次开发。这个模板主要包含以下几个方面: 一、微信公众号接入 首先,需要在微信公众号后台注册并获取开发者ID和开发者密钥,然后使用Java框架接入微信公众号的开放平台。接入之后,就可以使用微信公众号提供的API实现发送消息功能。 二、定时任务 接下来,需要编写Java代码实现定时任务功能。Java提供了Timer和TimerTask类可以很方便地实现定时任务功能。开发者可以选择设定每日,周,月等不同的定时任务。 三、发送消息 定时任务设置好后,需要编写Java代码实现发送消息的功能。这个功能可以通过调用微信公众号提供的API来实现消息发送可以以文本消息,图文消息,视频消息等不同的形式呈现。 四、编写消息模板 最后,为了让开发者方便使用,我们需要编写一个消息模板来进行二次开发。消息模板可以包含以下几个方面:微信公众号接入的配置信息、定时任务的配置信息、发送消息的配置信息,以及发送消息实现方法等。这个模板可以让开发者快速地实现微信公众号定时发送消息的功能。 通过这个模板,开发者可以很方便地搭建起微信公众号定时发送消息的系统,从而可以提高服务的质量和用户体验。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 7
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值