实现步骤:
说明:
code是用户登录凭证(有效期五分钟)。
openid 是小程序用户唯一标识。
access_token是小程序全局唯一后台接口调用凭据。
实现效果:

具体实现:
1.申请消息模板
首先登入微信公众平台,选择自己的小程序,进入到小程序页面。
如图操作:





2.订阅对应模板消息
找到申请的模板,复制模板ID,小程序调用wx.requestSubscribeMessage接口订阅模板消息。
如图操作:

3.获取用户登入code
小程序调用wx.login接口获取用户code。
如图操作:

4.获取用户登入code
使用获取到的code调用后端用户授权接口获取用户openid。
官方接口文档:
auth.code2Session | 微信开放文档 (qq.com)
如图操作:
public UserAuthRes userAuth(String code) {
Map<String, Object> requestParams = new HashMap<>();
requestParams.put("appid", "wx4a6ad6d7e5e6xxxx"); //小程序appid
requestParams.put("secret", "eb05b4885d3fbe18d46b1822fca7xxxx"); //小程序secret
requestParams.put("js_code", code); //用户登入code
requestParams.put("grant_type", "authorization_code"); //授权类型这里填写authorization_code
UserAuthRes userAuthRes = restTemplate.getForObject("https://api.weixin.qq.com/sns/jscode2session?" +
"appid={appid}&secret={secret}&js_code={js_code}&grant_type={grant_type}", UserAuthRes.class, requestParams);
if (null != userAuthRes) {
log.info("授权用户信息:{}", userAuthRes);
//TODO 可以记录用户信息到mysql/redis(openid、session_key、unionid)备注:session_key 有效期为三天
return userAuthRes;
} else {
throw new RuntimeException("用户授权失败");
}
}
返回值结构:
public class UserAuthRes {
private String openid; //用户唯一标识openid
private String session_key; //用户会话密钥session_key
private String unionid; //用户在开放平台的唯一标识符,若当前小程序已绑定到微信开放平台帐号下会返回
private String errcode; //错误码
private String errmsg; //错误信息
}
5.获取access_token
后端调用接口传入appid、secret、grant_type获取access_token。
官方接口文档:
获取接口调用凭据 | 微信开放文档 (qq.com)
如图操作:
public String getUserAccessToken() {
Map<String, Object> requestParams = new HashMap<>();
requestParams.put("appid", "wx4a6ad6d7e5e6xxxx"); //小程序appid
requestParams.put("secret", "eb05b4885d3fbe18d46b1822fca7xxxx"); //小程序secret
requestParams.put("grant_type", "client_credential"); //授权类型这里填写client_credential
JSONObject result = restTemplate.getForObject("https://api.weixin.qq.com/cgi-bin/token?grant_type={grant_type}&appid={appid}&secret={secret}", JSONObject.class, requestParams);
if (null == result) {
throw new RuntimeException("获取access token失败");
}
String accessToken = result.get("access_token").toString();
if (StringUtils.isNoneBlank(accessToken)) {
log.info("获取access token成功:{}", accessToken);
return accessToken;
} else {
throw new RuntimeException("获取access token失败");
}
}
6.进行消息推送
打开要要推送的模板详情,查看模板占位符、后端调用接口传入touser、template_id、data,进行消息推送。注意请求参数data里的属性名一定要与模板占位符保持一致!
官方文档:
subscribeMessage.send | 微信开放文档 (qq.com)
如图操作: 

