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

微信开发入门(一)

一、开发前准备1)微信公众平台账号订阅号:个人版用户,每天可以群发一条消息服务号:企业版用户,每天可以群发四条消息2)外网映射工具—花生壳,让自己本机tomcat服务器能然外网访问与微信对接的url要

一、开发前准备

1)微信公众平台账号

订阅号:个人版用户,每天可以群发一条消息

服务号:企业版用户,每天可以群发四条消息

2)外网映射工具—花生壳,让自己本机tomcat服务器能然外网访问

与微信对接的url要具备以下条件:

  • 在公网上能够访问
  • 端口只支持80端口

下面是用花生壳做映射:

访问外网地址,可见就可以访问自己主机上的网站:

 

如果需要将自己项目部署到在线服务器上:

3)在线虚拟主机或服务器(SAE云引擎、BAE云引擎、阿里云引擎)

4)TortoiseSVN(SVN客户端软件)—将代码上传到服务器

开发模式和编辑模式是互斥的:

二、服务器验证

在eclipse+tomcat上开发

微信服务器的验证要求(详见微信平台的开发者文档):
微信通过通过get请求传进四个参数

signature 微信加密签名,signature结合了开发者填写的token参数和请求中的timestamp参数、nonce参数。
timestamp 时间戳
nonce 随机数
echostr 随机字符串

开发者通过检验signature对请求进行校验(下面有校验方式)。若确认此次GET请求来自微信服务器,请原样返回echostr参数内容,则接入生效,成为开发者成功,否则接入失败。加密/校验流程如下:
1)将token、timestamp、nonce三个参数进行字典序排序
2)将三个参数字符串拼接成一个字符串进行sha1加密
3)开发者获得加密后的字符串可与signature对比,标识该请求来源于微信

其中 token是自定的。

点击提交,这时微信会主动访问我们写的servlet,获得我们返回的参数中的echostr的内容即可认证通过。

认证过程原理分析:

 

 

这其中最重要的是对消息的处理,当普通微信用户向公众账号发消息时,微信服务器将POST消息的XML数据包到开发者填写的URL上。

首先将xml数据进行格式解析,然后对解析获取的数据进行处理,然后再将处理后的结果再以xml数据形式返回。

不同消息类型的推送XML数据包结构不同:

关于消息管理,详见文档。

注意导包

dom4j包 用来解析xml
xstream包 用来封装类

WeixinServlet.java

public class WeixinServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
String signature
=req.getParameter("signature");
String timestamp
=req.getParameter("timestamp");
String nonce
=req.getParameter("nonce");
String echostr
=req.getParameter("echostr");

PrintWriter out
= resp.getWriter();
if(CheckUtil.checkSignature(signature, timestamp, nonce)){
out.print(echostr);
}
}

@Override
protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
req.setCharacterEncoding(
"UTF-8");
resp.setCharacterEncoding(
"UTF-8");
PrintWriter out
= resp.getWriter();
try {
Map
map = MessageUtil.xmlToMap(req);
String toUserName
= map.get("ToUserName");
String fromUserName
= map.get("FromUserName");
String msgType
= map.get("MsgType");
String content
= map.get("Content");

String message
= null;
//文本类型
if("text".equals(msgType))
{
//按关键字进行判断
if("喜欢你".equals(content)){
message
=MessageUtil.initText(toUserName, fromUserName,MessageUtil.firstMenu());;
}
else if("很喜欢你".equals(content)){
message
=MessageUtil.initText(toUserName, fromUserName,MessageUtil.secondMenu());
}
else if("?".equals(content)||"?".equals(content)){
message
=MessageUtil.initText(toUserName, fromUserName, MessageUtil.menuText());
}
System.out.println(message);
}
//关注事件
else if(MessageUtil.MESSAGE_EVENT.equals(msgType)){
String eventType
= map.get("Event");
if(MessageUtil.EVENT_SUB.equals(eventType)){
String mycontent
= MessageUtil.menuText();
message
= MessageUtil.initText(toUserName, fromUserName, mycontent);

}
}
out.print(message);

}
catch (Exception e) {
e.printStackTrace();
}
finally{
out.close();
}
}
}

校验:

public class CheckUtil {

private static final String token="huaweiclub";
public static boolean checkSignature(String signature,String timestamp,String nonce){
String[] arr
=new String[]{token,timestamp,nonce};
//排序
Arrays.sort(arr);
//生成字符串
StringBuffer cOntent=new StringBuffer();
for(int i=0;i){
content.append(arr[i]);

}

//sha1加密
String temp=getSha1(content.toString());
return temp.equals(signature);
}

public static String getSha1(String str){
if(str==null||str.length()==0){
return null;
}
char hexDigits[] = {'0','1','2','3','4','5','6','7','8','9',
'a','b','c','d','e','f'};
try {
MessageDigest mdTemp
= MessageDigest.getInstance("SHA1");
mdTemp.update(str.getBytes(
"UTF-8"));

byte[] md = mdTemp.digest();
int j = md.length;
char buf[] = new char[j*2];
int k = 0;
for (int i = 0; i ) {
byte byte0 = md[i];
buf[k
++] = hexDigits[byte0 >>> 4 & 0xf];
buf[k
++] = hexDigits[byte0 & 0xf];
}
return new String(buf);
}
catch (Exception e) {
// TODO: handle exception
return null;
}
}

}

