java小程序支付對接

文檔:https://pay.weixin.qq.com/wiki/doc/api/wxa/wxa_api.php?chapter=9_1

JSAPI下單

必傳參數(shù):

參數(shù)名 描述
appid 應(yīng)用ID, 注意appid要和mchid關(guān)聯(lián)
mchid 直連商戶號
openid openid
body 商品描述
out_trade_no 商戶訂單號
notify_url 通知地址,注意不能攜帶參數(shù)
total_fee 金額努酸,單位分
nonce_str 隨機(jī)字符串
spbill_create_ip 小程序請求ip
trade_type 支付類型洞拨,JSAPI
sign 簽名

nonce_str方法:

/**
* 獲取32位隨機(jī)字符串
* @return
*/
public static String getNonceStr() {
    Random random = new Random();
    return MD5Util.md5(String.valueOf(random.nextInt(10000)));
}

sign生成方法:

/**
* 微信支付sign簽名 --> 參數(shù)按ascii排序蛾派,key=value& 形式拼接授艰,最后拼接上api_key胶哲,md5加密轉(zhuǎn)大寫
* @param parameters 除了sign以外的請求參數(shù)
* @param api_key    商戶api key
* @return
*/
public static String createSign(SortedMap<Object, Object> parameters,String api_key) {
    StringBuffer sb = new StringBuffer();
    Set<Map.Entry<Object, Object>> es = parameters.entrySet();
    Iterator<Map.Entry<Object, Object>> it = es.iterator();
    while (it.hasNext()) {
        Map.Entry<Object, Object> entry = (Map.Entry<Object, Object>) it.next();
        String k = (String) entry.getKey();
        Object v = entry.getValue();
        /** 如果參數(shù)為key或者sign纲岭,則不參與加密簽名 */
        if (null != v && !"".equals(v) && !"sign".equals(k) && !"key".equals(k)) {
            sb.append(k + "=" + v + "&");
        }
    }
    /** 支付密鑰必須參與加密拥刻,放在字符串最后面 */
    sb.append("key=" + api_key);
    /** 記得最后一定要轉(zhuǎn)換為大寫 */
    String sign = MD5Util.md5(sb.toString()).toUpperCase();
    return sign;
}

paySign生成方法

/**
* paySign生成方法 --> 參數(shù)按ascii排序怜瞒,key=value& 形式拼接,最后拼接上api_key般哼,md5加密轉(zhuǎn)大寫
* @param parameters 參數(shù)包括appId吴汪、timeStamp、nonceStr蒸眠、package漾橙、signType
* @param api_key    商戶api key
* @return
*/
public static String getPaySign(SortedMap<Object, Object> resultParams){
    StringBuilder sb = new StringBuilder();
    for(Map.Entry entry : resultParams.entrySet()){
        sb.append(entry.getKey()).append("=").append(entry.getValue()).append("&");
    }
    sb.append("key=").append(MCH_KEY);
    return MD5Util.md5(sb.toString()).toUpperCase();
}

jsapi下單方法

/**
* jsapi方法 --> 將參數(shù)以xml格式post
* @param openId openId
* @param money  金額
* @param ip 小程序端ip
* @return 返回小程序支付需要的參數(shù)
*/
public static WechatPayModel wechatPay(String openId, int money, String ip){
    SortedMap<Object, Object> parameters = new TreeMap<Object, Object>();
    parameters.put("appid", appid);
    parameters.put("mch_id", MCH_ID);
    parameters.put("openid", openId);
    parameters.put("nonce_str", getNonceStr());
    parameters.put("body", "智能柜支付");
    parameters.put("out_trade_no", createBillNo());
    parameters.put("total_fee", money);
    parameters.put("spbill_create_ip", ip);
    parameters.put("notify_url", "http://test.chinasuperbox.com/notifyUrl");
    parameters.put("trade_type", "JSAPI");
    String sign = createSign(parameters, MCH_KEY);
    parameters.put("sign", sign);
    String requestXML = XmlUtil.getRequestXml(parameters);
    Map<String,String> resultMap = payPost(wechatPayUrl, requestXML);
    WechatPayModel wechatPayModel = new WechatPayModel();
    if( resultMap.get("return_code").equals("SUCCESS") && resultMap.get("result_code").equals("SUCCESS") ){
        log.info("支付預(yù)請求成功");
        String prepay_id = resultMap.get("prepay_id");
        String nonce_str = resultMap.get("nonce_str");
        long timeStamp = System.currentTimeMillis()/1000;

        SortedMap<Object, Object> resultParams = new TreeMap<Object, Object>();
        resultParams.put("appId", appid);
        resultParams.put("timeStamp", timeStamp);
        resultParams.put("nonceStr", nonce_str);
        resultParams.put("package", "prepay_id="+prepay_id);
        resultParams.put("signType", "MD5");
        String paySign = getPaySign(resultParams);
        wechatPayModel.setPaySign(paySign);

        wechatPayModel.setAppId(appid);
        wechatPayModel.setNonceStr(nonce_str);
        wechatPayModel.setPackageStr("prepay_id="+prepay_id);
        wechatPayModel.setSignType("MD5");
        wechatPayModel.setTimeStamp(timeStamp);
        wechatPayModel.setMsg("支付預(yù)請求成功");
        return wechatPayModel;
    }else{
        if( resultMap.containsKey("err_code_des") ){
            wechatPayModel.setMsg(resultMap.get("err_code_des"));
        }else{
            wechatPayModel.setMsg(resultMap.get("return_msg"));
        }
        log.info("支付預(yù)請求失敗,  return_msg = {}, err_code_des = {}", resultMap.get("return_msg"), resultMap.get("err_code_des"));
        return wechatPayModel;
    }
}

