java微信公眾號(hào)支付開(kāi)發(fā)之現(xiàn)金紅包
我們先來(lái)看看公眾號(hào)發(fā)放現(xiàn)金紅包的效果:
需要調(diào)用商戶平臺(tái)的接口,接口發(fā)放規(guī)則如下:
1.發(fā)送頻率限制——默認(rèn)1800/min
2.發(fā)送個(gè)數(shù)上限——按照默認(rèn)1800/min算
3.金額上限——根據(jù)傳入場(chǎng)景id不同默認(rèn)上限不同,可以在商戶平臺(tái)產(chǎn)品設(shè)置進(jìn)行設(shè)置和申請(qǐng),最大不大于4999元/個(gè)
4.其他的“量”上的限制還有哪些?——用戶當(dāng)天的領(lǐng)取上限次數(shù),默認(rèn)是10
5.如果量上滿足不了我們的需求,如何提高各個(gè)上限?——金額上限和用戶當(dāng)天領(lǐng)取次數(shù)上限可以在商戶平臺(tái)進(jìn)行設(shè)置
注意-紅包金額大于200時(shí),請(qǐng)求參數(shù)scene_id必傳,參數(shù)說(shuō)明見(jiàn)下文。
注意2-根據(jù)監(jiān)管要求,新申請(qǐng)商戶號(hào)使用現(xiàn)金紅包需要滿足兩個(gè)條件:1、入駐時(shí)間超過(guò)90天 2、連續(xù)正常交易30天。
請(qǐng)求Url https://api.mch.weixin.qq.com/mmpaymkttransfers/sendredpack
是否需要證書(shū) 是(證書(shū)及使用說(shuō)明詳見(jiàn)商戶證書(shū))
請(qǐng)求方式 POST
請(qǐng)求數(shù)據(jù)示例:
<xml> <sign><![CDATA[E1EE61A91C8E90F299DE6AE075D60A2D]]></sign> <mch_billno><![CDATA[0010010404201411170000046545]]></mch_billno> <mch_id><![CDATA[888]]></mch_id> <wxappid><![CDATA[wxcbda96de0b165486]]></wxappid> <send_name><![CDATA[send_name]]></send_name> <re_openid><![CDATA[onqOjjmM1tad-3ROpncN-yUfa6uI]]></re_openid> <total_amount><![CDATA[200]]></total_amount> <total_num><![CDATA[1]]></total_num> <wishing><![CDATA[恭喜發(fā)財(cái)]]></wishing> <client_ip><![CDATA[127.0.0.1]]></client_ip> <act_name><![CDATA[新年紅包]]></act_name> <remark><![CDATA[新年紅包]]></remark> <scene_id><![CDATA[PRODUCT_2]]></scene_id> <consume_mch_id><![CDATA[10000097]]></consume_mch_id> <nonce_str><![CDATA[50780e0cca98c8c8e814883e5caa672e]]></nonce_str> <risk_info>posttime%3d123123412%26clientversion%3d234134%26mobile%3d122344545%26deviceid%3dIOS</risk_info> </xml>
接口需要調(diào)用商戶平臺(tái)的證書(shū),證書(shū)需要去商戶平臺(tái)下載:
然后在接口中使用證書(shū),首先我們新建一個(gè)WeixinSSL 類
@Component public class WeiXinSSL { /** * 證書(shū)類型 */ @Value("${werchant.storekey}") private String storekey; /** * 文件路徑 */ @Value("${werchant.ssLfile}") private String ssLfile; /** * 商戶號(hào) */ @Value("${werchant.merchantNumber}") private String merchantNumber; public String getStorekey() { return storekey; } public void setStorekey(String storekey) { this.storekey = storekey; } public String getSsLfile() { return ssLfile; } public void setSsLfile(String ssLfile) { this.ssLfile = ssLfile; } public String getMerchantNumber() { return merchantNumber; } public void setMerchantNumber(String merchantNumber) { this.merchantNumber = merchantNumber; } }
封裝HttpClientSSL 類實(shí)現(xiàn) https 請(qǐng)求加證書(shū):
@Component public class HttpClientSSL { @Autowired private WeiXinSSL weiXinSSL; // 請(qǐng)求超時(shí)時(shí)間(毫秒) 5秒 public static RequestConfig requestConfig; // 響應(yīng)超時(shí)時(shí)間(毫秒) 60秒 public static int HTTP_RESPONSE_TIMEOUT = 60 * 1000; // httpClient字符編碼 public static String encoding = "UTF-8"; public static RequestConfig getRequestConfig() { return RequestConfig.custom().setConnectTimeout(5 * 1000) .setConnectionRequestTimeout(HTTP_RESPONSE_TIMEOUT).build(); } public static void setRequestConfig(RequestConfig requestConfig) { HttpClientSSL.requestConfig = requestConfig; } /** * https請(qǐng)求偽造證書(shū) * @return */ public CloseableHttpClient defaultSSLClient() { SSLContext sslContext = null; try { new SSLContextBuilder().loadTrustMaterial(null,new TrustStrategy(){ @Override public boolean isTrusted(X509Certificate[] chain, String authType) throws java.security.cert.CertificateException { return false; } }); } catch (NoSuchAlgorithmException | KeyStoreException e) { e.printStackTrace(); } SSLConnectionSocketFactory factory = new SSLConnectionSocketFactory(sslContext); return HttpClients.custom().setSSLSocketFactory(factory).build(); } /** * https請(qǐng)求加證書(shū) * @return */ public CloseableHttpClient defaultSSLClientFile() { if (this.weiXinSSL == null){ return this.defaultSSLClient(); } FileInputStream inputStream = null; KeyStore keyStore = null; try { // ssl類型 keyStore = KeyStore.getInstance(weiXinSSL.getStorekey()); // ssl文件 inputStream = new FileInputStream(weiXinSSL.getSsLfile()); // 設(shè)置ssl密碼 keyStore.load(inputStream,weiXinSSL.getMerchantNumber().toCharArray()); } catch (KeyStoreException | NoSuchAlgorithmException | CertificateException | IOException e1) { e1.printStackTrace(); } finally { try { inputStream.close(); } catch (IOException e) { e.printStackTrace(); } } SSLContext sslContext = null; try { sslContext = SSLContexts.custom().loadKeyMaterial(keyStore,weiXinSSL.getMerchantNumber().toCharArray()).build(); } catch (UnrecoverableKeyException | NoSuchAlgorithmException | KeyStoreException | KeyManagementException e) { e.printStackTrace(); } SSLConnectionSocketFactory factory = new SSLConnectionSocketFactory(sslContext, new String[] { "TLSv1" }, null, SSLConnectionSocketFactory.BROWSER_COMPATIBLE_HOSTNAME_VERIFIER); return HttpClients.custom().setSSLSocketFactory(factory).build(); } /** * 封裝發(fā)送請(qǐng)求的方法 * @throws UnsupportedEncodingException */ public String send(String url, String data, CloseableHttpClient closeableHttpClient) throws UnsupportedEncodingException { CloseableHttpClient client = closeableHttpClient; HttpPost httpPost = new HttpPost(URLDecoder.decode(url, encoding)); httpPost.addHeader("Connection", "keep-alive"); httpPost.addHeader("Accept", "*/*"); httpPost.addHeader("Content-Type", "application/x-www-form-urlencoded; charset=UTF-8"); httpPost.addHeader("Host", "api.mch.weixin.qq.com"); httpPost.addHeader("X-Requested-With", "XMLHttpRequest"); httpPost.addHeader("Cache-Control", "max-age=0"); httpPost.addHeader("User-Agent", "Mozilla/4.0 (compatible; MSIE 8.0; Windows NT 6.0) "); httpPost.setConfig(this.getRequestConfig());// 設(shè)置超時(shí)時(shí)間 CloseableHttpResponse response = null; // 參數(shù)放入 StringEntity entity = new StringEntity(data, encoding); entity.setContentEncoding(encoding); entity.setContentType("application/xml"); httpPost.setEntity(entity); try { response = client.execute(httpPost); if (response.getStatusLine().getStatusCode() == 200) { HttpEntity httpEntity = (HttpEntity) response.getEntity(); if (response != null) { return EntityUtils.toString(httpEntity,encoding); } } } catch (IOException e) { e.printStackTrace(); } return null; } }
這樣我們就封裝了一個(gè)https請(qǐng)求加證書(shū)的實(shí)體類,接下來(lái)我們生成請(qǐng)求微信紅包接口:
https://api.mch.weixin.qq.com/mmpaymkttransfers/sendredpack 的參數(shù)簽名:
/** * 紅包參數(shù)實(shí)體類 * @throws UnsupportedEncodingException */ @Component public class SendRedPack implements Serializable{ /** * */ private static final long serialVersionUID = -1000489228099916099L; private String nonce_str;// 隨機(jī)字符串 private String sign;// 簽名 private String mch_billno;// 商戶訂單號(hào) private String mch_id;// 商戶號(hào) private String wxappid;// 公眾賬號(hào) private String send_name;// 商戶名稱 private String re_openid;// 用戶 private int total_amount;// 付款金額 單位:分 private int total_num;// 紅包發(fā)放總?cè)藬?shù) private String wishing;// 紅包祝福語(yǔ) private String client_ip;// Ip地址 private String act_name;// 活動(dòng)名稱 private String remark;// 備注 public String getNonce_str() { return nonce_str; } public void setNonce_str(String nonce_str) { this.nonce_str = nonce_str; } public String getSign() { return sign; } public void setSign(String sign) { this.sign = sign; } public String getMch_billno() { return mch_billno; } public void setMch_billno(String mch_billno) { this.mch_billno = mch_billno; } public String getMch_id() { return mch_id; } public void setMch_id(String mch_id) { this.mch_id = mch_id; } public String getWxappid() { return wxappid; } public void setWxappid(String wxappid) { this.wxappid = wxappid; } public String getSend_name() { return send_name; } public void setSend_name(String send_name) { this.send_name = send_name; } public String getRe_openid() { return re_openid; } public void setRe_openid(String re_openid) { this.re_openid = re_openid; } public int getTotal_amount() { return total_amount; } public void setTotal_amount(int total_amount) { this.total_amount = total_amount; } public int getTotal_num() { return total_num; } public void setTotal_num(int total_num) { this.total_num = total_num; } public String getWishing() { return wishing; } public void setWishing(String wishing) { this.wishing = wishing; } public String getClient_ip() { return client_ip; } public void setClient_ip(String client_ip) { this.client_ip = client_ip; } public String getAct_name() { return act_name; } public void setAct_name(String act_name) { this.act_name = act_name; } public String getRemark() { return remark; } public void setRemark(String remark) { this.remark = remark; } }
接下來(lái)是發(fā)送紅包的控制器:
/** * 領(lǐng)紅包控制器 * @author zengliang */ @Controller @RequestMapping(value="/redenveLopesReceive") public class RedEnvelopesReceiveController { //微信唯一標(biāo)識(shí) @Value("${weixin.appid}") private String appid; //微信開(kāi)發(fā)者密碼標(biāo)識(shí) @Value("${weixin.appsecret}") public String appsecret; @Autowired private SendRedPack sendredpack; @Autowired private HttpClientSSL httpclientssl; /** * 發(fā)送XML參數(shù) * @author zengliang */ @ResponseBody @RequestMapping(value="/sendXml") public String sendXml(String openid,Long redenveLopes_id ,String mch_billno){ RedenveLopes redenve = redenveLopesService.findOne(redenveLopes_id); XMLUtil xmlUtil= new XMLUtil(); sendredpack.setAct_name(redenve.getAct_name()); sendredpack.setNonce_str(xmlUtil.random()); sendredpack.setRe_openid(openid); sendredpack.setClient_ip(redenve.getClient_ip()); sendredpack.setMch_billno(mch_billno); sendredpack.setMch_id(redenve.getMch_id()); String xx = redenve.getRemark(); sendredpack.setRemark(StringUtils.isEmpty(xx) == false?xx:"空"); sendredpack.setSend_name(redenve.getSend_name()); sendredpack.setTotal_amount(redenve.getTotal_amount()); sendredpack.setTotal_num(redenve.getTotal_num()); sendredpack.setWishing(redenve.getWishing()); sendredpack.setWxappid(redenve.getWxappidxx()); //生成簽名 String params = this.createSendRedPackOrderSign(sendredpack,redenve.getStore_key()); sendredpack.setSign(params); xmlUtil.xstream().alias("xml",sendredpack.getClass()); //擴(kuò)展xstream,使其支持CDATA塊 String requestXml = xmlUtil.xstream().toXML(sendredpack); String result; try { result = httpclientssl.send("https://api.mch.weixin.qq.com/mmpaymkttransfers/sendredpack",requestXml,httpclientssl.defaultSSLClientFile()); System.out.println("成功返回值"+result); return result; } catch (UnsupportedEncodingException e) { e.printStackTrace(); } return null; } /** * 生成簽名 * @param redPack * @return */ public String createSendRedPackOrderSign(SendRedPack redPack,String storekey){ StringBuffer sign = new StringBuffer(); sign.append("act_name=").append(redPack.getAct_name()); sign.append("&client_ip=").append(redPack.getClient_ip()); sign.append("&mch_billno=").append(redPack.getMch_billno()); sign.append("&mch_id=").append(redPack.getMch_id()); sign.append("&nonce_str=").append(redPack.getNonce_str()); sign.append("&re_openid=").append(redPack.getRe_openid()); sign.append("&remark=").append(redPack.getRemark()); sign.append("&send_name=").append(redPack.getSend_name()); sign.append("&total_amount=").append(redPack.getTotal_amount()); sign.append("&total_num=").append(redPack.getTotal_num()); sign.append("&wishing=").append(redPack.getWishing()); sign.append("&wxappid=").append(redPack.getWxappid()); sign.append("&key=").append(storekey); return DigestUtils.md5Hex(sign.toString()).toUpperCase(); } }
然后我們需要用一個(gè)解析XML的工具類實(shí)現(xiàn)解析微信返回的XML
/** * 解析XML工具類 * @author zengliang */ @Component public class XMLUtil { /** * 解析微信返回的XML * @param xml * @return * @throws Exception */ @SuppressWarnings("unchecked") public Map<String, String> parseXml(String xml)throws Exception { Map<String,String> map = new HashMap<String,String>(); Document doc = null; try { doc = DocumentHelper.parseText(xml); // 將字符串轉(zhuǎn)為XML Element rootElt = doc.getRootElement(); // 獲取根節(jié)點(diǎn) List<Element> list = rootElt.elements();//獲取根節(jié)點(diǎn)下所有節(jié)點(diǎn) for (Element element : list) { //遍歷節(jié)點(diǎn) map.put(element.getName(), element.getText()); //節(jié)點(diǎn)的name為map的key,text為map的value } } catch (DocumentException e) { e.printStackTrace(); } catch (Exception e) { e.printStackTrace(); } return map; } /** * 擴(kuò)展xstream,使其支持CDATA塊 */ private XStream xstream = new XStream(new XppDriver(new NoNameCoder()) { @Override public HierarchicalStreamWriter createWriter(Writer out) { return new PrettyPrintWriter(out) { // 對(duì)所有xml節(jié)點(diǎn)的轉(zhuǎn)換都增加CDATA標(biāo)記 boolean cdata = true; @Override @SuppressWarnings("rawtypes") public void startNode(String name, Class clazz) { super.startNode(name, clazz); } @Override public String encodeNode(String name) { return name; } @Override protected void writeText(QuickWriter writer, String text) { if (cdata) { writer.write("<![CDATA["); writer.write(text); writer.write("]]>"); } else { writer.write(text); } } }; } }); private XStream inclueUnderlineXstream = new XStream(new DomDriver(null,new XmlFriendlyNameCoder("_-", "_"))); public XStream getXstreamInclueUnderline() { return inclueUnderlineXstream; } public XStream xstream() { return xstream; } /** * 生成隨機(jī)數(shù) * @return */ public String random(){ String random = UUID.randomUUID().toString().replace("-", ""); return random; } }
然后我們調(diào)用 sendXML 方法公眾號(hào)就能向用戶發(fā)送紅包了。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- java微信公眾號(hào)企業(yè)付款開(kāi)發(fā)
- Java版微信公眾號(hào)支付開(kāi)發(fā)全過(guò)程
- 微信公眾號(hào)開(kāi)發(fā)之設(shè)置自定義菜單實(shí)例代碼【java版】
- 微信公眾號(hào)開(kāi)發(fā)之回復(fù)圖文消息java代碼
- Java微信公眾號(hào)開(kāi)發(fā)之通過(guò)微信公眾號(hào)獲取用戶信息
- java微信公眾號(hào)開(kāi)發(fā)案例
- 用Java設(shè)計(jì)模式中的觀察者模式開(kāi)發(fā)微信公眾號(hào)的例子
- java微信公眾號(hào)開(kāi)發(fā)第一步 公眾號(hào)接入和access_token管理
- Java開(kāi)發(fā)微信公眾號(hào)接收和被動(dòng)回復(fù)普通消息
- java微信公眾號(hào)開(kāi)發(fā)(搭建本地測(cè)試環(huán)境)
- java開(kāi)發(fā)微信公眾號(hào)支付
- Java 微信公眾號(hào)開(kāi)發(fā)相關(guān)總結(jié)
相關(guān)文章
為spring get請(qǐng)求添加自定義的參數(shù)處理操作(如下劃線轉(zhuǎn)駝峰)
這篇文章主要介紹了為spring get請(qǐng)求添加自定義的參數(shù)處理操作(如下劃線轉(zhuǎn)駝峰),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-09-09Mybatis執(zhí)行SQL時(shí)多了一個(gè)limit的問(wèn)題及解決方法
這篇文章主要介紹了Mybatis執(zhí)行SQL時(shí)多了一個(gè)limit的問(wèn)題及解決方法,Mybatis攔截器方法識(shí)別到配置中參數(shù)supportMethodsArguments 為ture時(shí)會(huì)分頁(yè)處理,本文結(jié)合示例代碼給大家講解的非常詳細(xì),需要的朋友可以參考下2022-10-10Java時(shí)間復(fù)雜度、空間復(fù)雜度的深入詳解
對(duì)于一個(gè)算法,其時(shí)間復(fù)雜度和空間復(fù)雜度往往是相互影響的,當(dāng)追求一個(gè)較好的時(shí)間復(fù)雜度時(shí),可能會(huì)使空間復(fù)雜度的性能變差,即可能導(dǎo)致占用較多的存儲(chǔ)空間,這篇文章主要給大家介紹了關(guān)于Java時(shí)間復(fù)雜度、空間復(fù)雜度的相關(guān)資料,需要的朋友可以參考下2021-11-11如何使用mybatis-plus實(shí)現(xiàn)分頁(yè)查詢功能
最近在研究mybatis,然后就去找簡(jiǎn)化mybatis開(kāi)發(fā)的工具,發(fā)現(xiàn)就有通用Mapper和mybatis-plus兩個(gè)比較好的可是使用,可是經(jīng)過(guò)對(duì)比發(fā)現(xiàn)還是mybatis-plus比較好,下面這篇文章主要給大家介紹了關(guān)于如何使用mybatis-plus實(shí)現(xiàn)分頁(yè)查詢功能的相關(guān)資料,需要的朋友可以參考下2022-06-06[Java]詳解Socket和ServerSocket學(xué)習(xí)筆記
即時(shí)類應(yīng)用或者即時(shí)類的游戲,HTTP協(xié)議很多時(shí)候無(wú)法滿足于我們的需求,這會(huì),Socket對(duì)于我們來(lái)說(shuō)就非常實(shí)用了。本篇文章主要介紹了Socket和ServerSocket,有興趣的可以了解一下。2016-12-12SpringCloud Feign多參數(shù)傳遞及需要注意的問(wèn)題
這篇文章主要介紹了SpringCloud Feign多參數(shù)傳遞及需要注意的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-03-03Java?Zookeeper分布式分片算法超詳細(xì)講解流程
ZooKeeper是一個(gè)分布式的,開(kāi)放源碼的分布式應(yīng)用程序協(xié)調(diào)服務(wù),是Google的Chubby一個(gè)開(kāi)源的實(shí)現(xiàn),是Hadoop和Hbase的重要組件。它是一個(gè)為分布式應(yīng)用提供一致性的軟件,提供的功能包括:配置維護(hù)、域名服務(wù)、分布式同步、組服務(wù)等2023-03-03Java代碼實(shí)現(xiàn)循環(huán)隊(duì)列的示例代碼
隊(duì)列作為基礎(chǔ)的數(shù)據(jù)結(jié)構(gòu),是程序員的入門課。也是所有程序員必須掌握的一種數(shù)據(jù)結(jié)構(gòu),隊(duì)列在程序中廣泛應(yīng)用,因此我們應(yīng)該對(duì)隊(duì)列有深入的了解,接下來(lái)我們通過(guò)代碼來(lái)對(duì)隊(duì)列這種數(shù)據(jù)結(jié)構(gòu)進(jìn)行深度解析,感興趣的朋友一起看看吧2021-09-09Java生成10個(gè)1000以內(nèi)的隨機(jī)數(shù)并用消息框顯示數(shù)組內(nèi)容然后求和輸出
這篇文章主要介紹了Java生成10個(gè)1000以內(nèi)的隨機(jī)數(shù)并用消息框顯示數(shù)組內(nèi)容然后求和輸出,需要的朋友可以參考下2015-10-10