简单内网映射到公网方法--免费

我是一名Android API Player,最近公司需要做微信公众号二次开发,我跟着学学,公司后台.net。

我mac安装windows之后用vs感觉太差了,可能是我的mac要淘汰了吧。

所以我决定用java后台来跟着做。

仔细一想我没有服务器啊。

再仔细一想我没有公众号啊或者服务号也行啊,申请太麻烦还要提交证明还要花钱,用公司的怕给玩坏了。

不过这都不是问题,解决方法总比问题多。


下面一步一步来记录下这些问题的解决,可能文笔不好,有的点会漏掉,只能慢慢来优化了。


第一步首先我们来解决没有服务器的问题:

大概思路就是将我们的本地的服务器地址映射到公网上,这样外网就可以访问我们的电脑指定路径了。

思路确定之后开始找软件了,试了几个之后,最后选定Sunny-Ngrok(免费的版本就够用了,放心吧,我也是穷人)。

本来一开始看博客有人推荐了Ngrok,试了以后发现他服务器在国外,映射的网址访问慢的不行。

下面是Sunny-Ngrok的官网:

https://www.ngrok.cc/

首先主页下拉下载你电脑对应的版本:



之后你要注册一个账号,登陆。(这个网站我发现了几个Bug,等我给他们反馈一下)

登陆之后文档里有一篇《隧道开通》的,我就是照着弄的:

http://www.sunnyos.com/article-show-67.html

写到这里忘了说了,之后我们会用到Tomcat,不会配置的同学可以参考下面这篇漏文:

http://blog.csdn.net/geanwen/article/details/78410595

都是基本操作,都坐好。


上面继续Sunny-Ngrok网站,登陆之后就可以到了具体操作界面了:

右侧有个开通隧道,点进去新建一个隧道:



选择免费的就可以:



新建填一些配置:



这些上面贴出的文档里应该都有,跟着走就可以,不过这里微信开发需要80端口需要注意一下。

这里填的除了前置域名固定的,其他的都可以后来修改。

创建之后来到了隧道管理界面,你创建的条目里有一个隧道id,

这时候回到你上面下载的文件,通过终端进入文件所在的文件夹,输入下面的命令:

  1. ./sunny clientid 隧道id
之后你会看到:



其中:http://aool.ngrok.cc就是你本地路径映射后的结果。

web界面127.0.0.1:4040就是web界面的展示,到时候调试请求出问题了可以通过这里查看详细的错误或日志。

到这里,没有服务器的问题解决了。


接下来先看看Java后台的代码,很简单,按照微信官方文档需要验证。

代码也是我看文章找的:

package servlet;

import bean.TextMessage;
import util.CheckUtil;
import util.MessageUtils;

import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.PrintWriter;
import java.util.Date;
import java.util.Map;

public class WechatServlet extends HttpServlet{

    /**
     * 接收微信服务器发送的4个参数并返回echostr
     */
    public void doGet(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {

        // 接收微信服务器以Get请求发送的4个参数
        String signature = request.getParameter("signature");
        String timestamp = request.getParameter("timestamp");
        String nonce = request.getParameter("nonce");
        String echostr = request.getParameter("echostr");

        PrintWriter out = response.getWriter();
        if (CheckUtil.checkSignature(signature, timestamp, nonce)) {
            out.print(echostr);        // 校验通过,原样返回echostr参数内容
        }
    }

    /**
     * 接收并处理微信客户端发送的请求
     */
    public void doPost(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {

        request.setCharacterEncoding("UTF-8");
        response.setCharacterEncoding("UTF-8");
        response.setContentType("text/xml;charset=utf-8");
        PrintWriter out = response.getWriter();
        Map<String, String> map = MessageUtils.xmlToMap(request);
        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)) {                // 对文本消息进行处理
            TextMessage text = new TextMessage();
            text.setFromUserName(toUserName);         // 发送和回复是反向的
            text.setTouserName(fromUserName);
            text.setMsgType("text");
            text.setCreateTime(String.valueOf(new Date().getTime()));
            text.setContent("你发送的消息是:" + content);
            message = MessageUtils.textMessageToXML(text);
            System.out.println(message);
        }
        out.print(message);                            // 将回应发送给微信服务器
    }
}

里面的工具类CheckUtils:

package util;

import java.security.MessageDigest;
import java.util.Arrays;

public class CheckUtil {

    private static final String token = "geanwen";

    public static boolean checkSignature(String signature, String timestamps, String nonce){

        String[] arr = new String[]{token, timestamps, nonce};

        // 排序
        Arrays.sort(arr);

        // 生成字符串
        StringBuilder content = new StringBuilder();
        for (int i = 0; i < arr.length; i++) {
            content.append(arr[i]);
        }

        // sha1加密
        String temp = encode(content.toString());

        return temp.equals(signature); // 与微信传递过来的签名进行比较
    }


