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();
}
}


猜你喜欢
- 一、在java中遍历一个文件夹里边的所有文件,可以有两种方式:1.递归遍历,通常也是开发者第一时间能想到的方法,递归遍历的优点是:实现起来相
- 目录一、使用BeanFactoryPostProcessor注入Bean:第一步:创建实现SpringUtils 接口工具(组件)来获取sp
- 前言多线程是我们开发过程中经常遇到的,也是必不可少需要掌握的。当我们知道需要进行多线程开发时首先需要知道的自然是如何实现多线程,也就是我们应
- 我们经常在项目开放中需要进行很多配置, 那么这些配置基本上都是动态的, 如果我直接写在代码中, 修改起来很麻烦, 如果该配置在多处进行引用啦
- 传统的单例写法解决了什么问题首先,在大多数情况下(不包含面试),传统的单例写法已经完全够用了。通过 synchronized 关键字解决了多
- 本文实例讲述了C#数据结构之堆栈(Stack)。分享给大家供大家参考,具体如下:堆栈(Stack)最明显的特征就是“先进后出”,本质上讲堆栈
- 一、说明1.spring aop中的 * 主要有两种方式,jdk * 和cglib * 2.从实现接口、继承父类的角度讨论区别3.从限
- 判断JSONObject是否存在某个KeyJSONObject jsonObj = new JSONObject();jsonObj.put
- 在android开发中,当不满足触发条件就按返回键的时候,就要对此进行检测。尤其是当前Activity需要往前一个Activity传送消息时
- 本文实例为大家分享了java实现发送邮箱验证码的具体代码,供大家参考,具体内容如下添加依赖<!-- 邮箱验证码 https://mvn
- 本篇博客要分享的一个UI效果——实现底部切换标签,想必大家在一些应用上面遇到过这种效果了,最典型的就是微信了,可以左右滑动切换页面,也可以点
- 本文实例为大家分享了C#绘制饼状图和柱状图的具体代码,供大家参考,具体内容如下#代码如下:using System;using System
- 在微服务架构下,我们在完成一个订单流程时经常遇到下面的场景:一个订单创建接口,第一次调用超时了,然后调用方重试了一次在订单创建时,我们需要去
- 在项目中,时常会有异步调用的需求web.xml配置<servlet> <description>spri
- 基本概念包名值得是你APP的包,在创建工程时候设置的,需要在微信支付平台上面设置。签名指的是你生成APK时候所用的签名文件的md5,去掉:全
- 前言:由于最近有解析协议的一些业务场景,需要用到一些字节操作工具,这里封装了一些比较常用的转换方法,测试后基本没有问题,可能一些比较偏门的数
- 前言在上一篇 Spring boot + LayIM + t-io 文件上传、 监听用户状态的实现 中,已经介绍了两个小细节:用户的在离线状
- 本文实例讲述了Java上传文件进度条的实现方法。分享给大家供大家参考,具体如下:东西很简单,主要用到commons-fileupload,其
- 合理的使用规则引擎可以极大的减少代码复杂度,提升代码可维护性。业界知名的开源规则引擎有Drools,功能丰富,但也比较庞大。在一些简单的场景
- Spring内置 * 对于 Web 应用来说,ServletContext 对象是唯一的,一个 Web 应用,只有一个ServletCont