Java第三方调用微信支付第三方服务商表结构设计

JAVA版微信支付V3-完全版
我的图书馆
JAVA版微信支付V3-完全版
本人用的开发框架是:struts2(用了struts2的0配置,对于struts的0配置不熟悉的可以看看这个博客了解下)
本人做的是微信V3版本的微信支付,也是目前最新的微信支付接口。官方文档下载地址
微信支付成功后&&你的邮件会有以下信息:
1、 信息包括:商户ID(mch_id)、申请编号、登录帐号、登录密码、商户API密码(key)
2.、证书包括:商户API证书、证书密钥、CA证书
开发前,我们先登录自己的服务号,点击微信支付-----&开发配置。
如果这里的授权路径和下面参数的notify_url不对&&调用付款接口时会弹出access_denied。
比如,我的授权目录是http://183.33.212.175/wxweb/config/,那么我对应的notify_url的回调方法必须是String notify_url = "http://183.33.212.175:8016/wxweb/config/pay!paySuccess.action";这样的
下图是微信支付JSAPI支付方法的流程图。
登陆Oauth2授权的问题,我会另外写篇博客。在此略过。需要注意一点的是,微信支付是微信5.0以上版本才能支持,所以,可能有些用户的微信版本是低版本无法进行微信支付而用户又不知道为什么就是不能支付,我们需要进行一个版本判断,如果用户的微信版本低于5.0就告知用户一些提示信息。那么,如何判断用户的微信版本是多少呢?我们可以通过这个方法可以获取一些信息然后进行判断:request.getHeader("user-agent")获取信息。
以 iPhone 版本为例,可以通过 user agent 可获取如下微信版本示例信息:
"Mozilla/5.0(CPU iphone OS 5_1_1 like Mac OS X) AppleWebKit/534.46(KHTML,like Geocko) Mobile/9B206 MicroMessenger/5.0"
其中 5.0 为用户安装的微信版本号,商户可以判定版本号是否高于或者等于 5.0。具体判断代码请留意后面代码。
流程图中,我们看到需要先调用一次统一接口,然后微信会返回一个prepayId给我们。那么我们先来调用统一接口。调用接口的URL地址:https://api.mch.weixin.qq.com/pay/unifiedorder,调用统一接口需要以下参数。
看到这些参数,不难理解需要传入什么。那么参数中的sign需要以下规则进行生成。
下面是我生成sign的方法。注意,参数是必填项的必须加进去。sign的生成规则是,除了上除列表中sign字段为所有的参数都可以参与sign的生成且参与生成sign的参数值不为空.
&& & * @author 李欣桦
&& & * @date 下午2:29:34
&& & * @Description:sign签名
&& & * @param characterEncoding 编码格式
&& & * @param parameters 请求参数
&& & * @return
&& &public static String createSign(String characterEncoding,SortedMap&Object,Object& parameters){
&& &&& &StringBuffer sb = new StringBuffer();
&& &&& &Set es = parameters.entrySet();
&& &&& &Iterator it = es.iterator();
&& &&& &while(it.hasNext()) {
&& &&& &&& &Map.Entry entry = (Map.Entry)it.next();
&& &&& &&& &String k = (String)entry.getKey();
&& &&& &&& &Object v = entry.getValue();
&& &&& &&& &if(null != v && !"".equals(v)
&& &&& &&& &&& &&& &&& !"sign".equals(k) && !"key".equals(k)) {
&& &&& &&& &&& &sb.append(k + "=" + v + "&");
&& &&& &&& &}
&& &&& &sb.append("key=" + ConfigUtil.API_KEY);
&& &&& &String sign = MD5Util.MD5Encode(sb.toString(), characterEncoding).toUpperCase();
MD5Util工具类中相关的方法
public class MD5Util {
&& &private static String byteArrayToHexString(byte b[]) {
&& &&& &StringBuffer resultSb = new StringBuffer();
&& &&& &for (int i = 0; i & b. i++)
&& &&& &&& &resultSb.append(byteToHexString(b[i]));
&& &&& &return resultSb.toString();
&& &private static String byteToHexString(byte b) {
&& &&& &int n =
&& &&& &if (n & 0)
&& &&& &&& &n += 256;
&& &&& &int d1 = n / 16;
&& &&& &int d2 = n % 16;
&& &&& &return hexDigits[d1] + hexDigits[d2];
&& &public static String MD5Encode(String origin, String charsetname) {
&& &&& &String resultString =
&& &&& &try {
&& &&& &&& &resultString = new String(origin);
&& &&& &&& &MessageDigest md = MessageDigest.getInstance("MD5");
&& &&& &&& &if (charsetname == null || "".equals(charsetname))
&& &&& &&& &&& &resultString = byteArrayToHexString(md.digest(resultString
&& &&& &&& &&& &&& &&& &.getBytes()));
&& &&& &&& &else
&& &&& &&& &&& &resultString = byteArrayToHexString(md.digest(resultString
&& &&& &&& &&& &&& &&& &.getBytes(charsetname)));
&& &&& &} catch (Exception exception) {
&& &&& &return resultS
&& &private static final String hexDigits[] = { "0", "1", "2", "3", "4", "5",
&& &&& &"6", "7", "8", "9", "a", "b", "c", "d", "e", "f" };
各位需要记住,我们发送给微信服务器的参数是xml格式的string,微信返回来的信息也是xml格式的string。
我们提交的正确数据应该是这样的。
&appid&wx0ec43b&/appid&
&attach&&![CDATA[att1]]&&/attach&
&body&&![CDATA[JSAPI 支付测试]]&&/body&
&device_info&1000&/device_info&
&mch_id&&/mch_id&
&nonce_str&bd1d9ed8b2&/nonce_str&
&notify_url&http://wxpay.weixin.qq.com/pub_v2/pay/notify.php&;/notify_url&
&out_trade_no&&/out_trade_no&
&spbill_create_ip&127.0.0.1&/spbill_create_ip&
&total_fee&1&/total_fee&
&trade_type&JSAPI&/trade_type&
&sign&&![CDATA[3CA89BCCF4]]&&/sign&
那么。我们来生成这样的post data
SortedMap&Object,Object& parameters = new TreeMap&Object,Object&();
parameters.put("appid", ConfigUtil.APPID);
parameters.put("mch_id", ConfigUtil.MCH_ID);
parameters.put("nonce_str", PayCommonUtil.CreateNoncestr());
parameters.put("body", "LEO测试");
parameters.put("out_trade_no", "");
parameters.put("total_fee", "1");
parameters.put("spbill_create_ip",IpAddressUtil.getIpAddr(request));
parameters.put("notify_url", ConfigUtil.NOTIFY_URL);
parameters.put("trade_type", "JSAPI");
parameters.put("openid", "o7W6yt9DUOBpjEYogs4by1hD_OQE");
String sign = PayCommonUtil.createSign("UTF-8", parameters);
parameters.put("sign", sign);
String requestXML = PayCommonUtil.getRequestXml(parameters);
PayCommonUtil.getRequestXml(parameters)方法如下:
&& & * @author 李欣桦
&& & * @date 下午2:32:05
&& & * @Description:将请求参数转换为xml格式的string
&& & * @param parameters& 请求参数
&& & * @return
&& &public static String getRequestXml(SortedMap&Object,Object& parameters){
&& &&& &StringBuffer sb = new StringBuffer();
&& &&& &sb.append("&xml&");
&& &&& &Set es = parameters.entrySet();
&& &&& &Iterator it = es.iterator();
&& &&& &while(it.hasNext()) {
&& &&& &&& &Map.Entry entry = (Map.Entry)it.next();
&& &&& &&& &String k = (String)entry.getKey();
&& &&& &&& &String v = (String)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();
最后我们要将这些参数以POST方式调用微信统一支付接口:代码如下
String result =CommonUtil.httpsRequest(ConfigUtil.UNIFIED_ORDER_URL, "POST", requestXML);
CommonUtil.httpsRequest(ConfigUtil.UNIFIED_ORDER_URL, "POST", requestXML);方法如下:
&& & * 发送https请求
&& & * @param requestUrl 请求地址
&& & * @param requestMethod 请求方式(GET、POST)
&& & * @param outputStr 提交的数据
&& & * @return 返回微信服务器响应的信息
&& &public static String httpsRequest(String requestUrl, String requestMethod, String outputStr) {
&& &&& &try {
&& &&& &&& &// 创建SSLContext对象,并使用我们指定的信任管理器初始化
&& &&& &&& &TrustManager[] tm = { new MyX509TrustManager() };
&& &&& &&& &SSLContext sslContext = SSLContext.getInstance("SSL", "SunJSSE");
&& &&& &&& &sslContext.init(null, tm, new java.security.SecureRandom());
&& &&& &&& &// 从上述SSLContext对象中得到SSLSocketFactory对象
&& &&& &&& &SSLSocketFactory ssf = sslContext.getSocketFactory();
&& &&& &&& &URL url = new URL(requestUrl);
&& &&& &&& &HttpsURLConnection conn = (HttpsURLConnection) url.openConnection();
&& &&& &&& &conn.setSSLSocketFactory(ssf);
&& &&& &&& &conn.setDoOutput(true);
&& &&& &&& &conn.setDoInput(true);
&& &&& &&& &conn.setUseCaches(false);
&& &&& &&& &// 设置请求方式(GET/POST)
&& &&& &&& &conn.setRequestMethod(requestMethod);
&& &&& &&& &conn.setRequestProperty("content-type", "application/x-www-form-urlencoded");
&& &&& &&& &// 当outputStr不为null时向输出流写数据
&& &&& &&& &if (null != outputStr) {
&& &&& &&& &&& &OutputStream outputStream = conn.getOutputStream();
&& &&& &&& &&& &// 注意编码格式
&& &&& &&& &&& &outputStream.write(outputStr.getBytes("UTF-8"));
&& &&& &&& &&& &outputStream.close();
&& &&& &&& &}
&& &&& &&& &// 从输入流读取返回内容
&& &&& &&& &InputStream inputStream = conn.getInputStream();
&& &&& &&& &InputStreamReader inputStreamReader = new InputStreamReader(inputStream, "utf-8");
&& &&& &&& &BufferedReader bufferedReader = new BufferedReader(inputStreamReader);
&& &&& &&& &String str =
&& &&& &&& &StringBuffer buffer = new StringBuffer();
&& &&& &&& &while ((str = bufferedReader.readLine()) != null) {
&& &&& &&& &&& &buffer.append(str);
&& &&& &&& &}
&& &&& &&& &// 释放资源
&& &&& &&& &bufferedReader.close();
&& &&& &&& &inputStreamReader.close();
&& &&& &&& &inputStream.close();
&& &&& &&& &inputStream =
&& &&& &&& &conn.disconnect();
&& &&& &&& &return buffer.toString();
&& &&& &} catch (ConnectException ce) {
&& &&& &&& &log.error("连接超时:{}", ce);
&& &&& &} catch (Exception e) {
&& &&& &&& &log.error("https请求异常:{}", e);
接着,我们来解析微信返回给我们的信息。返回信息是xml格式的String
XMLUtil.工具类如下:
public class XMLUtil {
&& & * 解析xml,返回第一级元素键值对。如果第一级元素有子节点,则此节点的值是子节点的xml数据。
&& & * @param strxml
&& & * @return
&& & * @throws JDOMException
&& & * @throws IOException
&& &public static Map doXMLParse(String strxml) throws JDOMException, IOException {
&& &&& &strxml = strxml.replaceFirst("encoding=\".*\"", "encoding=\"UTF-8\"");
&& &&& &if(null == strxml || "".equals(strxml)) {
&& &&& &&& &
&& &&& &Map m = new HashMap();
&& &&& &InputStream in = new ByteArrayInputStream(strxml.getBytes("UTF-8"));
&& &&& &SAXBuilder builder = new SAXBuilder();
&& &&& &Document doc = builder.build(in);
&& &&& &Element root = doc.getRootElement();
&& &&& &List list = root.getChildren();
&& &&& &Iterator it = list.iterator();
&& &&& &while(it.hasNext()) {
&& &&& &&& &Element e = (Element) it.next();
&& &&& &&& &String k = e.getName();
&& &&& &&& &String v = "";
&& &&& &&& &List children = e.getChildren();
&& &&& &&& &if(children.isEmpty()) {
&& &&& &&& &&& &v = e.getTextNormalize();
&& &&& &&& &} else {
&& &&& &&& &&& &v = XMLUtil.getChildrenText(children);
&& &&& &&& &}
&& &&& &&& &
&& &&& &&& &m.put(k, v);
&& &&& &//关闭流
&& &&& &in.close();
&& & * 获取子结点的xml
&& & * @param children
&& & * @return String
&& &public static String getChildrenText(List children) {
&& &&& &StringBuffer sb = new StringBuffer();
&& &&& &if(!children.isEmpty()) {
&& &&& &&& &Iterator it = children.iterator();
&& &&& &&& &while(it.hasNext()) {
&& &&& &&& &&& &Element e = (Element) it.next();
&& &&& &&& &&& &String name = e.getName();
&& &&& &&& &&& &String value = e.getTextNormalize();
&& &&& &&& &&& &List list = e.getChildren();
&& &&& &&& &&& &sb.append("&" + name + "&");
&& &&& &&& &&& &if(!list.isEmpty()) {
&& &&& &&& &&& &&& &sb.append(XMLUtil.getChildrenText(list));
&& &&& &&& &&& &}
&& &&& &&& &&& &sb.append(value);
&& &&& &&& &&& &sb.append("&/" + name + "&");
&& &&& &&& &}
&& &&& &return sb.toString();
Map&String, String& map = XMLUtil.doXMLParse(result);//解析微信返回的信息,以Map形式存储便于取值
现在我们可以拿到想要拿的值,那么可以调用微信H5网页端调用支付接口。在微信浏览器里面打开 H5 网页中执行 JS 调起支付。接口输入输出数据格式为 JSON。
注意:WeixinJSBridge 内置对象在其他浏览器中无效;列表中参数名区分大小。
现在我们将返回的值传入到支付jsp页面,在支付jsp页面调用支付接口。
SortedMap&Object,Object& params = new TreeMap&Object,Object&();
&& &&& &params.put("appId", "wx0953bae287adfeee");
&& &&& &params.put("timeStamp", Long.toString(new Date().getTime()));
&& &&& &params.put("nonceStr", PayCommonUtil.CreateNoncestr());
&& &&& &params.put("package", "prepay_id="+map.get("prepay_id"));
&& &&& &params.put("signType", ConfigUtil.SIGN_TYPE);
&& &&& &String paySign =& PayCommonUtil.createSign("UTF-8", params);
&& &&& &params.put("packageValue", "prepay_id="+map.get("prepay_id"));&&& //这里用packageValue是预防package是关键字在js获取值出错
&& &&& &params.put("paySign", paySign);&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&& //paySign的生成规则和Sign的生成规则一致
&& &&& &params.put("sendUrl", ConfigUtil.SUCCESS_URL);&&&&&&&&&&&&&&&&&&&&&&&&&&&&&& //付款成功后跳转的页面
&& &&& &String userAgent = request.getHeader("user-agent");
&& &&& &char agent = userAgent.charAt(userAgent.indexOf("MicroMessenger")+15);
&& &&& &params.put("agent", new String(new char[]{agent}));//微信版本号,用于前面提到的判断用户手机微信的版本是否是5.0以上版本。
&& &&& &String json = JSONObject.fromObject(params).toString();
下面是一个简洁的支付页面。代码如下:
[html] &body&&&&&&&&form&action=""&method="post"&&&&&&&&&&&&&input&type="button"&value="确认支付"&name="ajaxLoadId"&id="test"/&&&&&&&&/form&&&&&&&&script&type="text/javascript"&&&&&&&var&basePath&=&"&%=basePath%&";&&&&&&$("#test").one("click",function(){&&&&&&&&&&$.ajax({&&&&&&&&&&&&&&url:basePath+"config/pay!execute.action"&&&&&&&&&&&&//&span&style="font-family:微软雅黑;"&ajax调用微信统一接口获取prepayId&/span&&&&&&&&&&&}).done(function(data){&&&&&&&&&&&&&&var&obj&=&eval('('&+&data&+&')');&&&&&&&&&&&&&&if(parseInt(obj.agent)&5){&&&&&&&&&&&&&&&&&&alert("您的微信版本低于5.0无法使用微信支付");&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&}&&&&&&&&&&&&&&WeixinJSBridge.invoke('getBrandWCPayRequest',{&&&&&&&&&&&&&&&&&&"appId"&:&obj.appId,&&&&&&&&&&&&&&&&&&//公众号名称,由商户传入&&&&&&&&&&&&&&&&&&"timeStamp":obj.timeStamp,&&&&&&&&&&//时间戳,自&1970&年以来的秒数&&&&&&&&&&&&&&&&&&"nonceStr"&:&obj.nonceStr,&&&&&&&&&//随机串&&&&&&&&&&&&&&&&&&"package"&:&obj.packageValue,&&&&&&//&span&style="font-family:微软雅黑;"&商品包信息&/span&&&&&&&&&&&&&&&&&&&"signType"&:&obj.signType,&&&&&&&&//微信签名方式:&&&&&&&&&&&&&&&&&&"paySign"&:&obj.paySign&&&&&&&&&&&//微信签名&&&&&&&&&&&&&&&&&&},function(res){&&&&&&&&&&&&&&&&&&&&&&&&&&alert(res.err_msg);&&&&&&&&&&&&&&&&&&if(res.err_msg&==&"get_brand_wcpay_request:ok"&)&{&&&&&&&&&&&&&&&&&&&&&&window.location.href=obj.sendU&&&&&&&&&&&&&&&&&&}else{&&&&&&&&&&&&&&&&&&&&&&alert("fail");&&&&&&&&&&&&&&&&&&&&&&window.location.href="http://183.45.18.197:8016/wxweb/config/oauth!execute.action";&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&&//&span&style="font-family:微软雅黑;"&当失败后,继续跳转该支付页面让用户可以继续付款,贴别注意不能直接调转jsp,&/span&&span&style="font-size:10.5pt"&不然会报&/span&&span&style="font-size:12.0pt"&&system:access_denied。&/span&&&&&&&&&&&&&&&&&&&}&&&&&&&&&&&&&&});&&&&&&&&&&&&&&&&&&&&&&&&});&&&&&&});&&&&&&&/script&&&&&&/body&&&支付成功后,微信就会调用你填写的notify_url的方法,本人微信支付的开发配置中说明了我的notify_url为http://183.33.212.175:8016/wxweb/config/pay!paySuccess.action
对后台通知交互时,如果微信收到商户的应答不是成功或超时,微信认为通知失败,微信会通过一定的策略(如
30 分钟共 8 次)定期重新发起通知,尽可能提高通知的成功率,
但微信不保证通知最终能成功。由于存在重新収送后台通知的情况,因此同样的通知可能会多次収送给商户系统。 商户系统必须能够正确处理重复的通知。
推荐的做法是,当收到通知进行处理时,首先检查对应业务数据的状态,判断该通知是否已经处理过,如果没有处理过再进行处理,如果处理过直接返回结果成功。在对业务数据
进行状态检查和处理之前,要采用数据锁进行幵収控制,以避免凼数重入造成的数据混乱。
判断完成后,我们需要通知微信,我们收到信息了,不然微信就会通过一定的策略定期重新发起通知。
怎么通知微信呢。方法如下。
void paySuccess() throws Exception{
&& &&& &HttpServletRequest request = ServletActionContext.getRequest();
&& &&& &HttpServletResponse response = ServletActionContext.getResponse();
&& &&& &InputStream inStream = request.getInputStream();
&& &&& &ByteArrayOutputStream outSteam = new ByteArrayOutputStream();
&& &&& &byte[] buffer = new byte[1024];
&& &&& &int len = 0;
&& &&& &while ((len = inStream.read(buffer)) != -1) {
&& &&& &&& &outSteam.write(buffer, 0, len);
&& &&& &System.out.println("~~~~~~~~~~~~~~~~付款成功~~~~~~~~~");
&& &&& &outSteam.close();
&& &&& &inStream.close();
&& &&& &String result& = new String(outSteam.toByteArray(),"utf-8");//获取微信调用我们notify_url的返回信息
&& &&& &Map&Object, Object& map = XMLUtil.doXMLParse(result);
&& &&& &for(Object keyValue : map.keySet()){
&& &&& &&& &System.out.println(keyValue+"="+map.get(keyValue));
&& &&& &if (map.get("result_code").toString().equalsIgnoreCase("SUCCESS")) {
&& &&& &&& &//TODO 对数据库的操作
&& &&& &&& &response.getWriter().write(PayCommonUtil.setXML("SUCCESS", ""));&& //告诉微信服务器,我收到信息了,不要在调用回调action了
&& &&& &&& &System.out.println("-------------"+PayCommonUtil.setXML("SUCCESS", ""));
PayCommonUtil.setXML("SUCCESS",
""))方法如下:
static String setXML(String return_code, String return_msg) {
& && &&&return "&xml&&return_code&&![CDATA[" + return_code
& && && && && & + "]]&&/return_code&&return_msg&&![CDATA[" + return_msg
& && && && && & + "]]&&/return_msg&&/xml&";
上面代码中工具类打包下载地址:http://download.csdn.net/detail/u54883
官方文档有少许常见错误解决方法,观众们可自行下载查阅,仍有不懂的可留言。
第一次写这么长的博客,不足之处请留言告之便于本人成长,再次感谢你的阅读。
TA的最新馆藏[转]&[转]&[转]&[转]&[转]&
喜欢该文的人也喜欢java支付接口开发原理
java支付接口开发原理。其实这些在线支付的接口从原理上来讲都是一样的。这里简单地阐述一下java支付接口开发原理:首先,开发这些接口的第三方公司会给商户提供一个唯一标识pid,而且与该标识相对应的会有一个KEY,双方使用这个KEY作为加密的密钥。
其次,商户使用java支付接口的时候,根据不同支付方的开发文档的要求,将所需要的参数拼接成相应的形式。在发送请求之前需要对数据进行加密,而加密的方式会由支付方提供,文档中会有相应的描述,有的java支付接口也会提供现成的方法供商户使用。加密后就可以调用接口向支付网关发送请求了。&
最后,是对于java支付接口结果的处理。支付方在对相应的资金进行处理后会给发送请求的商户网站发送响应。这里有两种方式:1、页面通知,用来显示给实际支付的客户,支付方只返回一次。2、服务器通知,运行在后台,实际在线支付的客户看不到,支付方会在一定时期内发送多次,直到收到商户程序反馈的约定好的标识为止。
已投稿到:
以上网友发言只代表其个人观点,不代表新浪网的观点或立场。Java编程调用微信支付功能的方法详解
转载 & & 作者:qq_
这篇文章主要介绍了Java编程调用微信支付功能的方法,结合实例形式详细分析了java微信支付功能的原理、操作流程及相关实现技巧,需要的朋友可以参考下
本文实例讲述了Java编程调用微信支付功能的方法。分享给大家供大家参考,具体如下:
微信开发文档地址:
从调用处开始
我的流程: 1.点击“支付”按钮,去后台 —-& 2.后台生成支付所需数据返回页面 —-& 3.页面点击“确认支付”调用微信支付js。完成支付功能。
&div class="button" id="pay" onclick="payBox()"&支付&/div&
支付按钮js
function payBox(){
//获得支付的钱数
var money = $(".money input").val();
//后台路径,加上参数
location.href = "/XXX/XXX/XXXX/XXXX?money ="+
后台方法(例:index())
getPara( ) == request.getParameter(name);
setAttr( ) == request.setAttribute(name, value);
render() == 我现在所用框架返回页面的一种方法。
首先得OpenId;下面是具体方法。
public String getOpenId(){
String code = getPara("code");
String openid = "";
if (StringUtils.isEmpty(openid) && !StringUtils.isEmpty(code)) {
SnsAccessToken token = SnsAccessTokenApi.getSnsAccessToken("你的APPID","你的appsecret密码",
openid = token.getOpenid();
getSession().setAttribute("openandid",openid);
public void index() throws Exception{
String openid = getOpenId();
//得到金额
String money= getPara("money");
Map&String ,String & map=new HashMap&String,String&();
//获取随机串
String nonceStr=UUID.randomUUID().toString().substring(0, 32);
//可以是支付物品的订单号。一个号码,看自己怎么给
String out_trade_no="";
//支付金额。微信默认支付是(1=0.01)的比例,下面是将金额换算成微信可识别的
BigDecimal re1=new BigDecimal(expressCharge);
BigDecimal re2=new BigDecimal(Float.toString(100.00f));
Float aa = re1.multiply(re2).floatValue();
String total_fee = String.valueOf(aa);
String[] smill = total_fee.split("\\.");
total_fee = smill[0];
//微信的appid
String appid="XXXXXXXXXXXXXXXXX";
String mch_id="XXXXXXXXX";//商户号
String body="xxxxxxx";//商品信息,可以自己起最好写英文
//密匙,商户平台的支付API密匙,注意是商户平台,不是微信平台
String key = "XXXXXXXXXXXXXXXXXXXXXXXXXXXX";
long timestamp = System.currentTimeMillis() / 1000;
map.put("appid", appid );
map.put("mch_id", mch_id);
map.put("nonce_str",nonceStr);
map.put("body", body);
map.put("out_trade_no", out_trade_no);
map.put("total_fee", total_fee);
map.put("spbill_create_ip",getRequest().getRemoteAddr());
//这里是支付成功后返回的地址,微信会以XML形式放回数据,就是本篇文章的下一类(例:wxxml())方法名。
map.put("notify_url", "http://www.XXXX.com/XXXX/XXXX/xxxx/wxxml");
map.put("trade_type", "JSAPI");
map.put("openid", openid);//传入OpenId
//这里传入Map集合和key商户支付密匙
String paySign=getPayCustomSign(map,key);
map.put("sign",paySign);
//将map转为XML格式
String xml= ArrayToXml(map);
//统一下单,这里不用改
String url = "https://api.mch.weixin.qq.com/pay/unifiedorder";
String xmlStr = HttpKit.post(url, xml);
//prepayid由微信返回的 。
String prepayid = "";
if (xmlStr.indexOf("SUCCESS") != -1) {
Map&String, String& map2 = doXMLParse(xmlStr);
prepayid = (String) map2.get("prepay_id");
String paySign2=getPayCustomSign(signMap,key);
setAttr("model", model);
setAttr("appId", appid);
setAttr("paytimestamp", String.valueOf(timestamp));
setAttr("paynonceStr", nonceStr);
setAttr("paypackage", "prepay_id="+prepayid);
setAttr("paysignType","MD5");
setAttr("paySign", paySign2);
//去到确认支付页面,返回页面方式不同,(例:pay.html页面),下面
render("/XXXX/pay.html");
* 获取支付所需签名
* @param ticket
* @param timeStamp
* @param card_id
* @param code
* @throws Exception
public static String getPayCustomSign(Map&String, String& bizObj,String key) throws Exception {
String bizString = FormatBizQueryParaMap(bizObj, false);
return sign(bizString, key);
* 字典排序
* @param paraMap
* @param urlencode
* @throws Exception
public static String FormatBizQueryParaMap(Map&String, String& paraMap,
boolean urlencode) throws Exception {
String buff = "";
List&Map.Entry&String, String&& infoIds = new ArrayList&Map.Entry&String, String&&(paraMap.entrySet());
Collections.sort(infoIds,
new Comparator&Map.Entry&String, String&&() {
public int compare(Map.Entry&String, String& o1,
Map.Entry&String, String& o2) {
return (o1.getKey()).toString().compareTo(
o2.getKey());
for (int i = 0; i & infoIds.size(); i++) {
Map.Entry&String, String& item = infoIds.get(i);
//System.out.println(item.getKey());
if (item.getKey() != "") {
String key = item.getKey();
String val = item.getValue();
if (urlencode) {
val = URLEncoder.encode(val, "utf-8");
buff += key + "=" + val + "&";
if (buff.isEmpty() == false) {
buff = buff.substring(0, buff.length() - 1);
} catch (Exception e) {
throw new Exception(e.getMessage());
//支付所需签名处调用此方法
public static String sign(String content, String key)
throws Exception{
String signStr = "";
signStr = content + "&key=" +
return MD5(signStr).toUpperCase();
//上一方法,MD5加密处理
public final static String MD5(String s) {
char hexDigits[]={'0','1','2','3','4','5','6','7','8','9','A','B','C','D','E','F'};
byte[] btInput = s.getBytes();
MessageDigest mdInst = MessageDigest.getInstance("MD5");
mdInst.update(btInput);
byte[] md = mdInst.digest();
int j = md.
char str[] = new char[j * 2];
int k = 0;
for (int i = 0; i & i++) {
byte byte0 = md[i];
str[k++] = hexDigits[byte0 &&& 4 & 0xf];
str[k++] = hexDigits[byte0 & 0xf];
return new String(str);
} catch (Exception e) {
e.printStackTrace();
//转为XML格式
public static String ArrayToXml(Map&String, String& arr) {
String xml = "&xml&";
Iterator&Entry&String, String&& iter = arr.entrySet().iterator();
while (iter.hasNext()) {
Entry&String, String& entry = iter.next();
String key = entry.getKey();
String val = entry.getValue();
if (IsNumeric(val)) {
xml += "&" + key + "&" + val + "&/" + key + "&";
xml += "&" + key + "&&![CDATA[" + val + "]]&&/" + key + "&";
xml += "&/xml&";
public static boolean IsNumeric(String str) {
if (str.matches("\\d *")) {
private Map&String, String& doXMLParse(String xml)
throws XmlPullParserException, IOException {
InputStream inputStream = new ByteArrayInputStream(xml.getBytes());
Map&String, String& map =
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&String, String&();
case XmlPullParser.START_TAG:
String key = pullParser.getName();
if (key.equals("xml"))
String value = pullParser.nextText();
map.put(key, value);
case XmlPullParser.END_TAG:
eventType = pullParser.next();
pay页面(上面步骤执行完后去的页面)
此处是页面js代码,接受后台代码传回来的参数。现在用的是BSL模板引擎,参数可以以EL表达式方式接收。可先将后台传会的参数,放在几个input类型type=”hidden”标签标签中。
&input type="hidden" name="appId" value="${appId}" id="appid" /&
js中得到值
var appid = $("#appid").val();
&script type="text/javascript" src="${staticPath}/front/js/weixin.js"&&/script&
下面是JS代码,由于是bsl,自己看着传参数吧,反正都是后台来的。
&div class="button" id="onlinePayNow"&确认支付&/div&
//先写一个点击事件,当点击id为onlinePayNow的按钮时,触发该事件。
$("#onlinePayNow").click(function getpay(){
if (typeof WeixinJSBridge=="undefined") {
if (document.addEventListener) {document.addEventListener('WeixinJSBridgeReady',onBridgeReady,false);
}else if(document.attachEvent){document.attachEvent('WeixinJSBridgeReady',onBridgeReady);document.attachEvent('onWeixinJSBridgeReady',onBridgeReady);
//如果报错,可用下面方法看看是不是参数缺少。
/* alert('${appId}');
alert('${paytimestamp}');
alert('${paynonceStr}');
alert('${paypackage}');
alert('${paysignType}');
alert('${paySign}'); */
//调用下面方法。开启微信支付。
onBridgeReady();
function onBridgeReady(){
WeixinJSBridge.invoke('getBrandWCPayRequest', {
"appId" : '${appId}', //公众号名称,由商户传入
"timeStamp" : '${paytimestamp}', //时间戳,自1970年以来的秒数
"nonceStr" : '${paynonceStr}', //随机串
"package" : '${paypackage}',
"signType" : '${paysignType}', //微信签名方式:
"paySign" : '${paySign}' //微信签名
}, function(res) {
//alert(res.err_msg); // 使用以上方式判断前端返回,微信团队郑重提示:res.err_msg将在用户支付成功后返
if(res.err_msg == "get_brand_wcpay_request:ok"){
//支付成功,完成后去到哪个页面。
window.location.href="/XXXX/xxxx.html" rel="external nofollow" ;
在微信公众平台配置,支付授权目录。
授权目录建议:
http://www.XXXX.com/XXXX/xxx/index/
我觉得最好写后台是action地址就写Action地址,Controller就写Controller地址,如果有Spring注解,就写注解后名称。
我所导入的包(java后台,就是index方法。)
import java.io.ByteArrayInputS
import java.io.IOE
import java.io.InputS
import java.math.BigD
import java.net.URLE
import java.security.MessageD
import java.util.ArrayL
import java.util.C
import java.util.C
import java.util.HashM
import java.util.I
import java.util.L
import java.util.M
import java.util.Map.E
import java.util.UUID;
import org.xmlpull.v1.XmlPullP
import org.xmlpull.v1.XmlPullParserE
import org.xmlpull.v1.XmlPullParserF
import com.jfinal.kit.HttpK
import com.uitrs.express.common.C
更多关于java算法相关内容感兴趣的读者可查看本站专题:《》、《》、《》、《》和《》
希望本文所述对大家java程序设计有所帮助。
您可能感兴趣的文章:
大家感兴趣的内容
12345678910
最近更新的内容
常用在线小工具

我要回帖

更多关于 第三方h5调用微信支付 的文章

 

随机推荐