微信開發入門(一)

来源:http://www.cnblogs.com/xiangkejin/archive/2016/11/20/6083631.html
-Advertisement-
Play Games

一、開發前準備 1)微信公眾平臺賬號 訂閱號:個人版用戶,每天可以群發一條消息 服務號:企業版用戶,每天可以群發四條消息 2)外網映射工具—花生殼,讓自己本機tomcat伺服器能然外網訪問 與微信對接的url要具備以下條件: 在公網上能夠訪問 埠只支持80埠 下麵是用花生殼做映射: 訪問外網地址 ...


一、開發前準備

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是自定的。

這其中最重要的是對消息的處理,當普通微信用戶向公眾賬號發消息時,微信伺服器將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<String,String> 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<arr.length;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 < j; 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<String, String> xmlToMap(HttpServletRequest request ) throws DocumentException, IOException
        {
            Map<String,String> map = new HashMap<String, String>();
            //SAXreader用於解析xml
            SAXReader reader = new SAXReader();
            InputStream ins = request.getInputStream();
            //從輸入流中讀取xml文檔
            Document doc = reader.read(ins);
            //取得root節點
            Element root = doc.getRootElement();
            List<Element> 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;
        }
}

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 在公司無聊的時候看了前輩寫的python代碼,突然發現一個比較好玩的python表達式: 1 lambda x,y:x+y 咋一看,這個應該類似方法之類的,上網查了查,所以特此總結下 lambda:上代碼 冒號前邊的代表參數冒號後邊的代表表達式返回值類型跟參數有關 **返回值: **從這裡可以看得出 ...
  • 聚聚科技是一個剛創立的公司,很小很小,人很少,老闆感覺是個典型的北京小伙兒,戾氣很重,很有個性。筆試題倒是簡單: 1. echo(), print(), print_r()的區別? echo是PHP語言結構, print和print_r是函數。語言結構沒有返回值,函數可以有返回值(即便沒有用) 。 ...
  • JavaMail API使用javax.mail.Message類來表示一封郵件,Message類是一個抽象類,所以我們需要使用其子類javax.mail.internet.MimeMessage類來創建Message類的實例對象,如果我們創建的是一個簡單文本郵件,那麼MimeMessage類就可以 ...
  • 一直以來把資料庫的表轉換成Entity或DTO都是一件讓人頭痛的事情,既浪費時間又很繁瑣,用其他工具生成多少會有一些不盡人意的地方,於是就自己用Swing寫了一個通過資料庫的表生成JavaBean的工具,支持MySQL、Oracle、SQLServce、PostgreSQL,完美支持JPA註解,可... ...
  • Dr.com是城市熱點公司開發的寬頻計費系統,可以控制網路進行管理,認證,計費,限速……許多的高校與企業都有使用。 從接觸到drcom就很感興趣(原因想必大家都懂...) drcom登陸(認證)方式又有很多不同 這裡說下web端登陸的加密方式 ...
  • spider提供了多重安全保障機制,目前主要支持接入握手校驗,報文完整性校驗,報文加密,報文長度檢查四種機制。 接入認證 spider使用兩次握手校驗,其握手流程如下: 簽名AES加密的方式實現。 license信息存儲在classpath*:spider.dat文件中,在程式中固定。 報文完整性校 ...
  • 多線程的實現方法: 繼承Thread類 實現Runnable類 1. 繼承Thread類 繼承Thread類之後,需要覆蓋父類的 public void run() 方法,作為線程的主方法。 所有線程的執行一定是併發的,即:同一個時間段上會有多個線程交替執行。為了達到這樣的目的,絕對不能直接調用ru ...
  • ...
一周排行
    -Advertisement-
    Play Games
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...