消息的格式转化:

/*
* xml和javabean的转换
*/
public class MessageUtil {

/*
* 定义不同的消息类型
*/
public static final String MESSAGE_TEXT = "text";
public static final String MESSAGE_IMAGE = "image";
public static final String MESSAGE_VOICE = "voice";
public static final String MESSAGE_VIDEO = "video";
public static final String MESSAGE_LINK = "link";
public static final String MESSAGE_LOCATION = "location";
public static final String MESSAGE_EVENT = "event";

public static final String EVENT_SUB = "subscribe";
public static final String EVENT_UNSUB = "unsubscribe";
public static final String EVENT_CLICK = "CLICK";
public static final String EVENT_VIEW = "VIEW";

/**
* xml转为map
*
@param request
*
@return
*
@throws DocumentException
*
@throws IOException
*/
public static Map xmlToMap(HttpServletRequest request ) throws DocumentException, IOException
{
Map
map = new HashMap();
//SAXreader用于解析xml
SAXReader reader = new SAXReader();
InputStream ins
= request.getInputStream();
//从输入流中读取xml文档
Document doc = reader.read(ins);
//取得root节点
Element root = doc.getRootElement();
List
list = root.elements();
for (Element e : list) {
map.put(e.getName(), e.getText());
}
ins.close();
return map;

}

/*
* 将文本消息转换为xml类型
* xstream能将bean对象序列化xml
*/
public static String textMessageToXml(TextMessage textMessage){
XStream xstream
= new XStream();
//将序列化中的类全量名称,用别名替换
xstream.alias("xml", textMessage.getClass());
return xstream.toXML(textMessage);
}

public static String initText(String toUserName, String fromUserName, String content){
TextMessage text
= new TextMessage();
text.setFromUserName(toUserName);
text.setToUserName(fromUserName);
text.setMsgType(MESSAGE_TEXT);
text.setCreateTime(
new Date().getTime());
text.setContent(content);
return textMessageToXml(text);

}

/*
* 根据关键字回复内容
*/
public static String menuText(){
StringBuffer sb
= new StringBuffer();
sb.append(
"谢谢关注");
return sb.toString();
}

public static String firstMenu(){
StringBuffer sb
= new StringBuffer();
sb.append(
"好感+1");
return sb.toString();
}

public static String secondMenu(){
StringBuffer sb
= new StringBuffer();
sb.append(
"好感+2");
return sb.toString();
}

}

xml文档的bean对象:

package com.xidian.bean;
/**
微信通信用的是XML,所以我们要新建一个javabean
*/
public class TextMessage {

private String ToUserName;
private String FromUserName;
private long CreateTime;
private String MsgType;
private String Content;
private long MsgId;
public String getToUserName() {
return ToUserName;
}
public void setToUserName(String toUserName) {
ToUserName
= toUserName;
}
public String getFromUserName() {
return FromUserName;
}
public void setFromUserName(String fromUserName) {
FromUserName
= fromUserName;
}
public long getCreateTime() {
return CreateTime;
}
public void setCreateTime(long l) {
CreateTime
= l;
}
public String getMsgType() {
return MsgType;
}
public void setMsgType(String msgType) {
MsgType
= msgType;
}
public String getContent() {
return Content;
}
public void setContent(String content) {
Content
= content;
}
public long getMsgId() {
return MsgId;
}
public void setMsgId(long msgId) {
MsgId
= msgId;
}
}

