1. 程式人生 > >JSSDK-java配置--實現分享自定義功能

JSSDK-java配置--實現分享自定義功能

為了實現微信分享時自定義圖片和文字,耗費了不少心力,把JSSDK翻了個底朝天

首先我們來看一下微信開發給我們的樣例檔案:

import java.util.UUID;
import java.util.Map;
import java.util.HashMap;
import java.util.Formatter;
import java.security.MessageDigest;
import java.security.NoSuchAlgorithmException;
import java.io.UnsupportedEncodingException;  


class Sign {
    public static void main(String[] args) {
        String jsapi_ticket = "jsapi_ticket";


        // 注意 URL 一定要動態獲取,不能 hardcode
        String url = "http://example.com";
        Map<String, String> ret = sign(jsapi_ticket, url);
        for (Map.Entry entry : ret.entrySet()) {
            System.out.println(entry.getKey() + ", " + entry.getValue());
        }
    };


    public static Map<String, String> sign(String jsapi_ticket, String url) {
        Map<String, String> ret = new HashMap<String, String>();
        String nonce_str = create_nonce_str();
        String timestamp = create_timestamp();
        String string1;
        String signature = "";


        //注意這裡引數名必須全部小寫,且必須有序
        string1 = "jsapi_ticket=" + jsapi_ticket +
                  "&noncestr=" + nonce_str +
                  "×tamp=" + timestamp +
                  "&url=" + url;
        System.out.println(string1);


        try
        {
            MessageDigest crypt = MessageDigest.getInstance("SHA-1");
            crypt.reset();
            crypt.update(string1.getBytes("UTF-8"));
            signature = byteToHex(crypt.digest());
        }
        catch (NoSuchAlgorithmException e)
        {
            e.printStackTrace();
        }
        catch (UnsupportedEncodingException e)
        {
            e.printStackTrace();
        }


        ret.put("url", url);
        ret.put("jsapi_ticket", jsapi_ticket);
        ret.put("nonceStr", nonce_str);
        ret.put("timestamp", timestamp);
        ret.put("signature", signature);


        return ret;
    }


    private static String byteToHex(final byte[] hash) {
        Formatter formatter = new Formatter();
        for (byte b : hash)
        {
            formatter.format("%02x", b);
        }
        String result = formatter.toString();
        formatter.close();
        return result;
    }


    private static String create_nonce_str() {
        return UUID.randomUUID().toString();
    }


    private static String create_timestamp() {
        return Long.toString(System.currentTimeMillis() / 1000);
    }
}

且附帶了一個txt告訴我們“需要開發者傳入 jsapi_ticket 和 url”,但是坑來了:

1.用在自己程式裡時一定要刪除public static void main(String[] args) {}這個主函式(有點基礎的人都懂,這是小問題)

2.該例子返回的是一個map值,但是這個map值是缺少東西的,在map裡面加上:

    ret.put("appId", appId);

  //當然這個APPID是需要獲取的(每個人獲取方式不一樣,不予贅述),後面會附上我的所有程式碼

//appid屬性在微信公眾平臺就能看到,記錄在後臺上

3.jsapi_ticket 是需要輸入的,但是在樣例檔案中是沒有的,以下程式碼是獲取jsapi_ticket 的程式碼,隸屬於工具類。但是對於acess_token和appid和appsecret需要自己提供

public static String getJSApiTicket() throws FileNotFoundException, IOException{ 

       //獲取token
       String acess_token= AccessTokenUtil.getAccessToken();//token需要自己提供,不贅述具體方法
       String urlStr = //"http://gy7tgr.natappfree.cc//wxapi/weixinMall/index.jsp";
    		   "https://api.weixin.qq.com/cgi-bin/ticket/getticket?access_token="+acess_token+"&type=jsapi";  
       String backData=jsapi_ticketGetUtil.sendGet(urlStr, "utf-8", 10000);  
       String ticket = (String) JSONObject.fromObject(backData).get("ticket");  
       return  ticket;  
          
   }  
	/***
     * 獲取acess_token 
     * @return
     *//*
    public static String getAccessToken() throws FileNotFoundException, IOException{
	    String appid = Configure.getAppid();//此處需要自己提供,configure方法是程式裡方法,據個人情況而定
	    String appSecret = Configure.getAppSecrte();//此處需自己提供。
        String url ="https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid="+appid+"&secret="+appSecret+"";
        String backData=jsapi_ticketGetUtil.sendGet(url, "utf-8", 10000);
        String accessToken = (String) JSONObject.fromObject(backData).get("access_token");  
        return accessToken;
    }
   
   */
   /***
    * 模擬get請求
    * @param url
    * @param charset
    * @param timeout
    * @return
    */
    public static String sendGet(String url, String charset, int timeout)
     {
       String result = "";
       try
       {
         URL u = new URL(url);
         try
         {
           URLConnection conn = u.openConnection();
           conn.connect();
           conn.setConnectTimeout(timeout);
           BufferedReader in = new BufferedReader(new InputStreamReader(conn.getInputStream(), charset));
           String line="";
           while ((line = in.readLine()) != null)
           {
            
             result = result + line;
           }
           in.close();
         } catch (IOException e) {
           return result;
         }
       }
       catch (MalformedURLException e)
       {
         return result;
       }
      
       return result;
     }