工具類:

/**
* post方法,將返回的xml解析成map返回
* @param requestUrl 請求地址
* @param requestXML 請求xml字符串
* @return 返回map
*/
public static Map<String,String> payPost(String requestUrl, String requestXML){
    CloseableHttpClient httpclient = HttpClients.custom().build();
    Map<String,String> result = new HashMap<String,String>();
    HttpPost httpPost = new HttpPost(requestUrl);
    StringEntity reqEntity  = new StringEntity(requestXML, "utf-8");
    // 設(shè)置content type類型
    reqEntity.setContentType("application/x-www-form-urlencoded");
    httpPost.setEntity(reqEntity);
    try{
        CloseableHttpResponse response = httpclient.execute(httpPost);
        HttpEntity entity = response.getEntity();
        if (entity != null) {
            // 從request中取得輸入流
            InputStream inputStream = entity.getContent();
            // 讀取輸入流
            SAXReader reader = new SAXReader();
            Document document = reader.read(inputStream);
            // 得到xml根元素
            Element root = document.getRootElement();
            // 得到根元素的所有子節(jié)點
            List<Element> elementList = root.elements();
            // 遍歷所有子節(jié)點
            for (Element e : elementList) {
                result.put(e.getName(), e.getText());
            }
        }
        return result;
    }catch (Exception e){
        e.printStackTrace();
        return null;
    }
}

/**
 * 將請求參數(shù)轉(zhuǎn)換為xml格式的string
 * @param parameters
 * @return
 */
public static String getRequestXml(SortedMap<Object, Object> parameters) {
    StringBuffer sb = new StringBuffer();
    sb.append("<xml>");
    Set<Map.Entry<Object, Object>> es = parameters.entrySet();
    Iterator<Map.Entry<Object, Object>> it = es.iterator();
    while (it.hasNext()) {
        Map.Entry<Object, Object> entry = (Map.Entry<Object, Object>) it.next();
        String k = (String) entry.getKey();
        String v = entry.getValue() + "";
        if ("attach".equalsIgnoreCase(k) || "body".equalsIgnoreCase(k) || "sign".equalsIgnoreCase(k)) {
            sb.append("<" + k + ">" + "<![CDATA[" + v + "]]></" + k + ">");
        } else {
            sb.append("<" + k + ">" + v + "</" + k + ">");
        }
    }
    sb.append("</xml>");
    return sb.toString();
}

回調(diào)xml

<?xml version="1.0" encoding="utf-8"?>
<xml>
  <appid><![CDATA[wx303cbfa214a64a4c]]></appid>
  <bank_type><![CDATA[OTHERS]]></bank_type>
  <cash_fee><![CDATA[1]]></cash_fee>
  <fee_type><![CDATA[CNY]]></fee_type>
  <is_subscribe><![CDATA[N]]></is_subscribe>
  <mch_id><![CDATA[1536185011]]></mch_id>
  <nonce_str><![CDATA[bb57db42f77807a9c5823bd8c2d9aaef]]></nonce_str>
  <openid><![CDATA[o_W745D0ptmzIyj7kmykJqciJPYU]]></openid>
  <out_trade_no><![CDATA[075500305202132034782700473]]></out_trade_no>
  <result_code><![CDATA[SUCCESS]]></result_code>
  <return_code><![CDATA[SUCCESS]]></return_code>
  <sign><![CDATA[4C8B3F62B4212F84BC7C830858D5BFFF]]></sign>
  <time_end><![CDATA[20210603093212]]></time_end>
  <total_fee>1</total_fee>
  <trade_type><![CDATA[JSAPI]]></trade_type>
  <transaction_id><![CDATA[4200001039202106031250761263]]></transaction_id>
