微信支付传送门:https://www.cnblogs.com/dingxiansen/p/9209159.html
一、支付宝支付
1. 支付宝支付流程图
2. 集成前准备
- 去蚂蚁金服注册应用获取appKey等信息
- 创建应用,添加APP支付功能
- 找到APP支付开发文档,下载 SDK&Demo
3. 开始集成
1、导入Demo中需要用到的Jar包
2、配置AndroidManifest.xml(这里直接放常用的权限)
<!--所需权限-->
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
<uses-permission android:name="android.permission.ACCESS_FINE_LOCATION" />
<uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
<uses-permission android:name="android.permission.CHANGE_WIFI_STATE" />
<uses-permission android:name="android.permission.READ_PHONE_STATE" />
<uses-permission android:name="android.permission.INTERNET" />
<uses-permission android:name="android.permission.READ_LOGS" />
<uses-permission android:name="android.permission.VIBRATE" />
<uses-permission android:name="android.permission.WAKE_LOCK" />
<uses-permission android:name="android.permission.WRITE_SETTINGS" /> <!--广播跳转-->
<uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" /> <!--百度地图-->
<!--百度地图权限-->
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
<uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS" /> <!--Mob分享-->
<uses-permission android:name="android.permission.GET_TASKS" />
<!-- 短信验证登陆功能需要添加次权限来自动填充验证码,用不到可以去掉 -->
<uses-permission android:name="android.permission.RECEIVE_SMS" />
<uses-permission android:name="android.permission.READ_SMS" /> <!--微信支付权限-->
<uses-permission android:name="android.permission.MODIFY_AUDIO_SETTINGS" /> <!--支付宝支付权限-->
<!-- 安卓读写sd权限 -->
<uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE" />
支付回调
<!--支付宝支付-->
<!-- 支付宝H5页面支付用的 -->
<activity
android:name="com.alipay.sdk.app.H5PayActivity"
android:configChanges="orientation|keyboardHidden|navigation"
android:exported="false"
android:screenOrientation="behind" />
<!-- 支付宝App支付页面用的 -->
<activity
android:name="com.alipay.sdk.auth.AuthActivity"
android:configChanges="orientation|keyboardHidden|navigation"
android:exported="false"
android:screenOrientation="behind" />
3、请求后台接口拿到签过名的信息(appKey,订单信息等等),这里我获取数据,是直接给后台传递的商品id,如果有的需要商品数量直接给传给后台,后台根据id查询出商品的信息,然后返回给前台所需要的支付参数。
支付宝返回参数用例(开始集成的时候看过好多博客都没有返回参数数据结构,结果一脸懵逼,在这里贴出,未加密之前的0.0):
格式化后的:
{
"msg":"success",
"code":200,
"data":"alipay_sdk=alipay-sdk-java-dynamicVersionNo&app_id=你们申请的app_id&biz_content=%7B%22out_trade_no%22%3A%22HY201806210002%22%2C%22product_code%22%3A%22QUICK_MSECURITY_PAY%22%2C%22subject%22%3A%22%E4%BC%98%E9%93%BA%E4%BC%9A%E5%91%98%E6%9C%8D%E5%8A%A1%E8%B4%B9%22%2C%22total_amount%22%3A%220.01%22%7D&charset=UTF-8&format=json&method=alipay.trade.app.pay¬ify_url=http%3A%2F%2F你们的回调地址%2FaliPayCallBack%2FcallBack&sign=buipaoj2F8xl5XCAUVoJz%2Fbh8dHbaoRmdzoAEzqKRJqtZATT4bfFdzSHurAAL5C5gvntFrDTGHgNRGw%2BNZBtG4DfetOzcpHMAyjslrmUIMIr1YGC7Qya04mFBCh%2B0UIa1E7RISZWSbIVCHpZISknNgF2oTuTixNosXvDXzkGYGBUoaxdh1f6%2F%2Bw9lqKz7mkhsUc0x8lCeJHw4MnTS4gSLU%2BDmOCk6Tkiwb4Yv4Mz%2F6j7XReeagfX7qxs5qbObnnPX%2FFmu9T%2BF0LwJaPxr5Xys8kr8E4bhd4f7Y5FimXiw%2BG7EFkY0I69boiRob7zo%2BbWQ%2F53TAMeTXX5RJybEdXhrA%3D%3D&sign_type=RSA2×tamp=2018-06-21+14%3A11%3A40&version=1.0",
"status":"0"
}
后台返回信息之后接下来就是我们的事情了,调起支付宝进行支付
/*支付宝测试*/
private void testZfbPay(final String key, final String value) {
StringRequest stringRequest = new StringRequest(Request.Method.POST, NetWorkUrl.ZFBPAY, new Response.Listener<String>() {
@RequiresApi(api = Build.VERSION_CODES.M)
@Override
public void onResponse(String s) {
Log.e("GoPayOrderActivity", "-------getJson2-------" + s.toString()); /*判断code*/
String code = (JSONObject.parseObject(s.toString()).getString("code")); if (code.equals("200")) {
String orderInfo = (JSONObject.parseObject(s.toString()).getString("data"));//返回的信息
MyALipayUtils.ALiPayBuilder builder = new MyALipayUtils.ALiPayBuilder(); builder.build().toALiPay(GoPayOrderActivity.this, orderInfo);
}
}
}, new Response.ErrorListener() {
@Override
public void onErrorResponse(VolleyError volleyError) { }
}) {
@Override
public Map<String, String> getParams() throws AuthFailureError {
Log.e("GoPayOrderActivity", "getParams:-----------------> " + userEntity.getPhone()); Map<String, String> map = new HashMap<String, String>();
map.put("account", userEntity.getPhone());
map.put(key, value);
map.put("token", userEntity.getToken());
return map;
} @Override
public Map<String, String> getHeaders() throws AuthFailureError {
HashMap<String, String> headers = new HashMap<String, String>();
if (userEntity.getToken().equals("") && userEntity != null) {
headers.put("Authorization", userEntity.getToken());
}
return headers;
} };
/*设置请求一次*/
stringRequest.setRetryPolicy(
new DefaultRetryPolicy(
5000,//默认超时时间,应设置一个稍微大点儿的,例如本处的500000
DefaultRetryPolicy.DEFAULT_MAX_RETRIES,//默认最大尝试次数
DefaultRetryPolicy.DEFAULT_BACKOFF_MULT
)
);
AppApplication.getHttpQueues().add(stringRequest);/*请求数据*/
}
MyALipayUtils.java这个类直接copy使用就可以
/**
* Created by dingchao on 2018/3/20.
*/ public class MyALipayUtils {
private static final int SDK_PAY_FLAG = 1;
private Activity context;
private ALiPayBuilder builder; private MyALipayUtils(ALiPayBuilder builder) {
this.builder = builder;
} private Handler mHandler = new Handler() {
public void handleMessage(Message msg) { // 返回码 含义
// 9000 订单支付成功
// 8000 正在处理中,支付结果未知(有可能已经支付成功),请查询商户订单列表中订单的支付状态
// 4000 订单支付失败
// 5000 重复请求
// 6001 用户中途取消
// 6002 网络连接出错
// 6004 支付结果未知(有可能已经支付成功),请查询商户订单列表中订单的支付状态
// 其它 其它支付错误
PayResult payResult = new PayResult((Map<String, String>) msg.obj);
switch (payResult.getResultStatus()) {
case "9000":
MobclickAgent.onEvent(context, "payment_success", "付款成功");
Toast.makeText(context, "支付成功", Toast.LENGTH_SHORT).show();
AppApplication.finishActivity();
context.finish();
// Toast.makeText(this, "支付成功", Toast.LENGTH_SHORT).show();
/*跳转我的会员页面*/
// Intent intent = new Intent(context, MyVipActivity.class);
// context.startActivity(intent);
break;
case "8000":
Toast.makeText(context, "正在处理中", Toast.LENGTH_SHORT).show();
break;
case "4000":
MobclickAgent.onEvent(context, "payment_fali", "付款失败");
Toast.makeText(context, "订单支付失败", Toast.LENGTH_SHORT).show();
break;
case "5000":
Toast.makeText(context, "重复请求", Toast.LENGTH_SHORT).show();
break;
case "6001":
Toast.makeText(context, "已取消支付", Toast.LENGTH_SHORT).show();
break;
case "6002":
Toast.makeText(context, "网络连接出错", Toast.LENGTH_SHORT).show();
break;
case "6004":
Toast.makeText(context, "正在处理中", Toast.LENGTH_SHORT).show();
break;
default:
MobclickAgent.onEvent(context, "payment_fali", "付款失败");
Toast.makeText(context, "支付失败", Toast.LENGTH_SHORT).show();
break;
}
}
}; /**
* 签名发在客户端来做。
*
* @param context
*/
public void toALiPay(final Activity context) {
this.context = context;
boolean rsa2 = (builder.getRsa2().length() > 0);
Map<String, String> params = buildOrderParamMap(rsa2);
String orderParam = buildOrderParam(params);
String privateKey = rsa2 ? builder.getRsa2() : builder.getRsa();
String sign = getSign(params, privateKey, rsa2);
final String orderInfo = orderParam + "&" + sign;
Log.e("chx", "toALiPay: " + orderInfo);
Runnable payRunnable = new Runnable() { @Override
public void run() {
PayTask alipay = new PayTask(context);
Map<String, String> result = alipay.payV2
(orderInfo, true);
Message msg = new Message();
msg.what = SDK_PAY_FLAG;
msg.obj = result;
mHandler.sendMessage(msg);
}
}; Thread payThread = new Thread(payRunnable);
payThread.start();
} /**
* 签名在服务端来做
*
* @param context
* @param orderInfo
*/
public void toALiPay(final Activity context, final String orderInfo) {
this.context = context;
Runnable payRunnable = new Runnable() { @Override
public void run() {
PayTask alipay = new PayTask(context);
Map<String, String> result = alipay.payV2
(orderInfo, true);
Message msg = new Message();
msg.what = SDK_PAY_FLAG;
msg.obj = result;
mHandler.sendMessage(msg);
}
}; Thread payThread = new Thread(payRunnable);
payThread.start();
} /**
* 构造支付订单参数列表
*
* @param
* @param
* @return
*/
private Map<String, String> buildOrderParamMap(boolean rsa2) {
Map<String, String> keyValues = new HashMap<String, String>(); keyValues.put("app_id", builder.appid); keyValues.put("biz_content", "{\"timeout_express\":\"30m\",\"product_code\":\"QUICK_MSECURITY_PAY\",\"total_amount\":\"" + builder.money + "\",\"subject\":\"" + builder.title + "\",\"out_trade_no\":\"" + builder.orderTradeId + "\"}"); keyValues.put("charset", "utf-8"); keyValues.put("method", "alipay.trade.app.pay");
//回调接口
keyValues.put("notify_url", builder.getNotifyUrl()); keyValues.put("sign_type", rsa2 ? "RSA2" : "RSA"); // keyValues.put("timestamp", "2016-07-29 16:55:53");
keyValues.put("timestamp", getCurrentTimeString()); keyValues.put("version", "1.0"); return keyValues;
} /**
* 构造支付订单参数信息
*
* @param map 支付订单参数
* @return
*/
private String buildOrderParam(Map<String, String> map) {
List<String> keys = new ArrayList<String>(map.keySet()); StringBuilder sb = new StringBuilder();
for (int i = 0; i < keys.size() - 1; i++) {
String key = keys.get(i);
String value = map.get(key);
sb.append(buildKeyValue(key, value, true));
sb.append("&");
Log.e("chx", "buildOrderParam: " + buildKeyValue(key, value, true));
} String tailKey = keys.get(keys.size() - 1);
String tailValue = map.get(tailKey);
sb.append(buildKeyValue(tailKey, tailValue, true)); return sb.toString();
} /**
* 对支付参数信息进行签名
*
* @param map 待签名授权信息
* @return
*/
private String getSign(Map<String, String> map, String rsaKey, boolean rsa2) {
List<String> keys = new ArrayList<String>(map.keySet());
// key排序
Collections.sort(keys); StringBuilder authInfo = new StringBuilder();
for (int i = 0; i < keys.size() - 1; i++) {
String key = keys.get(i);
String value = map.get(key);
authInfo.append(buildKeyValue(key, value, false));
authInfo.append("&");
} String tailKey = keys.get(keys.size() - 1);
String tailValue = map.get(tailKey);
authInfo.append(buildKeyValue(tailKey, tailValue, false)); String oriSign = sign(authInfo.toString(), rsaKey, rsa2);
String encodedSign = ""; try {
encodedSign = URLEncoder.encode(oriSign, "UTF-8");
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
}
return "sign=" + encodedSign;
} private static final String ALGORITHM = "RSA"; private static final String SIGN_ALGORITHMS = "SHA1WithRSA"; private static final String SIGN_SHA256RSA_ALGORITHMS = "SHA256WithRSA"; private static final String DEFAULT_CHARSET = "UTF-8"; private String getAlgorithms(boolean rsa2) {
return rsa2 ? SIGN_SHA256RSA_ALGORITHMS : SIGN_ALGORITHMS;
} private String sign(String content, String privateKey, boolean rsa2) {
try {
PKCS8EncodedKeySpec priPKCS8 = new PKCS8EncodedKeySpec(
Base64.decode(privateKey));
KeyFactory keyf = KeyFactory.getInstance(ALGORITHM);
PrivateKey priKey = keyf.generatePrivate(priPKCS8); java.security.Signature signature = java.security.Signature
.getInstance(getAlgorithms(rsa2)); signature.initSign(priKey);
signature.update(content.getBytes(DEFAULT_CHARSET)); byte[] signed = signature.sign(); return Base64.encode(signed);
} catch (Exception e) {
e.printStackTrace();
} return null;
} /**
* 拼接键值对
*
* @param key
* @param value
* @param isEncode
* @return
*/
private String buildKeyValue(String key, String value, boolean isEncode) {
StringBuilder sb = new StringBuilder();
sb.append(key);
sb.append("=");
if (isEncode) {
try {
sb.append(URLEncoder.encode(value, "UTF-8"));
} catch (UnsupportedEncodingException e) {
sb.append(value);
}
} else {
sb.append(value);
}
return sb.toString();
} /**
* 获取当前日期字符串
*
* @return
*/
private String getCurrentTimeString() {
SimpleDateFormat df = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
return df.format(new Date());
} public static class ALiPayBuilder {
private String rsa2 = "";
private String rsa = "";
private String appid;
private String money;
private String title;
private String notifyUrl;
private String orderTradeId; public MyALipayUtils build() {
return new MyALipayUtils(this);
} public String getOrderTradeId() {
return orderTradeId;
} public ALiPayBuilder setOrderTradeId(String orderTradeId) {
this.orderTradeId = orderTradeId;
return this;
} public String getRsa2() {
return rsa2;
} public ALiPayBuilder setRsa2(String rsa2) {
this.rsa2 = rsa2;
return this;
} public String getRsa() {
return rsa;
} public ALiPayBuilder setRsa(String rsa) {
this.rsa = rsa;
return this;
} public String getAppid() {
return appid;
} public ALiPayBuilder setAppid(String appid) {
this.appid = appid;
return this;
} public String getMoney() {
return money;
} public ALiPayBuilder setMoney(String money) {
this.money = money;
return this;
} public String getTitle() {
return title;
} public ALiPayBuilder setTitle(String title) {
this.title = title;
return this;
} public String getNotifyUrl() {
return notifyUrl;
} public ALiPayBuilder setNotifyUrl(String notifyUrl) {
this.notifyUrl = notifyUrl;
return this;
}
}
}
完成上述操作,你的app在蚂蚁金服后台应用上线之后,就完全可以调用支付了,但是在开发阶段,应用没有上线,你是不能进行调试的,所以支付宝有沙箱模式可以进行调试,ios没有哟
进行沙箱调试在Activity的的onCreate()方法中添加
EnvUtils.setEnv(EnvUtils.EnvEnum.SANDBOX);//支付宝沙箱环境,正式需注释
别忘了让你们后台把appKey等信息换成沙箱的,然后测试支付的时候,需要下载一个沙箱支付宝,这个你可以随意支付,附上链接
沙箱首页:https://sandbox.alipaydev.com/user/accountDetails.htm?currentBar=1
App支付接入文档:https://docs.open.alipay.com/204/105051
沙箱钱包下载:
调试都没有问题之后,就可以在蚂蚁金服开发者平台进行应用上线,然后沙箱代码就可以注释了,这样支付宝支付就接入完成
如果问题或建议请发送到我的邮箱:dingchao7323@qq.com