上面的工具類提供了jsapi_ticket ,而appid也是已知的,所以可以直接搭建程式了

********************************************************************************************************************************************************************************************************************************************************

以下為相關片段程式:

1.獲取jsapi_ticket工具類

public class jsapi_ticketGetUtil {
	
	/***
     * 獲取jsapiTicket
     * @return
     */
   public static String getJSApiTicket() throws FileNotFoundException, IOException{ 
       //獲取token
       String acess_token= AccessTokenUtil.getAccessToken();
       String urlStr = //"http://gy7tgr.natappfree.cc//wxapi/weixinMall/index.jsp";
    		   "https://api.weixin.qq.com/cgi-bin/ticket/getticket?access_token="+acess_token+"&type=jsapi";  
       String backData=jsapi_ticketGetUtil.sendGet(urlStr, "utf-8", 10000);  
       String ticket = (String) JSONObject.fromObject(backData).get("ticket");  
       return  ticket;  
          
   }  
	/***
     * 獲取acess_token 
     * @return
     *//*
    public static String getAccessToken() throws FileNotFoundException, IOException{
	    String appid = Configure.getAppid();
	    String appSecret = Configure.getAppSecrte();
        String url ="https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid="+appid+"&secret="+appSecret+"";
        String backData=jsapi_ticketGetUtil.sendGet(url, "utf-8", 10000);
        String accessToken = (String) JSONObject.fromObject(backData).get("access_token");  
        return accessToken;
    }
   
   */
   /***
    * 模擬get請求
    * @param url
    * @param charset
    * @param timeout
    * @return
    */
    public static String sendGet(String url, String charset, int timeout)
     {
       String result = "";
       try
       {
         URL u = new URL(url);
         try
         {
           URLConnection conn = u.openConnection();
           conn.connect();
           conn.setConnectTimeout(timeout);
           BufferedReader in = new BufferedReader(new InputStreamReader(conn.getInputStream(), charset));
           String line="";
           while ((line = in.readLine()) != null)
           {
            
             result = result + line;
           }
           in.close();
         } catch (IOException e) {
           return result;
         }
       }
       catch (MalformedURLException e)
       {
         return result;
       }
      
       return result;
     }
}

2.返回一個map,裡面存放appid,signature等資訊

public class Sign {

    public static Map<String, String> sign(String jsapi_ticket, String url) throws FileNotFoundException, IOException {
        Map<String, String> ret = new HashMap<String, String>();
        String nonce_str = create_nonce_str();
        String timestamp = create_timestamp();
        String string1;
        String signature = "";
        

        //注意這裡引數名必須全部小寫,且必須有序
        string1 = "jsapi_ticket=" + jsapi_ticket +
                  "&noncestr=" + nonce_str +
                  "×tamp=" + timestamp +
                  "&url=" + url;
        System.out.println(string1);

        try
        {
            MessageDigest crypt = MessageDigest.getInstance("SHA-1");
            crypt.reset();
            crypt.update(string1.getBytes("UTF-8"));
            signature = byteToHex(crypt.digest());
        }
        catch (NoSuchAlgorithmException e)
        {
            e.printStackTrace();
        }
        catch (UnsupportedEncodingException e)
        {
            e.printStackTrace();
        }
        
        //自動獲取WxTokenUtil.properties中appid屬性,避免appid改動時重複更改
	    String appId = Configure.getAppid();
	    
	    
        ret.put("url", url);
      //注意這裡 要加上自己的appId
        ret.put("appId", appId);
        ret.put("jsapi_ticket", jsapi_ticket);
        ret.put("nonceStr", nonce_str);
        ret.put("timestamp", timestamp);
        ret.put("signature", signature);

        return ret;
    }