推荐阅读
  • Android实战——jsoup实现网络爬虫,糗事百科项目的起步
    本文介绍了Android实战中使用jsoup实现网络爬虫的方法,以糗事百科项目为例。对于初学者来说,数据源的缺乏是做项目的最大烦恼之一。本文讲述了如何使用网络爬虫获取数据,并以糗事百科作为练手项目。同时,提到了使用jsoup需要结合前端基础知识,以及如果学过JS的话可以更轻松地使用该框架。 ... [详细]
  • Hibernate延迟加载深入分析-集合属性的延迟加载策略
    本文深入分析了Hibernate延迟加载的机制,特别是集合属性的延迟加载策略。通过延迟加载,可以降低系统的内存开销,提高Hibernate的运行性能。对于集合属性,推荐使用延迟加载策略,即在系统需要使用集合属性时才从数据库装载关联的数据,避免一次加载所有集合属性导致性能下降。 ... [详细]
  • Iamtryingtomakeaclassthatwillreadatextfileofnamesintoanarray,thenreturnthatarra ... [详细]
  • 本文介绍了一款名为TimeSelector的Android日期时间选择器,采用了Material Design风格,可以在Android Studio中通过gradle添加依赖来使用,也可以在Eclipse中下载源码使用。文章详细介绍了TimeSelector的构造方法和参数说明,以及如何使用回调函数来处理选取时间后的操作。同时还提供了示例代码和可选的起始时间和结束时间设置。 ... [详细]
  • 开发笔记:spring boot项目打成war包部署到服务器的步骤与注意事项
    本文介绍了将spring boot项目打成war包并部署到服务器的步骤与注意事项。通过本文的学习,读者可以了解到如何将spring boot项目打包成war包,并成功地部署到服务器上。 ... [详细]
  • 本文整理了Java中org.gwtbootstrap3.client.ui.Icon.addDomHandler()方法的一些代码示例,展示了Icon.ad ... [详细]
  • 初探PLC 的ST 语言转换成C++ 的方法
    自动控制软件绕不开ST(StructureText)语言。它是IEC61131-3标准中唯一的一个高级语言。目前,大多数PLC产品支持ST ... [详细]
  • Java程序设计第4周学习总结及注释应用的开发笔记
    本文由编程笔记#小编为大家整理,主要介绍了201521123087《Java程序设计》第4周学习总结相关的知识,包括注释的应用和使用类的注释与方法的注释进行注释的方法,并在Eclipse中查看。摘要内容大约为150字,提供了一定的参考价值。 ... [详细]
  • Week04面向对象设计与继承学习总结及作业要求
    本文总结了Week04面向对象设计与继承的重要知识点,包括对象、类、封装性、静态属性、静态方法、重载、继承和多态等。同时,还介绍了私有构造函数在类外部无法被调用、static不能访问非静态属性以及该类实例可以共享类里的static属性等内容。此外,还提到了作业要求,包括讲述一个在网上商城购物或在班级博客进行学习的故事,并使用Markdown的加粗标记和语句块标记标注关键名词和动词。最后,还提到了参考资料中关于UML类图如何绘制的范例。 ... [详细]
  • JavaScript和HTML之间的交互是经由过程事宜完成的。事宜:文档或浏览器窗口中发作的一些特定的交互霎时。能够运用侦听器(或处置惩罚递次来预订事宜),以便事宜发作时实行相应的 ... [详细]
  • React基础篇一 - JSX语法扩展与使用
    本文介绍了React基础篇一中的JSX语法扩展与使用。JSX是一种JavaScript的语法扩展,用于描述React中的用户界面。文章详细介绍了在JSX中使用表达式的方法,并给出了一个示例代码。最后,提到了JSX在编译后会被转化为普通的JavaScript对象。 ... [详细]
  • 基于移动平台的会展导游系统APP设计与实现的技术介绍与需求分析
    本文介绍了基于移动平台的会展导游系统APP的设计与实现过程。首先,对会展经济和移动互联网的概念进行了简要介绍,并阐述了将会展引入移动互联网的意义。接着,对基础技术进行了介绍,包括百度云开发环境、安卓系统和近场通讯技术。然后,进行了用户需求分析和系统需求分析,并提出了系统界面运行流畅和第三方授权等需求。最后,对系统的概要设计进行了详细阐述,包括系统前端设计和交互与原型设计。本文对基于移动平台的会展导游系统APP的设计与实现提供了技术支持和需求分析。 ... [详细]
  • 本文总结了在编写JS代码时,不同浏览器间的兼容性差异,并提供了相应的解决方法。其中包括阻止默认事件的代码示例和猎取兄弟节点的函数。这些方法可以帮助开发者在不同浏览器上实现一致的功能。 ... [详细]
  • 项目运行环境配置及可行性分析
    本文介绍了项目运行环境配置的要求,包括Jdk1.8、Tomcat7.0、Mysql、HBuilderX等工具的使用。同时对项目的技术可行性、操作可行性、经济可行性、时间可行性和法律可行性进行了分析。通过对数据库的设计和功能模块的设计,确保系统的完整性和安全性。在系统登录、系统功能模块、管理员功能模块等方面进行了详细的介绍和展示。最后提供了JAVA毕设帮助、指导、源码分享和调试部署的服务。 ... [详细]
  • 概述H.323是由ITU制定的通信控制协议,用于在分组交换网中提供多媒体业务。呼叫控制是其中的重要组成部分,它可用来建立点到点的媒体会话和多点间媒体会议 ... [详细]
author-avatar
每天进步一点就好
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有