</xml>

回調(diào)方法

/**
 * 回調(diào)方法 -- 注意接收字符串為xml字符串楞卡,返回格式為xml字符串
 * @param parameters
 * @return
 */
@PostMapping("/notifyUrl")
public String notifyUrl(HttpServletRequest request) {
    log.info("into notifyUrl");
    try {
        //微信返回的請求體
        String body = NetUtil.getRequestBody(request);
        Map<String, String> xmlMap = XmlUtil.xmlToMap(body);
        //業(yè)務(wù)邏輯處理
                payConsumService.paySuccess(xmlMap);

    } catch (Exception e) {
        e.printStackTrace();
    }
    return "<xml>"
            +"<return_code>SUCCESS</return_code>"
            +"<return_msg>OK</return_msg>"
            +"</xml>";
}
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末霜运,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子蒋腮,更是在濱河造成了極大的恐慌淘捡,老刑警劉巖,帶你破解...
    沈念sama閱讀 221,888評論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件池摧,死亡現(xiàn)場離奇詭異焦除,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)作彤,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,677評論 3 399
  • 文/潘曉璐 我一進(jìn)店門膘魄,熙熙樓的掌柜王于貴愁眉苦臉地迎上來乌逐,“玉大人,你說我怎么就攤上這事创葡≌闾撸” “怎么了?”我有些...
    開封第一講書人閱讀 168,386評論 0 360
  • 文/不壞的土叔 我叫張陵灿渴,是天一觀的道長成黄。 經(jīng)常有香客問我,道長逻杖,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 59,726評論 1 297
  • 正文 為了忘掉前任思瘟,我火速辦了婚禮荸百,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘滨攻。我一直安慰自己够话,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 68,729評論 6 397
  • 文/花漫 我一把揭開白布光绕。 她就那樣靜靜地躺著女嘲,像睡著了一般。 火紅的嫁衣襯著肌膚如雪诞帐。 梳的紋絲不亂的頭發(fā)上欣尼,一...
    開封第一講書人閱讀 52,337評論 1 310
  • 那天,我揣著相機(jī)與錄音停蕉,去河邊找鬼愕鼓。 笑死,一個胖子當(dāng)著我的面吹牛慧起,可吹牛的內(nèi)容都是我干的菇晃。 我是一名探鬼主播,決...
    沈念sama閱讀 40,902評論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼蚓挤,長吁一口氣:“原來是場噩夢啊……” “哼磺送!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起灿意,我...
    開封第一講書人閱讀 39,807評論 0 276
  • 序言:老撾萬榮一對情侶失蹤估灿,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后脾歧,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體甲捏,經(jīng)...
    沈念sama閱讀 46,349評論 1 318
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 38,439評論 3 340
  • 正文 我和宋清朗相戀三年鞭执,在試婚紗的時候發(fā)現(xiàn)自己被綠了司顿。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片芒粹。...
    茶點故事閱讀 40,567評論 1 352
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖大溜,靈堂內(nèi)的尸體忽然破棺而出化漆,到底是詐尸還是另有隱情,我是刑警寧澤钦奋,帶...
    沈念sama閱讀 36,242評論 5 350
  • 正文 年R本政府宣布座云,位于F島的核電站,受9級特大地震影響付材,放射性物質(zhì)發(fā)生泄漏朦拖。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,933評論 3 334
  • 文/蒙蒙 一厌衔、第九天 我趴在偏房一處隱蔽的房頂上張望璧帝。 院中可真熱鬧,春花似錦富寿、人聲如沸睬隶。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,420評論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽苏潜。三九已至,卻和暖如春变勇,著一層夾襖步出監(jiān)牢的瞬間恤左,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,531評論 1 272
  • 我被黑心中介騙來泰國打工搀绣, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留赃梧,地道東北人。 一個月前我還...
    沈念sama閱讀 48,995評論 3 377
  • 正文 我出身青樓豌熄,卻偏偏與公主長得像授嘀,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子锣险,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 45,585評論 2 359

推薦閱讀更多精彩內(nèi)容