    private static String byteToHex(final byte[] hash) {
        Formatter formatter = new Formatter();
        for (byte b : hash)
        {
            formatter.format("%02x", b);
        }
        String result = formatter.toString();
        formatter.close();
        return result;
    }

    private static String create_nonce_str() {
        return UUID.randomUUID().toString();
    }

    private static String create_timestamp() {
        return Long.toString(System.currentTimeMillis() / 1000);
    }
    
    
}

3.前後臺互動的方法,前端傳回url路徑,後臺讀取後,把appid,signature等作為json物件傳回前端

@RequestMapping("/weixin/JSSDKHELP.hn")
	public String JSSDKHelp(HttpServletRequest request,
			HttpServletResponse response) throws FileNotFoundException, IOException {
		String url=request.getParameter("url");
		//獲取jsapi_ticket
		jsapi_ticketGetUtil ticket = new jsapi_ticketGetUtil();
		String jsapi_ticket = ticket.getJSApiTicket();
		Map<String, String> ret = com.org.wxapi.util.Sign.sign(jsapi_ticket, url);
		JSONObject jsonObject = JSONObject.fromObject(ret); 
		ResponseUtils.renderJson(response, jsonObject.toString());
		return null;
	}

4.前端,寫在一個.JS檔案裡,若要呼叫,只需在介面上加上

<script type="text/javascript" src="../weixinMall/js/JSSDK.js"></script>

JS檔案如下:

document.write("<script language=javascript src='http://res.wx.qq.com/open/js/jweixin-1.1.0.js'></script>");
$(document).ready(function() {  
    var currurl = decodeURIComponent(location.href.split('#')[0]);  
  	
    //ajax注入許可權驗證    
    $.ajax({  
    	type: 'POST',   //請求方式
        url : "/wxapi/weixin/JSSDKHELP.hn",
        dataType : 'json',  
        data : {"url": currurl},
        complete : function(XMLHttpRequest, textStatus) {},  
        error : function(XMLHttpRequest, textStatus, errorThrown) {  
            alert("發生錯誤:" + errorThrown+"XMLHttpRequeststatus:"+XMLHttpRequest.status+"XMLHttpRequest.readyState:"+XMLHttpRequest.readyState+"textStatus:"+textStatus);  
        },  
        success : function(res) { 
            var appId = res.appId;  
            var nonceStr = res.nonceStr;  
            var jsapi_ticket = res.jsapi_ticket;  
            var timestamp = res.timestamp;  
            var signature = res.signature;  
            //alert("appId:"+appId +"------- " +"nonceStr:"+ nonceStr +" -----"+"jsapi_ticket:"+jsapi_ticket+"------ "+"timestamp:"+timestamp+"------- "+"signature:"+signature);  
            wx.config({  
                debug : false, //開啟除錯模式,呼叫的所有api的返回值會在客戶端alert出來,若要檢視傳入的引數,可以在pc端開啟,引數資訊會通過log打出,僅在pc端時才會列印。    
                appId : appId, //必填,公眾號的唯一標識    
                timestamp : timestamp, // 必填,生成簽名的時間戳    
                nonceStr : nonceStr, //必填,生成簽名的隨機串    
                signature : signature, // 必填,簽名,見附錄1    
                jsApiList : [ 'onMenuShareAppMessage', 'onMenuShareTimeline'] //必填,需要使用的JS介面列表,所有JS介面列表 見附錄2     
            }); // end wx.config  
  
  
            wx.ready(function() {  
                // config資訊驗證後會執行ready方法,所有介面呼叫都必須在config介面獲得結果之後,config是一個客戶端的非同步操作,所以如果需要在頁面載入時就呼叫相關介面,則須把相關介面放在ready函式中呼叫來確保正確執行。對於使用者觸發時才呼叫的介面,則可以直接呼叫,不需要放在ready函式中。  
                /*  
                wx.checkJsApi({ 
                    jsApiList : [ 'onMenuShareAppMessage' ], // 需要檢測的JS介面列表,所有JS介面列表見附錄2, 
                    success : function(res) { 
                        // 以鍵值對的形式返回,可用的api值true,不可用為false 
                        // 如:{"checkResult":{"chooseImage":true},"errMsg":"checkJsApi:ok"} 
                        alert(res.checkResult); 
                        alert(res.errMsg); 
                    } 
                }); // end checkJsApi 
                 */  
                wx.onMenuShareAppMessage({  
                    title : '拭拭吧', // 分享標題  
                    desc : '此商城絕無僅有,不買就是血虧', // 分享描述  
                    link : currurl, // 分享連結,該連結域名或路徑必須與當前頁面對應的公眾號JS安全域名一致  
                    //imgUrl : "../weixinMall/images/icon_imgs/logo.jpg", // 分享圖示 
                    imgUrl : "http://gy7tgr.natappfree.cc//wxapi/weixinMall/images/icon_imgs/logo.jpg",
                    type : 'link', // 分享型別,music、video或link,不填預設為link  
                    dataUrl : '', // 如果type是music或video,則要提供資料鏈接,預設為空  
                    success : function() {  
                        // 使用者確認分享後執行的回撥函式  
                        // alert('share successful');  
                    },  
                    cancel : function() {  
                        // 使用者取消分享後執行的回撥函式  
                        // alert('share cancel');  
                    }  
                }); // end onMenuShareAppMessage  
  
  
                wx.onMenuShareTimeline({  
                    title : "拭拭吧", // 分享標題  
                    desc : '此商城絕無僅有,不買就是血虧', // 分享描述  
                    link : currurl, // 分享連結,該連結域名或路徑必須與當前頁面對應的公眾號JS安全域名一致  
                    imgUrl : "http://gy7tgr.natappfree.cc//wxapi/weixinMall/images/icon_imgs/logo.jpg", // 分享圖示  
                    success : function() {  
                        // 使用者確認分享後執行的回撥函式  
                    },  
                    cancel : function() {  
                        // 使用者取消分享後執行的回撥函式  
                    }  
                }) ; // end onMenuShareTimeline  
            }); // end ready  
  
            wx.error(function(res) {  
                // config資訊驗證失敗會執行error函式,如簽名過期導致驗證失敗,具體錯誤資訊可以開啟config的debug模式檢視,也可以在返回的res引數中檢視,對於SPA可以在這裡更新簽名。  
                alert("error:"+res);  
            });  
        } // end success  
    }); // end ajax  
  
});  // end document  
  



