Android集成微信支付功能
作者:shineflowers 发布时间:2023-07-14 10:38:11
标签:Android,微信支付
准备工作这里就不说了,包括签约和申请APPID,附上微信开放平台APP开发步骤,不懂的同学可以参考这里:
https://pay.weixin.qq.com/wiki/doc/api/app/app.php?chapter=8_5
上面的步骤很详细,这里主要说下调起支付的注意事项。按照上面文档中说的商户服务器生成支付订单,先调用统一下单API生成预付单,获取到prepay_id后将参数再次签名传输给APP发起支付。
相关代码如下:
/**
* 商户服务器生成支付订单,先调用统一下单API(详见第7节)生成预付单,获取到prepay_id后将参数再次签名传输给APP发起支付。
*/
//商品描述
String body = "iphone6s";
//随机字符串
String nonce_str = ResourceUtil.createRandomString(32);
//通知地址
String notify_url = "http://www.weixin.qq.com/wxpay/pay.php";
//商户订单号
String out_trade_no = ResourceUtil.generateOutTradeNo(32);
//总金额(单位分)
int total_fee = 1;
String url = "https://api.mch.weixin.qq.com/pay/unifiedorder";
String sign = SignUtil.signByMD5("appid=" + Constants.APP_ID + "&body=" + body +
"&mch_id=" + Constants.MCH_ID + "&nonce_str=" + nonce_str + "¬ify_url=" + notify_url +
"&out_trade_no=" + out_trade_no + "&spbill_create_ip=127.0.0.1" +
"&total_fee=" + total_fee + "&trade_type=APP" + "&key=" + Constants.KEY).toUpperCase(Locale.getDefault());
//参数以xml格式传递
String entity = "<xml><appid>" + Constants.APP_ID + "</appid><mch_id>" + Constants.MCH_ID + "</mch_id><nonce_str>" + nonce_str +"</nonce_str><sign>" + sign +
"</sign><body>" + body + "</body><out_trade_no>" + out_trade_no + "</out_trade_no><total_fee>" + total_fee +
"</total_fee><spbill_create_ip>127.0.0.1</spbill_create_ip><notify_url>http://www.weixin.qq.com/wxpay/pay.php</notify_url><trade_type>APP</trade_type></xml>";
Log.d("entity", entity);
payButton.setEnabled(false);
Toast.makeText(PayActivity.this, "获取订单中...", Toast.LENGTH_SHORT).show();
byte[] buf = Util.httpPost(url, entity);
if (buf != null && buf.length > 0) {
String content = new String(buf);
Log.d("get server pay params:", content);
OrderResult orderResult = ResourceUtil.parseXml(new ByteArrayInputStream(content.getBytes()));
if (!TextUtils.equals(orderResult.getReturnCode(), "SUCCESS")) {
Toast.makeText(PayActivity.this, orderResult.getReturnMsg(), Toast.LENGTH_SHORT).show();
return;
}
if (!TextUtils.equals(orderResult.getResultCode(), "SUCCESS")) {
Toast.makeText(PayActivity.this, orderResult.getErrorDesc(), Toast.LENGTH_SHORT).show();
return;
}
//下单成功,调起支付
PayReq request = new PayReq();
request.appId = Constants.APP_ID;
request.partnerId = Constants.MCH_ID;
request.prepayId = orderResult.getPrepayId();
request.packageValue = "Sign=WXPay";
request.nonceStr = nonce_str;
String timeStamp = String.valueOf(System.currentTimeMillis() / 1000);
request.timeStamp = timeStamp;
request.sign = SignUtil.signByMD5("appid=" + Constants.APP_ID + "&noncestr=" + nonce_str + "&package=Sign=WXPay" +
"&partnerid=" + Constants.MCH_ID + "&prepayid=" + orderResult.getPrepayId() + "×tamp=" + timeStamp + "&key=" + Constants.KEY).toUpperCase(Locale.getDefault());
api.sendReq(request);
payButton.setEnabled(true);
}
}
});
相关参数说明在文档上都注明了,我这里面nonce_str和out_trade_no都是我随机生成的字符创,附上我的工具类,方便大家参考。
ResourceUtil.java
package com.xylpay.sdk.pay.uikit;
import java.io.IOException;
import java.io.InputStream;
import java.util.Random;
import org.xmlpull.v1.XmlPullParser;
import org.xmlpull.v1.XmlPullParserException;
import com.xylpay.sdk.pay.bean.OrderResult;
import android.util.Xml;
public class ResourceUtil {
/**
* 随机生成字符串
* @param length 字符串的长度
* @return 随机字符串
*/
public static String createRandomString(int length) {
String source = "0123456789abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ";
Random random = new Random();
StringBuilder builder = new StringBuilder();
for (int i = 0; i < length; i++) {
int position = random.nextInt(source.length());
builder.append(source.charAt(position));
}
return builder.toString();
}
public static String generateOutTradeNo(int n) {
StringBuilder builder = new StringBuilder();
Random random = new Random();
for (int i = 0; i < n; i++) {
builder.append(random.nextInt(10));
}
return builder.toString();
}
public static OrderResult parseXml(InputStream is) {
//PULL解析xml数据
XmlPullParser parser = Xml.newPullParser();
OrderResult orderResult = null;
try {
parser.setInput(is, "UTF-8");
int type = parser.getEventType();
while(type != XmlPullParser.END_DOCUMENT) {
switch (type) {
case XmlPullParser.START_DOCUMENT:
break;
case XmlPullParser.START_TAG:
if (parser.getName().equals("xml")) {
orderResult = new OrderResult();
} else if (parser.getName().equals("return_code")) {
orderResult.setReturnCode(parser.nextText());
} else if (parser.getName().equals("return_msg")) {
orderResult.setReturnMsg(parser.nextText());
} else if (parser.getName().equals("result_code")) {
orderResult.setResultCode(parser.nextText());
} else if (parser.getName().equals("err_code_des")) {
orderResult.setErrorDesc(parser.nextText());
} else if (parser.getName().equals("prepay_id")) {
orderResult.setPrepayId(parser.nextText());
}
break;
case XmlPullParser.END_TAG:
break;
}
type = parser.next();
}
} catch(XmlPullParserException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
return orderResult;
}
}
其中关于sign的生成,参数的顺序一定要严格按照上面的顺序加上key进行MD5加密,查看签名规范。
关于key的说明,这里的key是需要自己生成然后配置到微信开放平台的,参考商户支付密钥key的生成与设置进行配置,两边需要保持一致。另外,下单时,参数要以xml的格式来传递。
最后附上自己的签名算法:
SignUtil.java
package com.xylpay.sdk.pay.uikit;
import java.security.MessageDigest;
import java.security.NoSuchAlgorithmException;
/**
* Created by Jackie on 2016/2/15.
* * MD5加密
*/
public class SignUtil {
public static String signByMD5(String source) {
byte[] bytes = null;
try {
MessageDigest digest = MessageDigest.getInstance("MD5");
digest.update(source.getBytes()); //更新摘要
bytes = digest.digest(); //再通过执行诸如填充之类的最终操作完成哈希计算。在调用此方法之后,摘要被重置。
} catch (NoSuchAlgorithmException e) {
e.printStackTrace();
}
StringBuilder builder = new StringBuilder(bytes.length * 2);
for (byte b : bytes) {
/**
* 0xFF默认是整形,一个byte跟0xFF相与会先将那个byte转化成整形运算
*/
if ((b & 0xFF) < 0x10) { //如果为1位 前面补个0
builder.append("0");
}
builder.append(Integer.toHexString(b & 0xFF));
}
return builder.toString();
}
}
0
投稿
猜你喜欢
- 这篇文章主要介绍了Spring Bean初始化及销毁多种实现方式,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值
- 需要用到的知识:注解、AOP、ExpiringMap(带有有效期的映射)我们可以自定义注解,把注解添加到我们的接口上。定义一个切面,执行方法
- 前言Spring动态配置多数据源,即在大型应用中对数据进行切分,并且采用多个数据库实例进行管理,这样可以有效提高系统的水平伸缩性。而这样的方
- XmlTextWriter类允许你将XML写到一个文件中去。这个类包含了很多方法和属性,使用这些属性和方法可以使你更容易地处理XML。为了使
- 用Canvas画贝塞尔曲线,要画贝塞尔曲线首先了解贝塞尔曲线:由于用计算机画图大部分时间是操作鼠标来掌握线条的路径,与手绘的感觉和效果有很大
- JDK * ,代理接口没有实现类,实现 * JDK代理,代理的是接口,那么笔者想一想,既然代理的是接口,那如果没有实现类怎么办,能不能代
- 目录Spring自动注入失败如何解决?回答注入你的bean使用@Configurable手动查找bean:不推荐Spring自动注入失败如何
- 主要技术实现:spring、 springmvc、 redis、 springboot、 mybatis 、sessi
- Swing中的常用按钮在Swing中,常见的按钮组件有JButton,JCheckBox,JRadioButton等,它们都是抽象类Abst
- 前言本文主要介绍了关于JDK源码分析之String、StringBuilder和StringBuffer的相关内容,分享出来供大家参考学习,
- using System;using System.Collections.Generic;using System.Web.Script.
- 在实际项目中,在处理较大的文件时,常常将文件拆分为多个子文件进行处理,最后再合并这些子文件。下面就为各位介绍下Java中合并多个文件的方法。
- this总要有个事物来代表类的当前对象,就像C++中的this指针一样,Java中的this关键字就是代表当前对象的引用。它有三个主要的作用
- 前言在java里,当我们需要拷贝一个对象时,有两种类型的拷贝:浅拷贝与深拷贝。浅拷贝只是拷贝了源对象的地址,所以源对象的值发生变化时,拷贝对
- 界面开发中,经常使用观察者设计模式来实现文档/视图模式,当文档内容改变时,作为观察者的用户视图必须相应作出调整以向用户呈现文档的状态。由于语
- 使用ByteArrayOutputStream下载文件//文件名称String filepath = ServletActionContex
- Java 8的18个常用日期处理一、简介伴随lambda表达式、streams以及一系列小优化,Java 8 推出了全新的日期时间API。J
- RFC6749OAuth2的官方文档在RFC6749:https://datatracker.ietf.org/doc/html/rfc67
- JDK中的三个基本注解a、@Override:检查子类确实是覆盖了父类的方法。b、@Deprecated:说明已经过时了。c、@Suppre
- 本文实例为大家分享了用JavaMail发送HTML模板邮件的具体代码,供大家参考,具体内容如下依赖<dependency>&nb