    private static final char[] HEX_DIGITS = {'0', '1', '2', '3', '4', '5',
            '6', '7', '8', '9', 'a', 'b', 'c', 'd', 'e', 'f'};

    /**
     * Takes the raw bytes from the digest and formats them correct.
     *
     * @param bytes the raw bytes from the digest.
     * @return the formatted bytes.
     */
    private static String getFormattedText(byte[] bytes) {
        int len = bytes.length;
        StringBuilder buf = new StringBuilder(len * 2);
        // 把密文转换成十六进制的字符串形式
        for (int j = 0; j < len; j++) {
            buf.append(HEX_DIGITS[(bytes[j] >> 4) & 0x0f]);
            buf.append(HEX_DIGITS[bytes[j] & 0x0f]);
        }
        return buf.toString();
    }

    public static String encode(String str) {
        if (str == null) {
            return null;
        }
        try {
            MessageDigest messageDigest = MessageDigest.getInstance("SHA1");
            messageDigest.update(str.getBytes());
            return getFormattedText(messageDigest.digest());
        } catch (Exception e) {
            throw new RuntimeException(e);
        }
    }

}

MessageUtils:

package util;

import bean.TextMessage;
import com.thoughtworks.xstream.XStream;
import org.dom4j.Document;
import org.dom4j.DocumentException;
import org.dom4j.Element;
import org.dom4j.io.SAXReader;

import javax.servlet.http.HttpServletRequest;
import java.io.IOException;
import java.io.InputStream;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

public class MessageUtils {

    /**
     * xml转为map集合
     */
    public static Map<String, String> xmlToMap(HttpServletRequest request){
        Map<String, String> map = new HashMap<>();
        SAXReader reader = new SAXReader();

        try {
            // 从request中获取输入流
            InputStream ins = request.getInputStream();
            Document doc = reader.read(ins);

            // 获取xml中的跟元素
            Element root = doc.getRootElement();

            // 获取跟元素所有节点放到list中
            List<Element> list = root.elements();

            // 遍历
            for (Element e : list){
                map.put(e.getName(), e.getText());
            }

            ins.close();
            return map;
        } catch (IOException e) {
            e.printStackTrace();
            return null;
        } catch (DocumentException e) {
            e.printStackTrace();
            return null;
        }
    }

    /**
     * 将文本消息对象转换成XML
     */
    public static String textMessageToXML(TextMessage textMessage){

        XStream xstream = new XStream();              // 使用XStream将实体类的实例转换成xml格式
        xstream.alias("xml", textMessage.getClass()); // 将xml的默认根节点替换成“xml”
        return xstream.toXML(textMessage);

    }

}

按照微信要求的实体类TextMessage:

package bean;

public class TextMessage {

    private String TouserName;
    private String FromUserName;
    private String CreateTime;
    private String MsgType;
    private String Content;
    private String 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 String getCreateTime() {
        return CreateTime;
    }

    public void setCreateTime(String createTime) {
        CreateTime = createTime;
    }

    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 String getMsgId() {
        return MsgId;
    }

    public void setMsgId(String msgId) {
        MsgId = msgId;
    }
}

Web.xml中添加配置Servlet:

<servlet>
        <servlet-name>wechatServlet</servlet-name>
        <servlet-class>servlet.WechatServlet</servlet-class>
    </servlet>
    
    <servlet-mapping>
        <servlet-name>wechatServlet</servlet-name>
        <url-pattern>/wx.do</url-pattern>
    </servlet-mapping>


接下来配置Tomcat运行:


服务启动,这时候回去看看Sunny Ngrok的隧道的属性是否正确,没问题了就可以进入下一步:

微信测试号


一般我们的公众号开发都是在公众号已经开始运营的时候,贸然直接与后台连接可能会影响关注的粉丝们在公众号正常的使用;

即使我们没有多余的测试服务号,没有多余的测试的公众号。也可以


下面一步一步来


微信提供给我们开发者测试号的功能,具体是什么意思呢,就是我们申请一个测试账号,进入测试管理系统,就可以使用差不多公众号所有的功能;

测试账号网址:

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

只需要点击登陆扫码就好了:



登陆之后就可以进行配置:

配置URL--刚才我们写的微信get接口。

Token--我们上面代码中Token设置的是geanwen,所以这里也要一样。


我们将我们上面代码的服务启动,使用微信在线测试接口:

点击打开链接

输入对应参数,检查:



好了,这里潦草结尾,感觉写的太长了,后续在另起一张吧。

欢迎一起讨论,主要是记录一下大概思路,详细内容都可以针对去查。





猜你喜欢

转载自blog.csdn.net/geanwen/article/details/78733470