在近期,公司需要开发一个关于在线支付的模块,所以需要用到第三方支付平台
经过一周多的时间对这两种支付平台的研究,完成功能后将经验分享给大家,希望能帮助到有需求的朋友。
首先是支付宝的开发资料方面:
接口申请url
/order/?productId=2014110308141993
API开发文档
/public/api/base/WS_MOBILE_PAY_SDK_BASE.zip
申请流程
注册支付宝账号??进行实名认证??提交审核资料??审核通过
备注:申请通过后会获得:合作者身份ID(PID),该ID在项目配置中需要用到
开发流程:
第一步:
下载API开发文档后,即可获取官方Demo,该Demo中需要将审核通过后获取的PID替换,并且输入支付宝收款账户即可。这里非常简单,就不过多叙述。
第二步:
官方Api开发文档中,存在一个openssl的文件夹,该文件夹主要是用于生成支付宝所需要用到的公钥以及私钥。打开该文件夹可以看到详细的生成方式,根据提示生成公钥及私钥,请注意,密钥需要经过pkcs8二次加密。
第三步:
将生成的公钥和私钥配置到Demo中。
第四步(可省略):
为了方便后期维护,建议将支付宝相关的方法及配置项抽取出来做为单独的一个类,后期需要使用直接调用即可。代码如下:
package ; import ; import ; import ; import ; import ; import ; import ; import ; import ; import ; import .; import ; import ; import ; public class Alipay { // 商户PID public static final String PARTNER = "******"; // 商户收款账号 public static final String SELLER = "***@"; // 商户私钥,pkcs8格式 public static final String RSA_PRIVATE = "*****"; // 支付宝公钥 public static final String RSA_PUBLIC = "******"; public static final int SDK_PAY_FLAG = 1; public static final int SDK_CHECK_FLAG = 2; private Handler mHandler; private Activity activity; private String orderNo; public Alipay(Handler handler, Activity activity) { mHandler = handler; = activity; } /** * call alipay sdk pay. 调用SDK支付 * */ public void pay(PayInfo payinfo) { // 订单 DecimalFormat df = new DecimalFormat("0.00"); String orderInfo = getOrderInfo((), () + " ", (() * ())); // 对订单做RSA 签名 String sign = sign(orderInfo); try { // 仅需对sign 做URL编码 sign = (sign, "UTF-8"); } catch (UnsupportedEncodingException e) { (); } // 完整的符合支付宝参数规范的订单信息 final String payInfo = orderInfo + "&sign=\"" + sign + "\"&" + getSignType(); Runnable payRunnable = new Runnable() { @Override public void run() { // 构造PayTask 对象 PayTask alipay = new PayTask(activity); // 调用支付接口,获取支付结果 String result = (payInfo); Message msg = new Message(); = SDK_PAY_FLAG; = result; (msg); } }; // 必须异步调用 Thread payThread = new Thread(payRunnable); (); } /** * check whether the device has authentication alipay account. * 查询终端设备是否存在支付宝认证账户 * */ public void check(View v) { Runnable checkRunnable = new Runnable() { @Override public void run() { // 构造PayTask 对象 PayTask payTask = new PayTask(activity); // 调用查询接口,获取查询结果 boolean isExist = (); Message msg = new Message(); = SDK_CHECK_FLAG; = isExist; (msg); } }; Thread checkThread = new Thread(checkRunnable); (); } /** * get the sdk version. 获取SDK版本号 * */ public void getSDKVersion() { PayTask payTask = new PayTask(activity); String version = (); (activity, version, Toast.LENGTH_SHORT).show(); } /** * create the order info. 创建订单信息 * */ public String getOrderInfo(String subject, String body, String price) { // 签约合作者身份ID String orderInfo = "partner=" + "\"" + PARTNER + "\""; // 签约卖家支付宝账号 orderInfo += "&seller_\"" + SELLER + "\""; // 商户网站唯一订单号 orderInfo += "&out_trade_no=" + "\"" + getOutTradeNo() + "\""; // 商品名称 orderInfo += "&subject=" + "\"" + subject + "\""; // 商品详情 orderInfo += "&body=" + "\"" + body + "\""; // 商品金额 orderInfo += "&total_fee=" + "\"" + price + "\""; // 服务器异步通知页面路径 orderInfo += "¬ify_url=" + "\"" + "/" + "\""; // 服务接口名称, 固定值 orderInfo += "&service=\"\""; // 支付类型, 固定值 orderInfo += "&payment_type=\"1\""; // 参数编码, 固定值 orderInfo += "&_input_charset=\"utf-8\""; // 设置未付款交易的超时时间 // 默认30分钟,一旦超时,该笔交易就会自动被关闭。 // 取值范围:1m~15d。 // m-分钟,h-小时,d-天,1c-当天(无论交易何时创建,都在0点关闭)。 // 该参数数值不接受小数点,如1.5h,可转换为90m。 orderInfo += "&it_b_pay=\"30m\""; // extern_token为经过快登授权获取到的alipay_open_id,带上此参数用户将使用授权的账户进行支付 // orderInfo += "&extern_token=" + "\"" + extern_token + "\""; // 支付宝处理完请求后,当前页面跳转到商户指定页面的路径,可空 orderInfo += "&return_url=\"\""; // 调用银行卡支付,需配置此参数,参与签名, 固定值 (需要签约《无线银行卡快捷支付》才能使用) // orderInfo += "&paymethod=\"expressGateway\""; return orderInfo; } /** * get the out_trade_no for an order. 生成商户订单号,该值在商户端应保持唯一(可自定义格式规范) * */ public String getOutTradeNo() { SimpleDateFormat format = new SimpleDateFormat("MMddHHmmss", ()); Date date = new Date(); String key = (date); Random r = new Random(); key = key + (); key = (0, 15); String md5 = Constants.MD5(key); = md5; return md5; } /** * 获取已经生产的订单编号 * * @return */ public String getOrderNo() { return ; } /** * sign the order info. 对订单信息进行签名 * * @param content * 待签名订单信息 */ public String sign(String content) { return (content, RSA_PRIVATE); } /** * get the sign type we use. 获取签名方式 * */ public String getSignType() { return "sign_type=\"RSA\""; } }
从上面代码可以看出,程序的主要运行流程是:通过开启一个子线程去调用支付宝的支付功能,获取到支付结果后,通过Handler通知UI线程,根据支付结果去显示不同的。
到这里基本上整个开发流程已经大致完成了,具体细节根据需求去修改即可。官方建议支付完成后,将获取到的支付结果上传到自己的服务器,通过官方提供的API进行验证,建议添加该流程