Android集成微信支付功能

 更新时间:2016年11月07日 12:15:07   作者:shineflowers  
这篇文章主要为大家详细介绍了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(); 
 } 
} 

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

相关文章

  • Flutter实现图片滤镜效果

    Flutter实现图片滤镜效果

    这篇文章主要介绍了Flutter如何实现图片滤镜效果,帮助大家更好的理解和学习使用Flutter,感兴趣的朋友可以了解下
    2021-04-04
  • Android获取系统时间以及网络时间

    Android获取系统时间以及网络时间

    这篇文章主要为大家详细介绍了Android获取系统时间以及网络时间的方法,感兴趣的小伙伴们可以参考一下
    2016-07-07
  • 实例探究Android开发中Fragment状态的保存与恢复方法

    实例探究Android开发中Fragment状态的保存与恢复方法

    这篇文章主要介绍了实例探究Android开发中Fragment状态的保存与恢复方法,或许开发者们对Fragment的操作都比较熟悉,但onSaveInstanceState()方法并不能够很好地保存Fragment状态,需要的朋友可以参考下
    2016-04-04
  • Android开发之ImageSwitcher相册功能实例分析

    Android开发之ImageSwitcher相册功能实例分析

    这篇文章主要介绍了Android开发之ImageSwitcher相册功能,结合实例形式分析了Android ImageSwitcher相册的原理、使用方法及相关操作注意事项,需要的朋友可以参考下
    2019-03-03
  • Android自定view画圆效果

    Android自定view画圆效果

    这篇文章主要为大家详细介绍了Android自定view画圆效果,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-10-10
  • Android Studio实现单选对话框

    Android Studio实现单选对话框

    这篇文章主要为大家详细介绍了Android Studio实现单选对话框,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-05-05
  • Android获取SD卡中选中图片的路径(URL)示例

    Android获取SD卡中选中图片的路径(URL)示例

    一个图片上传功能需要提供上传图片在SD卡中的路径,总结了网上的一些列子,修改了一下,代码很简单,感兴趣的朋友可以参考下哈,希望对大家有所帮助
    2013-07-07
  • Android 自定义底部上拉控件的实现方法

    Android 自定义底部上拉控件的实现方法

    下面小编就为大家分享一篇Android 自定义底部上拉控件的实现方法,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2018-01-01
  • Android 多国语言value文件夹命名的方法

    Android 多国语言value文件夹命名的方法

    这篇文章主要介绍了Android 多国语言value文件夹命名的方法,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2018-03-03
  • Android小知识之图片的3种压缩方式小结

    Android小知识之图片的3种压缩方式小结

    这篇文章主要给大家介绍了关于Android小知识之图片的3种压缩方式的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2018-10-10

最新评论