相關推薦

JSSDK-java配置--實現分享定義功能

為了實現微信分享時自定義圖片和文字,耗費了不少心力,把JSSDK翻了個底朝天首先我們來看一下微信開發給我們的樣例檔案:import java.util.UUID; import java.util.Map; import java.util.HashMap; import j

java-微信分享定義內容功能(總結)

               java寫微信分享自定義內容功能遇到的一個bug    需求:使用者通過關注公眾號  開啟邀請好友頁面(前提必須先登入)  點選手機右上角豎形

java程式實現開機啟動功能

有時一個程式需要長時間執行,但如果重啟機器,沒人手動啟動軟體時,就沒有辦法執行起來。最好的辦法是能實現程式開機自啟動。 方案一: 工具: Instsrv.exe(可以給系統安裝和刪除服務) Srvany.exe(可以讓程式以服務的方式執行) 步驟:

Pig-使用java實現使用者定義函式編譯pig.jar包錯誤

從SVN庫中匯出程式碼建立本地的pig.jar檔案時報錯: 建立命令: svn co http://svn.apache.org/repos/asf/pig/trunk cd trunk ant 執行 ant 時報如下錯誤: BUILD FAILED /home/hadoo

Java-web利用模板檔案實現匯出定義word文件

由於專案開發需要,產品給出word模板,需要匯出該格式的word檔案。 1.通過word模板檔案生成我們需要的模板.ftl檔案。 步驟:將word檔案轉換成Microsoft XML格式檔案(開啟word,檔案另存為xml格式檔案),用notepad++編輯器開啟檔案,修

微信分享定義內容實現

做html5的童鞋,微信中的分享,包括分享給qq好友、分享到朋友圈等,如何來自定義其中的內容呢?下面咱們一起聊聊。     微信JS-SDK中的分享有兩種方案。     第1種,可以自定義分享的“title”與“圖片”,即使沒有微信JSSDK的許可權也可以操作。    

利用Java 動態代理,定義註解 讀取配置檔案中的屬性值

Java動態代理在一些中介軟體中經常用到,或者一些大型專案中都會用到。 這裡順帶使用一下自定義註解方式,基於java 反射機制讀取.properties格式檔案。 demo的大致內容包含以下: 1.配置檔案:config.properties url=http://www.

element-ui tree結構實現增刪改定義功能

首先是頁面部分 <template> <el-tree id="userMtree" ref="tree" :data="treeData" node-key="id" :render-content="renderContent"

