h5通过php微信支付宝支付,用H5调用支付微信公众号支付的解析

这篇文章主要为大家详细介绍了微信公众号支付H5调用支付,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

最近项目需要微信支付,然后看了下微信公众号支付,虽然不难,但是细节还是需要注意的,用了大半天时间写了个demo,并且完整的测试了一下支付流程,下面分享一下微信公众号支付的经验。

一、配置公众号微信支付

需要我们配置微信公众号支付地址和测试白名单。

d63f9b1b6e5af3a6a4a95bf87ff59d49.png

比如:支付JS页面的地址为 http://www.xxx.com/shop/pay/

那此处配置www.xxx.com/shop/pay/

二、开发流程

借用微信公众号支付api(地址 http://pay.weixin.qq.com/wiki/doc/api/index.php?chapter=7_4),我们需要开发的为红色标记出的。如下:

73a77b6204445b54adcd6b4a3e053c10.png

三、向微信服务器端下订单

调用统一下单接口,这样就能获取微信支付的prepay_id(http://pay.weixin.qq.com/wiki/doc/api/index.php?chapter=9_1)。

在调用该接口前有几个字段是H5支付必须填写的openid

3.1 获取openid

可以通过网页授权形式(http://mp.weixin.qq.com/wiki/17/c0f37d5704f0b64713d5d2c37b468d75.html)

在微信中发送如下链接

https://open.weixin.qq.com/connect/oauth2/authorize?appid=wx520c15f417810387&redirect_uri=要跳转的下订单的url&response_type=code&scope=snsapi_base&state=123#wechat_redirect

3.2 后台支付

代码如下,包含预处理订单,支付订单等接口。package org.andy.controller;

import java.io.ByteArrayInputStream;

import java.io.IOException;

import java.io.InputStream;

import java.io.UnsupportedEncodingException;

import java.util.Date;

import java.util.HashMap;

import java.util.Iterator;

import java.util.Map;

import java.util.Map.Entry;

import java.util.Random;

import javax.servlet.ServletInputStream;

import javax.servlet.http.HttpServletRequest;

import javax.servlet.http.HttpServletResponse;

import org.apache.commons.codec.digest.DigestUtils;

import org.springframework.stereotype.Controller;

import org.springframework.web.bind.annotation.RequestMapping;

import org.xmlpull.v1.XmlPullParser;

import org.xmlpull.v1.XmlPullParserException;

import org.xmlpull.v1.XmlPullParserFactory;

import com.fasterxml.jackson.databind.JsonNode;

import com.gson.oauth.Oauth;

import com.gson.oauth.Pay;

import com.gson.util.HttpKit;

import com.gson.util.Tools;

import org.andy.util.DatetimeUtil;

import org.andy.util.JsonUtil;

import org.andy.util.SessionUtil;

import org.andy.util.WebUtil;

@Controller

@RequestMapping("/pay")

public class WXPayController {

@RequestMapping(value = "wxprepay")

public void jspay(HttpServletRequest request, HttpServletResponse response, String callback) throws Exception {

// 获取openid

String openId = SessionUtil.getAtt(request, "openId");

if (openId == null) {

openId = getUserOpenId(request);

}

String appid = "wx16691fcb0523c1a4";

String partnerid = "22223670";

String paternerKey = "fjfjfjfjf1234567FFFFFFFFF1234567";

String out_trade_no = getTradeNo();

Map paraMap = new HashMap();

paraMap.put("appid", appid);

paraMap.put("attach", "测试支付");

paraMap.put("body", "测试购买Beacon支付");

paraMap.put("mch_id", partnerid);

paraMap.put("nonce_str", create_nonce_str());

paraMap.put("openid", openId);

paraMap.put("out_trade_no", out_trade_no);

paraMap.put("spbill_create_ip", getAddrIp(request));

paraMap.put("total_fee", "1");

paraMap.put("trade_type", "JSAPI");

paraMap.put("notify_url", "http://www.xxx.co/wxpay/pay/appPay_notify.shtml");

String sign = getSign(paraMap, paternerKey);

paraMap.put("sign", sign);

// 统一下单 https://api.mch.weixin.qq.com/pay/unifiedorder

String url = "https://api.mch.weixin.qq.com/pay/unifiedorder";

String xml = ArrayToXml(paraMap, false);

String xmlStr = HttpKit.post(url, xml);

// 预付商品id

String prepay_id = "";

if (xmlStr.indexOf("SUCCESS") != -1) {

Map map = doXMLParse(xmlStr);

prepay_id = (String) map.get("prepay_id");

}

Map payMap = new HashMap();

payMap.put("appId", appid);

payMap.put("timeStamp", create_timestamp());

payMap.put("nonceStr", create_nonce_str());

payMap.put("signType", "MD5");

payMap.put("package", "prepay_id=" + prepay_id);

String paySign = getSign(payMap, paternerKey);

payMap.put("pg", prepay_id);

payMap.put("paySign", paySign);

WebUtil.response(response, WebUtil.packJsonp(callback, JsonUtil.warpJsonNodeResponse(JsonUtil.objectToJsonNode(payMap)).toString()));

}

@RequestMapping(value = "appPay")

public void appPay(HttpServletRequest request, HttpServletResponse response, String body, String detail, String total_fee, String spbill_create_ip,

String notify_url, String trade_type, String callback) throws Exception {

String appid = "wx16691fcb0523c1a4";

String partnerid = "22223670";

String paternerKey = "fjfjfjfjf1234567FFFFFFFFF1234567";

String out_trade_no = getTradeNo();

Map paraMap = new HashMap();

paraMap.put("appid", appid);

paraMap.put("body", body);

paraMap.put("mch_id", partnerid);

paraMap.put("nonce_str", create_nonce_str());

paraMap.put("out_trade_no", out_trade_no);

paraMap.put("spbill_create_ip", spbill_create_ip);

paraMap.put("total_fee", total_fee);

paraMap.put("trade_type", trade_type);

paraMap.put("notify_url", notify_url);

String sign = getSign(paraMap, paternerKey);

paraMap.put("sign", sign);

// 统一下单 https://api.mch.weixin.qq.com/pay/unifiedorder

String url = "https://api.mch.weixin.qq.com/pay/unifiedorder";

String xml = ArrayToXml(paraMap, false);

String xmlStr = HttpKit.post(url, xml);

// 预付商品id

String prepay_id = "";

Map map = doXMLParse(xmlStr);

if (xmlStr.indexOf("SUCCESS") != -1) {

prepay_id = (String) map.get("prepay_id");

}

String result_code = map.get("result_code");

String err_code_des = map.get("err_code_des");

Map payMap = new HashMap();

payMap.put("appid", appid);

payMap.put("partnerid", partnerid);

payMap.put("prepayid", prepay_id);

payMap.put("package", "Sign=WXPay");

payMap.put("noncestr", create_nonce_str());

payMap.put("timestamp", create_timestamp());

String paySign = getSign(payMap, paternerKey);

payMap.put("sign", paySign);

payMap.put("result_code", result_code);

payMap.put("err_code_des", err_code_des);

WebUtil.response(response, WebUtil.packJsonp(callback, JsonUtil.warpJsonNodeResponse(JsonUtil.objectToJsonNode(payMap)).toString()));

}

@RequestMapping("/appPay_notify")

public void appPayNotify(HttpServletRequest request, HttpServletResponse response) throws Exception{

//String xml = "1";

response.setCharacterEncoding("UTF-8");

response.setContentType("text/xml");

ServletInputStream in = request.getInputStream();

String xmlMsg = Tools.inputStream2String(in);

Map map = doXMLParse(xmlMsg);

String return_code = map.get("return_code");

String return_msg = map.get("return_msg");

map = new HashMap();

map.put("return_code", return_code);

map.put("return_msg", return_msg);

//响应xml

String resXml = ArrayToXml(map, true);

response.getWriter().write(resXml);

}

@RequestMapping("/orderquery.do")

public void orderquery(HttpServletRequest request, HttpServletResponse response, String transaction_id, String out_trade_no, String callback) throws Exception{

String url = "https://api.mch.weixin.qq.com/pay/orderquery";

String appid = "wx16691fcb0523c1a4";

String partnerid = "22223670";

String paternerKey = "fjfjfjfjf1234567FFFFFFFFF1234567";

Map map = new HashMap();

map.put("appid", appid);

map.put("mch_id", partnerid);

if(transaction_id != null && !transaction_id.equals("")){

map.put("transaction_id", transaction_id);

}else {

map.put("out_trade_no", out_trade_no);

}

map.put("nonce_str", create_nonce_str());

String paySign = getSign(map, paternerKey);

map.put("sign", paySign);

String xml = ArrayToXml(map, false);

String xmlStr = HttpKit.post(url, xml);

Map orderMap = doXMLParse(xmlStr);

WebUtil.response(response, WebUtil.packJsonp(callback, JsonUtil.warpJsonNodeResponse(JsonUtil.objectToJsonNode(orderMap)).toString()));

}

/**

* map转成xml

*

* @param arr

* @return

*/

public String ArrayToXml(Map parm, boolean isAddCDATA) {

StringBuffer strbuff = new StringBuffer();

if (parm != null ) {

for (Entry entry : parm.entrySet()) {

strbuff.append("");

if (isAddCDATA) {

strbuff.append();

}else {

strbuff.append(entry.getValue());

}

strbuff.append("");

}

}

return strbuff.append().toString();

}

// 获取openId

private String getUserOpenId(HttpServletRequest request) throws Exception {

String code = request.getParameter("code");

if (code == null) {

String openId = request.getParameter("openId");

return openId;

}

Oauth o = new Oauth();

String token = o.getToken(code);

JsonNode node = JsonUtil.StringToJsonNode(token);

String openId = node.get("openid").asText();

return openId;

}

private String create_nonce_str() {

String chars = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789";

String res = "";

for (int i = 0; i < 16; i++) {

Random rd = new Random();

res += chars.charAt(rd.nextInt(chars.length() - 1));

}

return res;

}

private String getAddrIp(HttpServletRequest request){

return request.getRemoteAddr();

}

private String create_timestamp() {

return Long.toString(System.currentTimeMillis() / 1000);

}

private String getTradeNo(){

String timestamp = DatetimeUtil.formatDate(new Date(), DatetimeUtil.DATETIME_PATTERN);

return "HZNO" + timestamp;

}

private String getSign(Map params, String paternerKey )

throws UnsupportedEncodingException {

String string1 = Pay.createSign(params, false);

String stringSignTemp = string1 + "&key=" + paternerKey;

String signValue = DigestUtils.md5Hex(stringSignTemp).toUpperCase();

return signValue;

}

private Map doXMLParse(String xml)

throws XmlPullParserException, IOException {

InputStream inputStream = new ByteArrayInputStream(xml.getBytes());

Map map = null;

XmlPullParser pullParser = XmlPullParserFactory.newInstance()

.newPullParser();

pullParser.setInput(inputStream, "UTF-8"); // 为xml设置要解析的xml数据

int eventType = pullParser.getEventType();

while (eventType != XmlPullParser.END_DOCUMENT) {

switch (eventType) {

case XmlPullParser.START_DOCUMENT:

map = new HashMap();

break;

case XmlPullParser.START_TAG:

String key = pullParser.getName();

if (key.equals("xml"))

break;

String value = pullParser.nextText();

map.put(key, value);

break;

case XmlPullParser.END_TAG:

break;

}

eventType = pullParser.next();

}

return map;

}

}

wxprepay.shtm接口是预处理订单接口向微信服务器下订单。

appPay.shtml接口是支付接口。

appPay_notify.shtml接口是微信支付后异步通知结果接口。

orderquery.shtml接口是订单查询接口

3.3、涉及到的工具类SessionUtil.java工具类package org.andy.util;

import javax.servlet.http.HttpServletRequest;

public class SessionUtil {

public static void addAtt(HttpServletRequest request, String key, Object value){

request.getSession().setAttribute(key, value);

}

public static void removeAtt(HttpServletRequest request, String key){

request.getSession().removeAttribute(key);

}

public static String getAtt(HttpServletRequest request, String key){

return (String)request.getSession().getAttribute(key);

}

public static Object getAttObj(HttpServletRequest request, String key){

return request.getSession().getAttribute(key);

}

public static String optAtt(HttpServletRequest request, String key, String value){

String r = (String)request.getSession().getAttribute(key);

if (r == null){

r = value;

}

return r;

}

}

HttpKit 网络请求工具类/**

* https 请求 微信为https的请求

*

* @author andy

* @date 2015-10-9 下午2:40:19

*/

public class HttpKit {

private static final String DEFAULT_CHARSET = "UTF-8";

/**

* @return 返回类型:

* @throws IOException

* @throws UnsupportedEncodingException

* @throws NoSuchProviderException

* @throws NoSuchAlgorithmException

* @throws KeyManagementException

* @description 功能描述: get 请求

*/

public static String get(String url, Map params, Map headers) throws IOException, ExecutionException, InterruptedException {

AsyncHttpClient http = new AsyncHttpClient();

AsyncHttpClient.BoundRequestBuilder builder = http.prepareGet(url);

builder.setBodyEncoding(DEFAULT_CHARSET);

if (params != null && !params.isEmpty()) {

Set keys = params.keySet();

for (String key : keys) {

builder.addQueryParameter(key, params.get(key));

}

}

if (headers != null && !headers.isEmpty()) {

Set keys = headers.keySet();

for (String key : keys) {

builder.addHeader(key, params.get(key));

}

}

Future f = builder.execute();

String body = f.get().getResponseBody(DEFAULT_CHARSET);

http.close();

return body;

}

/**

* @return 返回类型:

* @throws IOException

* @throws UnsupportedEncodingException

* @throws NoSuchProviderException

* @throws NoSuchAlgorithmException

* @throws KeyManagementException

* @description 功能描述: get 请求

*/

public static String get(String url) throws KeyManagementException, NoSuchAlgorithmException, NoSuchProviderException, UnsupportedEncodingException, IOException, ExecutionException, InterruptedException {

return get(url, null);

}

/**

* @return 返回类型:

* @throws IOException

* @throws NoSuchProviderException

* @throws NoSuchAlgorithmException

* @throws KeyManagementException

* @throws UnsupportedEncodingException

* @description 功能描述: get 请求

*/

public static String get(String url, Map params) throws KeyManagementException, NoSuchAlgorithmException, NoSuchProviderException, UnsupportedEncodingException, IOException, ExecutionException, InterruptedException {

return get(url, params, null);

}

/**

* @return 返回类型:

* @throws IOException

* @throws NoSuchProviderException

* @throws NoSuchAlgorithmException

* @throws KeyManagementException

* @description 功能描述: POST 请求

*/

public static String post(String url, Map params) throws IOException, ExecutionException, InterruptedException {

AsyncHttpClient http = new AsyncHttpClient();

AsyncHttpClient.BoundRequestBuilder builder = http.preparePost(url);

builder.setBodyEncoding(DEFAULT_CHARSET);

if (params != null && !params.isEmpty()) {

Set keys = params.keySet();

for (String key : keys) {

builder.addParameter(key, params.get(key));

}

}

Future f = builder.execute();

String body = f.get().getResponseBody(DEFAULT_CHARSET);

http.close();

return body;

}

public static String post(String url, String s) throws IOException, ExecutionException, InterruptedException {

AsyncHttpClient http = new AsyncHttpClient();

AsyncHttpClient.BoundRequestBuilder builder = http.preparePost(url);

builder.setBodyEncoding(DEFAULT_CHARSET);

builder.setBody(s);

Future f = builder.execute();

String body = f.get().getResponseBody(DEFAULT_CHARSET);

http.close();

return body;

}

}

支付工具类pay.java/**

* 支付相关方法

* @author andy

*

*/

public class Pay {

// 发货通知接口

private static final String DELIVERNOTIFY_URL = "https://api.weixin.qq.com/pay/delivernotify?access_token=";

/**

* 参与 paySign 签名的字段包括:appid、timestamp、noncestr、package 以及 appkey。

* 这里 signType 并不参与签名微信的Package参数

* @param params

* @return

* @throws UnsupportedEncodingException

*/

public static String getPackage(Map params) throws UnsupportedEncodingException {

String partnerKey = ConfKit.get("partnerKey");

String partnerId = ConfKit.get("partnerId");

String notifyUrl = ConfKit.get("notify_url");

// 公共参数

params.put("bank_type", "WX");

params.put("attach", "yongle");

params.put("partner", partnerId);

params.put("notify_url", notifyUrl);

params.put("input_charset", "UTF-8");

return packageSign(params, partnerKey);

}

/**

* 构造签名

* @param params

* @param encode

* @return

* @throws UnsupportedEncodingException

*/

public static String createSign(Map params, boolean encode) throws UnsupportedEncodingException {

Set keysSet = params.keySet();

Object[] keys = keysSet.toArray();

Arrays.sort(keys);

StringBuffer temp = new StringBuffer();

boolean first = true;

for (Object key : keys) {

if (first) {

first = false;

} else {

temp.append("&");

}

temp.append(key).append("=");

Object value = params.get(key);

String valueString = "";

if (null != value) {

valueString = value.toString();

}

if (encode) {

temp.append(URLEncoder.encode(valueString, "UTF-8"));

} else {

temp.append(valueString);

}

}

return temp.toString();

}

/**

* @param params

* @param paternerKey

* @return

* @throws UnsupportedEncodingException

*/

private static String packageSign(Map params, String paternerKey) throws UnsupportedEncodingException {

String string1 = createSign(params, false);

String stringSignTemp = string1 + "&key=" + paternerKey;

String signValue = DigestUtils.md5Hex(stringSignTemp).toUpperCase();

String string2 = createSign(params, true);

return string2 + "&sign=" + signValue;

}

/**

* 支付签名

* @param timestamp

* @param noncestr

* @param packages

* @return

* @throws UnsupportedEncodingException

*/

public static String paySign(String timestamp, String noncestr,String packages) throws UnsupportedEncodingException {

Map paras = new HashMap();

paras.put("appid", ConfKit.get("AppId"));

paras.put("timestamp", timestamp);

paras.put("noncestr", noncestr);

paras.put("package", packages);

paras.put("appkey", ConfKit.get("paySignKey"));

// appid、timestamp、noncestr、package 以及 appkey。

String string1 = createSign(paras, false);

String paySign = DigestUtils.shaHex(string1);

return paySign;

}

/**

* 支付回调校验签名

* @param timestamp

* @param noncestr

* @param openid

* @param issubscribe

* @param appsignature

* @return

* @throws UnsupportedEncodingException

*/

public static boolean verifySign(long timestamp,

String noncestr, String openid, int issubscribe, String appsignature) throws UnsupportedEncodingException {

Map paras = new HashMap();

paras.put("appid", ConfKit.get("AppId"));

paras.put("appkey", ConfKit.get("paySignKey"));

paras.put("timestamp", String.valueOf(timestamp));

paras.put("noncestr", noncestr);

paras.put("openid", openid);

paras.put("issubscribe", String.valueOf(issubscribe));

// appid、appkey、productid、timestamp、noncestr、openid、issubscribe

String string1 = createSign(paras, false);

String paySign = DigestUtils.shaHex(string1);

return paySign.equalsIgnoreCase(appsignature);

}

/**

* 发货通知签名

* @param paras

* @return

* @throws UnsupportedEncodingException

*

* @参数 appid、appkey、openid、transid、out_trade_no、deliver_timestamp、deliver_status、deliver_msg;

*/

private static String deliverSign(Map paras) throws UnsupportedEncodingException {

paras.put("appkey", ConfKit.get("paySignKey"));

String string1 = createSign(paras, false);

String paySign = DigestUtils.shaHex(string1);

return paySign;

}

/**

* 发货通知

* @param access_token

* @param openid

* @param transid

* @param out_trade_no

* @return

* @throws IOException

* @throws NoSuchProviderException

* @throws NoSuchAlgorithmException

* @throws KeyManagementException

* @throws InterruptedException

* @throws ExecutionException

*/

public static boolean delivernotify(String access_token, String openid, String transid, String out_trade_no) throws IOException, ExecutionException, InterruptedException {

Map paras = new HashMap();

paras.put("appid", ConfKit.get("AppId"));

paras.put("openid", openid);

paras.put("transid", transid);

paras.put("out_trade_no", out_trade_no);

paras.put("deliver_timestamp", (System.currentTimeMillis() / 1000) + "");

paras.put("deliver_status", "1");

paras.put("deliver_msg", "ok");

// 签名

String app_signature = deliverSign(paras);

paras.put("app_signature", app_signature);

paras.put("sign_method", "sha1");

String json = HttpKit.post(DELIVERNOTIFY_URL.concat(access_token), JSONObject.toJSONString(paras));

if (StringUtils.isNotBlank(json)) {

JSONObject object = JSONObject.parseObject(json);

if (object.containsKey("errcode")) {

int errcode = object.getIntValue("errcode");

return errcode == 0;

}

}

return false;

}

}

流转化Tools.java工具类public final class Tools {

public static final String inputStream2String(InputStream in) throws UnsupportedEncodingException, IOException{

if(in == null)

return "";

StringBuffer out = new StringBuffer();

byte[] b = new byte[4096];

for (int n; (n = in.read(b)) != -1;) {

out.append(new String(b, 0, n, "UTF-8"));

}

return out.toString();

}

public static final boolean checkSignature(String token,String signature,String timestamp,String nonce){

List params = new ArrayList();

params.add(token);

params.add(timestamp);

params.add(nonce);

Collections.sort(params,new Comparator() {

@Override

public int compare(String o1, String o2) {

return o1.compareTo(o2);

}

});

String temp = params.get(0)+params.get(1)+params.get(2);

return SHA1.encode(temp).equals(signature);

}

}

相应前端数据工具WebUtil.java工具类public class WebUtil {

public static Object getSessionAttribute(HttpServletRequest req, String key) {

Object ret = null;

try {

ret = req.getSession(false).getAttribute(key);

} catch (Exception e) {

}

return ret;

}

public static void response(HttpServletResponse response, String result) {

try {

response.setContentType("application/json;charset=utf-8");

response.getWriter().write(result);

} catch (IOException e) {

e.printStackTrace();

}

}

public static void response(HttpServletResponse response, ResponseMessage result) {

try {

response.setContentType("application/json;charset=utf-8");

response.getWriter().write(JsonUtil.objectToJsonNode(result).toString());

} catch (Exception e) {

e.printStackTrace();

}

}

public static String packJsonp(String callback, String json) {

if (json == null) {

json = "";

}

if (callback == null || callback.isEmpty()) {

return json;

}

return callback + "&&" + callback + '(' + json + ')';

}

public static String packJsonp(String callback, ResponseMessage response) {

String json = null;

if (response == null) {

json = "";

} else {

json = JsonUtil.objectToJsonNode(response).toString();

}

if (callback == null || callback.isEmpty()) {

return json;

}

return callback + "&&" + callback + '(' + json + ')';

}

}

Json转换工具JsonUtil.javapublic class JsonUtil {

public static ObjectNode warpJsonNodeResponse(JsonNode obj){

ObjectNode objectNode=createObjectNode();

objectNode.put("code", 1);

objectNode.put("response", obj);

return objectNode;

}

public static JsonNode objectToJsonNode(Object obj){

try {

ObjectMapper objectMapper = new ObjectMapper();

String objJson=objectMapper.writeValueAsString(obj);

JsonNode jsonNode = objectMapper.readTree(objJson);

return jsonNode;

} catch (JsonProcessingException e) {

e.printStackTrace();

} catch (IOException e) {

// TODO Auto-generated catch block

e.printStackTrace();

}

return null;

}

}

四、微信H5调起支付

这个url需要后台实现,其实现功能如下:

1、接受微信服务器端发送的支付结果。

2、向微信服务器发送支付结果

具体 参考微信aip(http://pay.weixin.qq.com/wiki/doc/api/index.php?chapter=9_7)

具体代码如下:

4.1、授权向后台发起生成统一下订单页面wxrepay.jsp

String path = request.getContextPath();

String basePath = request.getScheme() + "://" + request.getServerName() + ":" + request.getServerPort() + path + "/";

long t = System.currentTimeMillis();

%>

微信公众号支付

商品

  • beacon 1分钱 1只
  • 测试支付信息

确定购买

var code = urlparameter("code");

function reppay(){

ajaxUtil({}, mainpath+"/pay/wxprepay.shtml?code=" + code, repay);

}

function repay(response){

var info = response;

var url = "wxpay?appId=" + info.appId + "&timeStamp=" +info.timeStamp + "&nonceStr=" + info.nonceStr +

"&pg=" +info.pg + "&signType=" +info.signType + "&paySign=" +info.paySign;

window.location.href= url + "&showwxpaytitle=1";

}

首先是请求服务端wxprepay.shml接口,后台向微信支付平台获取支付订单信息,返回前台,wxpay.jsp页面

4.2、确认支付页面

String path = request.getContextPath();

String basePath = request.getScheme() + "://" + request.getServerName() + ":" + request.getServerPort() + path + "/";

long t = System.currentTimeMillis();

%>

微信公众号支付

微信js支付测试

  • 测试支付信息

立即支付

var appId = urlparameter("appId");

var timeStamp = urlparameter("timeStamp");

var nonceStr = urlparameter("nonceStr");

var pg = urlparameter("pg");

var signType = urlparameter("signType");

var paySign = urlparameter("paySign");

function onBridgeReady(){

WeixinJSBridge.invoke(

'getBrandWCPayRequest', {

"appId" : appId, //公众号名称,由商户传入

"timeStamp": timeStamp, //时间戳,自1970年以来的秒数

"nonceStr" : nonceStr, //随机串

"package" : "prepay_id=" + pg,

"signType" : signType, //微信签名方式:

"paySign" : paySign //微信签名

},

function(res){

if(res.err_msg == "get_brand_wcpay_request:ok" ) {

alert("支付成功");

} // 使用以上方式判断前端返回,微信团队郑重提示:res.err_msg将在用户支付成功后返回 ok,但并不保证它绝对可靠。

}

);

}

function pay(){

if (typeof WeixinJSBridge == "undefined"){

if( document.addEventListener ){

document.addEventListener('WeixinJSBridgeReady', onBridgeReady, false);

}else if (document.attachEvent){

document.attachEvent('WeixinJSBridgeReady', onBridgeReady);

document.attachEvent('onWeixinJSBridgeReady', onBridgeReady);

}

}else{

onBridgeReady();

}

}

4.2、前台涉及到的工具类javascript工具类common.js,样式css.css就不贴了没意义。var path="wxpay";

var mainpath = "/wxpay";

var appid = "wx16691fcb0523c1a4";

var urlpre = "http://www.xxx.com/wxpay/page";

var urlhost = "http://www.xxx.com/";

$(document).ready(function(){

$(".refresher").click(function(){

refresh();

});

$("#goback").click(function(){

goback();

});

});

function popupMsg(msg){

alert(msg);

}

function printUtilViaGet(panel, requestdata, ajaxurl, printfunction){

$.ajax({

type: 'GET',

url: ajaxurl,

data: requestdata,

cache:false,

dataType:"json",

async: false,

success: function(response) {

if (response.code){

if (panel != null && panel.length > 0){

$(panel).html("");

if (printfunction != null)

$(panel).html(printfunction(response.response));

}

return true;

} else {

//alert(response.reason);

}

},

error: function(x, e) {

//alert("error", x);

},

complete: function(x) {

//alert("call complete");

}

});

return false;

}

function ajaxUtilViaGet(requestdata, ajaxurl, succFunction, failFunction){

$.ajax({

url: ajaxurl,

type: "GET",

dataType: "json",

cache:false,

data: requestdata,

async: false,

success: function(response) {

if (response.code){

if (succFunction != null)

succFunction(response.response);

} else {

if (failFunction != null)

failFunction(response.response);

}

},

error: function(x, e) {

//alert("error", x);

},

complete: function(x) {

}

});

return false;

}

function printUtil(panel, requestdata, ajaxurl, printfunction, ajaxasync) {

if (isEmpty(ajaxasync)) {

ajaxasync = false;

}

$.ajax({

type : 'POST',

url : ajaxurl,

data : requestdata,

cache : false,

dataType : "json",

async : ajaxasync,

success : function(response) {

if (response.code) {

if (panel != null && panel.length > 0) {

$(panel).html("");

if (printfunction != null)

$(panel).html(printfunction(response.response));

}

return true;

} else {

// alert(response.reason);

}

},

error : function(x, e) {

// alert("error", x);

},

complete : function(x) {

// alert("call complete");

}

});

return false;

}

function appendUtil(panel, requestdata, ajaxurl, printfunction, ajaxasync) {

if (isEmpty(ajaxasync)) {

ajaxasync = false;

}

$.ajax({

type : 'POST',

url : ajaxurl,

data : requestdata,

cache : false,

dataType : "json",

async : ajaxasync,

success : function(response) {

if (response.code) {

if (panel != null && panel.length > 0) {

if (printfunction != null)

$(panel).append(printfunction(response.response));

}

return true;

} else {

// alert(response.reason);

}

},

error : function(x, e) {

// alert("error", x);

},

complete : function(x) {

// alert("call complete");

}

});

return false;

}

function ajaxUtilAsync(requestdata, ajaxurl, succFunction, failFunction) {

$.ajax({

url : ajaxurl,

type : "POST",

dataType : "json",

cache : false,

data : requestdata,

async : true,

success : function(response) {

if (typeof response.code == "number") {

if (response.code > 0) {

if (succFunction != null)

succFunction(response.response);

} else {

if (failFunction != null)

failFunction(response.response);

}

} else {

if (response.result) {

if (succFunction != null)

succFunction(response.response);

} else {

if (failFunction != null)

failFunction(response.response);

}

}

},

error : function(x, e) {

// alert("error", x);

},

complete : function(x) {

}

});

return false;

}

function ajaxUtil(requestdata, ajaxurl, succFunction, failFunction){

$.ajax({

url: ajaxurl,

type: "POST",

dataType: "json",

cache:false,

data: requestdata,

async: false,

success: function(response) {

if (typeof response.code == "number"){

if (response.code > 0){

if (succFunction != null)

succFunction(response.response);

} else {

if (failFunction != null)

failFunction(response.response);

}

} else {

if (response.result){

if (succFunction != null)

succFunction(response.response);

} else {

if (failFunction != null)

failFunction(response.response);

}

}

},

error: function(x, e) {

//alert("error", x);

},

complete: function(x) {

}

});

return false;

}

function loadSelection(panel, requestdata, ajaxurl, itemName){

ajaxUtil(requestdata, ajaxurl, function(response){

var list = response.list;

for (var i = 0;i

$(panel).append(""+list[i][itemName]+"");

}

}, null);

}

function ajaxSubmitRefresh(formId) {

var hideForm = $(formId);

var options = {

dataType : "json",

beforeSubmit : function() {

},

success : function(result) {

if (result.result){

showMsg("提交成功");

} else {

alert("提交失败!");

}

},

error : function(result) {

alert("提交失败!");

}

};

hideForm.ajaxSubmit(options);

}

function ajaxSubmitWithJump(formId, nextPage) {

var hideForm = $(formId);

var options = {

dataType : "json",

beforeSubmit : function() {

},

success : function(result) {

if (result.result){

alert("提交成功");

window.location.href = nextPage;

} else {

alert("提交失败!");

}

},

error : function(result) {

alert("提交失败!");

}

};

hideForm.ajaxSubmit(options);

}

function refresh(){

window.location.href = window.location.href;

}

function goback(){

history.go(-1);

}

function urlparameter(paras){

var url = location.href;

var paraString = url.substring(url.indexOf("?")+1,url.length).split("&");

var paraObj = {};

for (var i=0; j=paraString[i]; i++){

paraObj[j.substring(0,j.indexOf("=")).toLowerCase()] = j.substring(j.indexOf("=")+1,j.length);

}

var returnValue = paraObj[paras.toLowerCase()];

if(typeof(returnValue)=="undefined"){

return "";

}else{

return returnValue;

}

}

String.prototype.endWith=function(str){

if(str==null||str==""||this.length==0||str.length>this.length)

return false;

if(this.substring(this.length-str.length)==str)

return true;

else

return false;

return true;

};

String.prototype.startWith=function(str){

if(str==null||str==""||this.length==0||str.length>this.length)

return false;

if(this.substr(0,str.length)==str)

return true;

else

return false;

return true;

};

function getFileUrl(sourceId) {

var url = "";

if (navigator.userAgent.indexOf("MSIE")>=1) { // IE

url = document.getElementById(sourceId).value;

} else if(navigator.userAgent.indexOf("Firefox")>0) { // Firefox

url = window.URL.createObjectURL(document.getElementById(sourceId).files.item(0));

} else if(navigator.userAgent.indexOf("Chrome")>0) { // Chrome

url = window.URL.createObjectURL(document.getElementById(sourceId).files.item(0));

}

return url;

}

function preImg(sourceId, targetId) {

var url = getFileUrl(sourceId);

var imgPre = document.getElementById(targetId);

imgPre.src = url;

}

function initWX(){

$.ajax({

url:mainpath+'/wechatjs.do',

type:'POST',

dataType:'json',

async: false,

data: {url:location.href.split('#')[0]},

success:function(result){

console.log(result);

var data=result['response']['map'];

if(result['code']==1){

wx.config({

debug: false,

appId:data['appId'],

timestamp:data['timestamp'],

nonceStr:data['nonceStr'],

signature:data['signature'],

jsApiList: ['onMenuShareTimeline','onMenuShareAppMessage','getLocation', 'onMenuShareQQ', 'onMenuShareWeibo']

});

}else{

alert("fail to get code");

window.alert('fail');

};

}

});

}

var EARTH_RADIUS = 6378137.0; //单位M

var PI = Math.PI;

function getRad(d){

return d*PI/180.0;

}

function getGreatCircleDistance(lat1,lng1,lat2,lng2){

var radLat1 = getRad(lat1);

var radLat2 = getRad(lat2);

var a = radLat1 - radLat2;

var b = getRad(lng1) - getRad(lng2);

var s = 2*Math.asin(Math.sqrt(Math.pow(Math.sin(a/2),2) + Math.cos(radLat1)*Math.cos(radLat2)*Math.pow(Math.sin(b/2),2)));

s = s*EARTH_RADIUS;

s = Math.round(s*10000)/10000.0;

s = Math.round(s);

return s;

}

//对Date的扩展,将 Date 转化为指定格式的String

//月(M)、日(d)、小时(h)、分(m)、秒(s)、季度(q) 可以用 1-2 个占位符,

//年(y)可以用 1-4 个占位符,毫秒(S)只能用 1 个占位符(是 1-3 位的数字)

//例子:

//(new Date()).Format("yyyy-MM-dd hh:mm:ss.S") ==> 2006-07-02 08:09:04.423

//(new Date()).Format("yyyy-M-d h:m:s.S") ==> 2006-7-2 8:9:4.18

Date.prototype.format = function(fmt)

{ //author: meizz

var o = {

"M+" : this.getMonth()+1, //月份

"d+" : this.getDate(), //日

"h+" : this.getHours(), //小时

"m+" : this.getMinutes(), //分

"s+" : this.getSeconds(), //秒

"q+" : Math.floor((this.getMonth()+3)/3), //季度

"S" : this.getMilliseconds() //毫秒

};

if(/(y+)/.test(fmt))

fmt=fmt.replace(RegExp.$1, (this.getFullYear()+"").substr(4 - RegExp.$1.length));

for(var k in o)

if(new RegExp("("+ k +")").test(fmt))

fmt = fmt.replace(RegExp.$1, (RegExp.$1.length==1) ? (o[k]) : (("00"+ o[k]).substr((""+ o[k]).length)));

return fmt;

};

//判断为空

function isEmpty(src){

if(("undefined" == typeof src) || (src == null) || ($.trim(src) == "") ){

return true;

}

return false;

}

//判断不为空

function notEmpty(src){

return !isEmpty(src);

}

//微信页面授权 snsapi_base方式

function wecharauto2burl(url) {

return "https://open.weixin.qq.com/connect/oauth2/authorize?appid=" + appid

+ "&redirect_uri=" + encodeURIComponent(url)

+ "&response_type=code&scope=snsapi_base&state=xybank#wechat_redirect";

}

//页面授权针对snsapi_base方式授权的url

function wecharauto2baseurl(url) {

return wecharauto2burl(urlpre+url);

}

//页面授权针对snsapi_userinfo方式授权的url

function wecharauto2userinfourl(url) {

return "https://open.weixin.qq.com/connect/oauth2/authorize?appid=" + appid

+ "&redirect_uri=" + encodeURIComponent(urlpre+url)

+ "&response_type=code&scope=snsapi_userinfo&state=xybank#wechat_redirect";

}

//微信分享 此方法需放在wx.ready中

function shareWeChat(title, link, imgUrl, desc){

wx.onMenuShareTimeline({

title: title, // 分享标题

link: link, // 分享链接

imgUrl: imgUrl, // 分享图标

success: function () {

// 用户确认分享后执行的回调函数

},

cancel: function () {

// 用户取消分享后执行的回调函数

}

});

//分享给朋友

wx.onMenuShareAppMessage({

title: title, // 分享标题

desc: desc, // 分享描述

link: link, // 分享链接

imgUrl: imgUrl, // 分享图标

type: 'link', // 分享类型,music、video或link,不填默认为link

dataUrl: '', // 如果type是music或video,则要提供数据链接,默认为空

success: function () {

// 用户确认分享后执行的回调函数

},

cancel: function () {

// 用户取消分享后执行的回调函数

}

});

//分享到QQ

wx.onMenuShareQQ({

title: title, // 分享标题

desc: desc, // 分享描述

link: link, // 分享链接

imgUrl: imgUrl, // 分享图标

success: function () {

// 用户确认分享后执行的回调函数

},

cancel: function () {

// 用户取消分享后执行的回调函数

}

});

//分享到腾讯微博

wx.onMenuShareWeibo({

title: title, // 分享标题

desc: desc, // 分享描述

link: link, // 分享链接

imgUrl: imgUrl, // 分享图标

success: function () {

// 用户确认分享后执行的回调函数

},

cancel: function () {

// 用户取消分享后执行的回调函数

}

});

}

五、支付结果

公众号调起效果如下:

742523589ebc4c100007d96591642377.png

支付成功后,微信服务器得到后台的Notify通知后,会发微信说明支付信息,支付凭证如下:

3f339fe2c2968d3a46115ab5f58e97e9.png

后续会全部更新微信app支付,微信支付退款,微信企业向个人付款,支付宝相关支付。而且会上传全部代码到csdn资源下载处,尽请关注。

以上就是本文的全部内容,希望对大家的学习有所帮助,更多相关内容请关注PHP中文网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/295208.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

这场戏里,到底是谁脏了?

1 让我表演个背包袱&#xff01;▼2 猝不及防的劈叉&#xff08;素材来源网络&#xff0c;侵删&#xff09;▼3 就很出戏&#xff08;素材来源网络&#xff0c;侵删&#xff09;▼4 意识流拔河很有感觉了▼5 纯爱&#xff08;素材来源网络&#xff0c;侵删&#xff09;▼6 …

Android之如何解决刚下载的Android studio(包括上面的菜单栏)乱码问题

这个问题是我的台式机第一次下载Andorid studio的时候出现的,这台台式机组装的,是之前玩ps的人用了的,感觉怪怪的,第一次安装Android studio不是很熟悉,出现了乱码,包括菜单栏目,只有部分图片了,没有全部图片了 解决这个问题的总共思路: 思路一: 请教了那…

关于高级导数的一个不等式估计

from: http://math.fudan.edu.cn/gdsx/XXYD.HTM

体验.NET Core使用IKVM对接Java

【导读】与第三方对接最麻烦的是语言不同&#xff0c;因语言不同内置实现相关标准加密算法还是略微有所差异对接单点登录场景再寻常不过&#xff0c;由于时间紧迫且对接方使用Java&#xff0c;所以留给我对接开发和联调的时间本就不多&#xff0c;于是乎&#xff0c;在熬夜发版…

对硕士而言,编制和稳定究竟有多重要?

随着考研大战如火如荼&#xff0c;2021研究生报考数目再创新高&#xff0c;越来越多的学生选择继续深造&#xff0c;进入更高的学府&#xff0c;为自己的学历镀金。然而现实是&#xff0c;很多的人毕业后并不一定比本科生拥有明显的优势&#xff0c;包括在体制内的工作&#xf…

无需Windbg | 使用VS 2019调试.NET程序的Crash异常

前言某台服务器上的IIS应用程序池&#xff0c;最近经常会自动关闭。查看服务器上的事件日志&#xff0c;发现在关闭时&#xff0c;w3p.exe抛出了stackoverflow异常。幸好&#xff0c;Windows自动帮我们抓取了Crash的dump文件&#xff1a;一般来说&#xff0c;我们会使用windbg来…

被夸了几十年,地球都要因为它变秃了,你还天天用它......

全世界只有3.14 % 的人关注了爆炸吧知识纸袋的流行不是环保要的结果模友们&#xff0c;“限塑令”在中国已经实行12年了&#xff0c;惊讶不&#xff1f;经过12年的努力&#xff0c;塑料袋终于不在大街上明目张胆的漫天飞舞。超模君相信大家都有一个共同认知&#xff1a;塑料袋污…

Dubbo与Zookeeper、SpringMVC整合和使用(负载均衡、容错)(转)

互联网的发展&#xff0c;网站应用的规模不断扩大&#xff0c;常规的垂直应用架构已无法应对&#xff0c;分布式服务架构以及流动计算架构势在必行&#xff0c;Dubbo是一个分布式服务框架&#xff0c;在这种情况下诞生的。现在核心业务抽取出来&#xff0c;作为独立的服务&…

孤掌难鸣-------堵水眼

序言: 一口水平井&#xff0c;井斜已经52度&#xff0c;方位在水平井的连线方向上。此段为增斜段&#xff0c;多次钻具组合的调整就为了正钻轨迹符合设计轨迹。然而在更换钻具组合下到底后&#xff0c;出现了两趟钻“堵水眼”的现象&#xff0c;第一次运气好顶开&#xff0c;而…

技术分享 | 一条神奇的曲线——贝塞尔曲线在前端的应用

源宝导读&#xff1a;在前端的开发中我们经常会遇到利用贝塞尔曲线帮助我们完成前端的动画和图形绘制&#xff0c;但是对其中的一些参数配置是一头雾水。本文将从贝塞尔曲线的原理讲起&#xff0c;由浅入深剖析一阶到多阶贝塞尔的实现原理&#xff0c;最后从三个方向来介绍它的…

女生来大姨妈该怎么哄她?

1 这捞人速度&#xff01;▼2 老师&#xff1a;我怀疑你在内涵我&#xff01;&#xff1f;&#xff08;素材来源网络&#xff0c;侵删&#xff09;▼3 就很出戏&#xff08;素材来源网络&#xff0c;侵删&#xff09;▼4 分手到底有多痛苦&#xff1f;▼5 家长会上看到的纸…

OAuth 2.1 的进化之路

背景2010年, OAuth 授权规范 1.0 (rfc 5849) 版本发布, 2年后, 更简单易用的 OAuth 2.0 规范发布&#xff08;rfc 6749&#xff09;, 这也是大家最熟悉并且在互联网上使用最广泛的版本, 在2012年的时候, iPhone 5 是全新的, 微软最新的浏览器还是 IE9, 单页面应用在当时还被称作…

中科院超牛的物理学家,摇滚界无人不知的“李白”,你一定听过他的歌!

全世界只有3.14 % 的人关注了爆炸吧知识一个热爱音乐的科研工作者最近&#xff0c;有不少模友在后台问&#xff0c;如何如何去平衡好学习/工作和兴趣爱好之间的关系&#xff1f;很多人会说这么简单的问题&#xff0c;为什么还要问&#xff1f;其实并不简单&#xff0c;在现实生…

Nginx源码安装及应用

一&#xff1a;Nginx简介&#xff1a; Nginx ("engine x") 是一个高性能的HTTP和反向代理服务器。在高连接并发的情况下&#xff0c;Nginx是Apache服务器不错的替代品。 Nginx作为负载均衡服务器&#xff1a;Nginx 既可以在内部直接支持Rails和PHP 程序对外进行服务&…

宝藏好物gRPCurl

gRPCurl简介gRPCurl[1]是一个与gRPC服务器交互的命令行工具&#xff0c;可认为是gRPC的curl工具。gRPCurl用于从命令行调用gRPC服务器支持的RPC方法&#xff0c;gRPC使用二进制编码(protobuf), 不能利用常规的curl工具(早期的curl版本还不支持HTTP/2)。1. gRPCurl工具接受json编…

你的输入法都暴露了些啥?

1 三哥对柠檬茶下手了要不要来一杯&#xff1f;▼2 能成功求婚的应该就是真爱了吧&#xff1f;▼3 气泡的花样玩法▼4 很正确&#xff1f;▼5 高手过招&#xff01;牛▼6 幽默中带着点死亡气息▼7 输入法&#xff1a;我暴露了一切▼你点的每个赞&#xff0c;我都认真当成了…

从 ThreadLocal 到 AsyncLocal

前些天跟大佬们在群里讨论如何在不使用构造函数&#xff0c;不增加方法参数的情况下把一个上下文注入到方法内部使用&#xff0c;得出的结论是 AsyncLocal 。感叹自己才疏学浅&#xff0c;居然才知道有 AsyncLocal 这种神器。于是赶紧恶补一下。ThreadLocal 要说 AsyncLocal 还…

Sersync+Rsync 增量实时同步

准备环境&#xff1a; rsync服务器&#xff1a;172.16.3.21sersync2服务器&#xff1a;172.16.3.23一、 rsync服务器先把原来的rsync包删掉yum remove rsync-2.6.8-3.1 -y,然后到http://rsync.samba.org/下载rsync包我用的包如下图所示利用securecrt上传本地[rootacong tongbu1…

浅谈RBF函数

所谓径向基函数 (Radial Basis Function 简称 RBF), 就是某种沿径向对称的标量函数。 通常定义为空间中任一点x到某一中心xc之间欧氏距离的单调函数 , 可记作 k(||x-xc||), 其作用往往是局部的 , 即当x远离xc时函数取值很小。 最常用的径向基函数是高斯核函数 ,形式为 k(||x-xc…

一个彻底改变世界的思想

全世界只有3.14 % 的人关注了爆炸吧知识因随机性的到来阔然开朗蒙特卡罗赌场蒙特卡罗&#xff08;Monte Carlo&#xff09;是摩纳哥公国&#xff08;Principality of Monaco&#xff09;的一座城市。摩纳哥公国坐落在法国的东南方&#xff0c;总面积为2.02平方公里&#xff0c;…