后端接口:
private void sendMessage() {
MessageTemplateEntity messageTemplateEntity = new MessageTemplateEntity();
String time = new SimpleDateFormat("yyyy-mm-dd HH:mm:ss").format(new Date());
messageTemplateEntity.setMessageData(new MessageValueEntity("项目名称"), new MessageValueEntity(time), new MessageValueEntity("这是描述"));
Map<String, Object> paramsMap = new HashMap<>();
paramsMap.put("touser", "wx4a6ad6d7e5e6xxxx"); //用户openid
paramsMap.put("template_id", "h7eKu5vOSbbObOxg9psqYZ-NiRyiiPi9wMuWqExxxx"); //推送消息模板id
paramsMap.put("data", messageTemplateEntity); //消息体:{{"thing1":"项目名称"},{"time2":"2022-08-23"},{"thing3":"这是描述"}}
HttpHeaders headers = new HttpHeaders(); //构建请求头
headers.setContentType(MediaType.APPLICATION_JSON); //设置内容类型为json
HttpEntity<Map<String, Object>> request = new HttpEntity<>(paramsMap, headers); //构建http请求实体
//发送请求路径拼接获取到的access_token
SendMessageVo sendMessageVo = restTemplate.postForObject("https://api.weixin.qq.com/cgi-bin/message/subscribe/send?access_token=" +
"60_mrPc2IpScFl8b8a9fhsPVw4nbMxxxx...", request, SendMessageVo.class);
if (null == sendMessageVo) {
throw new RuntimeException("推送消息失败");
}
if (sendMessageVo.getErrcode() != 0) {
log.error("推送消息失败,原因:{}", sendMessageVo.getErrmsg());
throw new RuntimeException("推送消息失败");
}
log.info("推送消息成功");
}
MessageTemplateEntity实体类结构:
@Data
public class MessageTemplateEntity {
private MessageValueEntity thing1;
private MessageValueEntity time2;
private MessageValueEntity thing3;
public void setMessageData(MessageValueEntity thing1, MessageValueEntity time2, MessageValueEntity thing3) {
this.thing1 = thing1;
this.time2 = time2;
this.thing3 = thing3;
}
}
MessageValueEntity实体类结构:
@Data
@AllArgsConstructor
@NoArgsConstructor
public class MessageValueEntity {
private String value;
}
注意事项:
(1)本篇消息推送教学为一次性订阅消息,也就是订阅一次可以推送一次消息,不订阅推送消息会失败,长期订阅消息如有需要,请去小程序官方申请长期消息模板。
(2)推送消息data属性值格式一定要严格按照官方给的格式,不然会报错,47003错误码说明有某个字段为null或者格式不正确。如果不清楚可以去官方文档查看对应格式。
官方文档:
subscribeMessage.send | 微信开放文档
关于我们:
创作不易,如有问题、疑难杂症,欢迎评论区交流病情(包治百病)。
我是rails的新手,想在form字段上应用验证。myviewsnew.html.erb.....模拟.rbclassSimulation{:in=>1..25,:message=>'Therowmustbebetween1and25'}end模拟Controller.rbclassSimulationsController我想检查模型类中row字段的整数范围,如果不在范围内则返回错误信息。我可以检查上面代码的范围,但无法返回错误消息提前致谢 最佳答案 关键是您使用的是模型表单,一种显示ActiveRecord模型实例属性的表单。c
我的工作要求我为某些测试自动生成电子邮件。我一直在四处寻找,但未能找到可以快速实现的合理解决方案。它需要在outlook而不是其他邮件服务器中,因为我们有一些奇怪的身份验证规则,我们需要保存草稿而不是仅仅发送邮件的选项。显然win32ole可以做到这一点,但我找不到任何相当简单的例子。 最佳答案 假设存储了Outlook凭据并且您设置为自动登录到Outlook,WIN32OLE可以很好地完成此操作:require'win32ole'outlook=WIN32OLE.new('Outlook.Application')message=
我正在使用Ruby,我正在与一个网络端点通信,该端点在发送消息本身之前需要格式化“header”。header中的第一个字段必须是消息长度,它被定义为网络字节顺序中的2二进制字节消息长度。比如我的消息长度是1024。如何将1024表示为二进制双字节? 最佳答案 Ruby(以及Perl和Python等)中字节整理的标准工具是pack和unpack。ruby的packisinArray.您的长度应该是两个字节长,并且按网络字节顺序排列,这听起来像是n格式说明符的工作:n|Integer|16-bitunsigned,network(bi
前言一般来说,前端根据后台返回code码展示对应内容只需要在前台判断code值展示对应的内容即可,但要是匹配的code码比较多或者多个页面用到时,为了便于后期维护,后台就会使用字典表让前端匹配,下面我将在微信小程序中通过wxs的方法实现这个操作。为什么要使用wxs?{{method(a,b)}}可以看到,上述代码是一个调用方法传值的操作,在vue中很常见,多用于数据之间的转换,但由于微信小程序诸多限制的原因,你并不能优雅的这样操作,可能有人会说,为什么不用if判断实现呢?但是if判断的局限性在于如果存在数据量过大时,大量重复性操作和if判断会让你的代码显得异常冗余。wxswxs相当于是一个独立
项目介绍随着我国经济迅速发展,人们对手机的需求越来越大,各种手机软件也都在被广泛应用,但是对于手机进行数据信息管理,对于手机的各种软件也是备受用户的喜爱小学生兴趣延时班预约小程序的设计与开发被用户普遍使用,为方便用户能够可以随时进行小学生兴趣延时班预约小程序的设计与开发的数据信息管理,特开发了小程序的设计与开发的管理系统。小学生兴趣延时班预约小程序的设计与开发的开发利用现有的成熟技术参考,以源代码为模板,分析功能调整与小学生兴趣延时班预约小程序的设计与开发的实际需求相结合,讨论了小学生兴趣延时班预约小程序的设计与开发的使用。开发环境开发说明:前端使用微信微信小程序开发工具:后端使用ssm:VU
@作者:SYFStrive @博客首页:HomePage📜:微信小程序📌:个人社区(欢迎大佬们加入)👉:社区链接🔗📌:觉得文章不错可以点点关注👉:专栏连接🔗💃:感谢支持,学累了可以先看小段由小胖给大家带来的街舞👉微信小程序(🔥)目录自定义组件-behaviors 1、什么是behaviors 2、behaviors的工作方式 3、创建behavior 4、导入并使用behavior 5、behavior中所有可用的节点 6、同名字段的覆盖和组合规则总结最后自定义组件-behaviors 1、什么是behaviorsbehaviors是小程序中,用于实现
如果我在模型中设置验证消息validates:name,:presence=>{:message=>'Thenamecantbeblank.'}我如何让该消息显示在闪光警报中,这是我迄今为止尝试过的方法defcreate@message=Message.new(params[:message])if@message.valid?ContactMailer.send_mail(@message).deliverredirect_to(root_path,:notice=>"Thanksforyourmessage,Iwillbeintouchsoon")elseflash[:error]
RSpec似乎按顺序匹配方法接收的消息。我不确定如何使以下代码工作:allow(a).toreceive(:f)expect(a).toreceive(:f).with(2)a.f(1)a.f(2)a.f(3)我问的原因是a.f的一些调用是由我的代码的上层控制的,所以我不能对这些方法调用添加期望。 最佳答案 RSpecspy是测试这种情况的一种方式。要监视一个方法,用allowstub,除了方法名称之外没有任何约束,调用该方法,然后expect确切的方法调用。例如:allow(a).toreceive(:f)a.f(2)a.f(1)
我有一个数组数组,想将元素附加到子数组。+=做我想做的,但我想了解为什么push不做。我期望的行为(并与+=一起工作):b=Array.new(3,[])b[0]+=["apple"]b[1]+=["orange"]b[2]+=["frog"]b=>[["苹果"],["橙子"],["Frog"]]通过推送,我将推送的元素附加到每个子数组(为什么?):a=Array.new(3,[])a[0].push("apple")a[1].push("orange")a[2].push("frog")a=>[[“苹果”、“橙子”、“Frog”]、[“苹果”、“橙子”、“Frog”]、[“苹果”、“
我以为它们存储在cookie中-但不,检查cookie没有任何结果。session也不存储它们。那么,我在哪里可以找到它们?我需要这个来直接设置它们(而不是通过flashhash)。 最佳答案 它们存储在inyoursessionstore.自rails2.0以来的默认设置是cookie存储,但请检查config/initializers/session_store.rb以检查您是否使用默认设置以外的东西。 关于ruby-on-rails-闪存消息存储在哪里?,我们在StackOverf