Java註解及通過定義Java註解實現資料庫的操作

在Java程式中通過註解,可以讓程式設計更加簡潔、程式碼更加清晰。因此在Java框架程式碼中,嵌入了大量的註解。 講註解,首先得講註解的概念:Java提供了一種原程式中的元素關聯任何資訊和任何元資料的途徑和方法。 一、註解的分類 1、按照執行機制分為原始碼註解、編譯註解和執

【SpringBoot 基礎系列】實現一個定義配置載入器(應用篇)

![](https://spring.hhui.top/spring-blog/imgs/200507/logo.jpg) > [【SpringBoot 基礎系列】實現一個自定義配置載入器(應用篇)](https://mp.weixin.qq.com/s?__biz=MzU3MTAzNTMzMQ==&

Struts2配置攔截器定義棧時拋異常:Unable to load configuration. - interceptor-ref - file:/D:/tomcat_install/webapps/crm/WEB-INF/classes/struts.xml

註意 其他 默認 才會 eba staf -i sta fault 代碼如下: <interceptors> <!-- 註冊自定義攔截器 --> <interceptor name="LoginInterceptor" class="co

C#實現ProperTyGrid定義屬性的方法

add order ont href foreach -c 表格 jquery desc 本文實例講解了C#實現ProperTyGrid自定義屬性的方法,分享給大家供大家參考。具體方法如下: 一般來說,C#如果要實現自定義屬性必須要需要實現接口ICustomTypeDes

實現unity定義資源

簡單的 xtu 轉換 indent exce 存儲 sum tle evel   在遊戲開發中,經常會用到一些配置文件保存一些數據,然後項目運行中讀取這些配置文件中的數據在遊戲中使用。 如:配置血條:根據角色類型(人物、動物、怪物等)配置不同的血條,包括血條大小,血條名或血

django admin有用的定義功能

轉載 ngx 博客園 title bsp div body nbsp 有用 引用園友 無名小妖 的博客 https://www.cnblogs.com/wumingxiaoyao/p/6928297.html 寫的很好,但是博客園不能轉載,不過我已經點贊了~django a

Shell實戰第一篇:自動批量創建用戶並配置保存定義密碼

shell個人筆記分享(在線閱讀):http://note.youdao.com/noteshare?id=c4ee31bdcef0d28ec525bed113c93763&sub=CFE5180F8455421E9D5252FEC98AB1EDShell實戰第一篇:自動批量創建用戶並配置保存自定義密碼

PHP實現定義圖像居中裁剪函數示例

() imagegif php5 exist 大致 結果 版本 mkdir 尺寸 圖像居中裁減的大致思路: 1.首先將圖像進行縮放,使得縮放後的圖像能夠恰好覆蓋裁減區域.(imagecopyresampled ― 重采樣拷貝部分圖像並調整大小) 2.將縮放後的圖像放置在裁減

功能:阿裏雲負載均衡SLB支持HTTP/HTTPS超時時間定義功能

算法 雲計算 摘要: 大家好,很高興的告訴大家,阿裏雲負載均衡SLB已經在新加坡、澳大利亞(悉尼)、馬來西亞(吉隆坡)、日本(東京)、美國(矽谷)、美國(弗吉尼亞)、德國(法蘭克福)、阿聯酋(迪拜)、印度(孟買)上述地域支持HTTP/HTTPS超時時間自定義功能(增加黑名單),國內地域也即將上線。1.

後臺文章編輯器的可視區域添加定義功能按鈕

nload rac IT 樣式 admin lte size gist != 有時候我們需要自定義一些樣式或者功能,要麽就是直接在Text區域直接寫html代碼, 還有一種方法就是直接將自定義一些樣式或者功能添加到後臺文章編輯器的 可視區域。具體實現方式如下: 首先在對應的

java 註解分析及定義註解

var AC IE 原始類型 length JD email 文檔 生成 註解概念:   java提供了一種原程序中的元素關聯任何信息和任何元數據的途徑與方法。 註解分類: 運行機制分類:源碼註解,編譯時註解,運行時註解。  來源分類:JDK的註解,第三方註解,自

ionic 調用restful API services時全局錯誤處理的實現定義錯誤處理的實現

ner images AS tor framework ets inject sta port 往往我們的ionic程序需要調用API Service. 比如天氣,地圖等等。當這些API Service 不穩定或者不可訪問時,我們可以通過在註冊一個自定義的ErrorHand