Android 微信支付&支付宝支付

た 入场券 2023-10-10 08:03 108阅读 0赞

由于项目需求,加入这2个功能记录一些需要注意的地方

一.微信支付

微信支付在2016年4月份左右稍微调整了一下支付过程,但是文档却没怎么更新,这也是百度上为什么那么多开发者都说微信是个大坑. 身为一个大型互联网公司,做的事却无法让人理解.

相比之下,支付宝就好太多了.

微信支付需要注意地方:

1.要先申请 “商家端”账户, 需要提交公司信息, 个人不好申请的. 申请成功后,需要去 创建一个应用

创建应用需要 注意2个地方

1>包名, 如果是Android studio开发 ,这个包名要用如下图所示位置的包名, 而不是 androidmanifest.xml中的,因为他俩可能不一样,要以下图为准,当然如果是eclipse开发那就以androidmanifest.xml里面的包名为准了.

538831-20160616085138260-1496682125.png

2>签名, 这个签名不能用 debug.keystore(Eclipse的签名后缀是.keystore), 或者 debug.jks(Androidstudio把后缀名换了), 要用自己创建的的签名文件,比如 shop.jsk , shop.keystore去生成签名,或者选择以前自己创建好的,都行,只有不是debug的就行.

android studio生成签名截图如下: 点击Teminal 启动后,输入图中命令,后面是你jks所在位置. 之后回车 输入密码, 得到MD5值, 它就是微信创建应用时用到的签名,需要去掉冒号,把大写全部改成小写即可. 当然微信还给我们提供了签名工具apk, 你可以把自己的工程打包后,安装到手机,然后启动微信的签名apk,输入包名即可得到签名, 你可以拿它对比一下用命令行得到的签名是否一致, 如果不一致那就说明一个问题 “ 你的应用没有经过打包” , 不要通过编译工具直接部署 到手机,这样得到的签名是不对的. 一定要是打包后的.

当测试微信的时候,也需要把打包后的apk安装到手机才能测试支付,如果只能调起一次微信,第二次无法调起,那就是签名或者 appid这一块有问题,检查一下. 不过我发现IOS不存在这个问题, 每次都能调起微信.

538831-20160616085617432-1942008653.png

二.下面说下微信在工程中创建需要注意地方:

  1. 如果你的包名是com.xxx.text 那么你需要在test之后创建一个包名 com.xxx.text.wxapi, 必须是wxapi,不能错了. 然后必须拷贝一个类过来:WXPayEntryActivity.java ,这个类的作用是微信支付后,接收微信的支付结果的. 你可以不做任何修改,直接拿来用就行,当然如果你想更改里面的布局,也是完全OK的, 我对他做了一下修改,代码如下, 我修改了2个地方,代码中给出了注释

    public class WXPayEntryActivity extends Activity implements IWXAPIEventHandler{

    1. private static final String TAG = "MicroMsg.SDKSample.WXPayEntryActivity";
    2. private IWXAPI api;
    3. @Override
    4. public void onCreate(Bundle savedInstanceState) {
    5. super.onCreate(savedInstanceState);

                                    //1.第一个修改的地方 我删掉了,它自带的 布局, 当然如果你想保留,完全OK.因为布局太难看所以我干掉他了

    1. api = WXAPIFactory.createWXAPI(this, Constants.APP_ID);
    2. api.handleIntent(getIntent(), this);
    3. }
    4. @Override
    5. protected void onNewIntent(Intent intent) {
    6. super.onNewIntent(intent);
    7. setIntent(intent);
    8. api.handleIntent(intent, this);
    9. }
    10. @Override
    11. public void onReq(BaseReq req) {}
    12. @Override
    13. public void onResp(BaseResp resp) {

      //2.这是我修改的第二个地方, 原版的是弹出一个对话框, 我觉得太丑,并且原版也没有给出详细问题提示,只给出了 0, -1 ,-2 这样会让客户不明所以,所以我替换成如下3个文本. 此外 //原版支付后,不会直接跳转到 你的应用,需要按一次 返回键,才行, 所以我加入了finish()便于在提示后,直接返回我的应用

    1. if(resp.errCode==0){
    2. Toast.makeText(this,"支付成功!",Toast.LENGTH_SHORT).show();
    3. }else if(resp.errCode==-1){
    4. Toast.makeText(this,"支付失败!",Toast.LENGTH_SHORT).show();
    5. }else if(resp.errCode==-2){
    6. Toast.makeText(this,"取消支付!",Toast.LENGTH_SHORT).show();
    7. }
    8. finish();
    9. }

    }

  2. androidmainfest.xml 的配置,直接上代码,这里只需要注意一个地方那就是 appid

        //必须单例
    1. <receiver
    2. android:name=".wxapi.AppRegister">
    3. <intent-filter>
    4. <action android:name="com.tencent.mm.plugin.openapi.Intent.ACTION_REFRESH_WXAPP" />
    5. </intent-filter>
    6. </receiver>




      //这里就是 APPID ,必须和你申请的 应用的 appid保持一致 .否则无法支付,注意,注意,注意

3.下面是我项目中的支付代码, 先说下支付流程, 首先你要 让负责接口的,给你一个接口, 这个接口 需要app端传递 商品的基本信息过去,比如 商品名字 , 描述,订单号 ,价格等等.也不需要传递太多东西, 我就传递了 以上我列举的4个 ,有人会问 “微信不是帮我们生成了一个 订单号吗?” 我干嘛还要传递给他, 我传递的这个订单号,是用于退款使用的. 所以我要传递一个, 这个订单号,在我 下单的同时, 会生成一个随机数, 当然这都是 后台 接口弄得,你只需要拿来使用就行了.

说的有点乱了,给个步骤吧

1>传递商品信息给我们自己的后台

2>我们自己的后台会拿着这些商品信息 然后加上 + 商户号id+商户秘钥 (申请商家端时获取的), 给我们app端返回一个json,该json中包含7个字段,需要解析,后面给出代码

3>我们解析后, 通过 api.sendReq(req); 调用微信支付

代码如下:

  1. float total3 = orderTotalPrice * 100;  //注意微信支付付款 是按照分为单位的,需要把商品价格 乘以100 ,然后强转为 int类型 ,这里(int)total3
  2. param ="?ordertype="+"old"+"&sworkOrderCode="+out_trade_no+"&serviceFee=" +(int)total3+"&serviceItem="+orderItem.orderName.trim()+"&serviceClass="+orderItem.serviceItem.trim();
  3. testWxPay(v);//开始解析接口给我们返回的json
  4. 参数:old 是,一个版本区分,因为公司的多个app都用到了微信支付,并且他们都有关联,所以你可以直接无视该参数,该参数仅用于我自己的项目
  5. 参数:out_trade_no 是我从我们的接口获取到的随机订单号
  6. 参数:(int)total3 是商品价格, "分" 为单位,必须转为 整形, 也不知道 int会不会越界,反正我用了int,你可用long
  7. 参数:serviceItem 商品名字
  8. 参数:serviceClass商品描述
  9. public void testWxPay(View view) {
  10. new Thread(new Runnable() {
  11. @Override
  12. public void run() {
  13. String url = HttpUrl.host1+"WeiXinApi.asmx/CreatePrePay_id"+param;
  14. Log.e("xxx",url);
  15. ToastUtil.shortToastInBackgroundThread(getActivity(), "获取订单中...");
  16. try {
  17. byte[] buf = Util.httpGet(url);
  18. if (buf != null && buf.length > 0) {
  19. String content = new String(buf);
  20. Log.e("get server pay params", content);
  21. JSONObject json = new JSONObject(content);
  22. if (null != json && !json.has("retcode")) {
  23. req = new PayReq();
  24. req.appId = json.getString("appid");    //appid
  25. req.partnerId = json.getString("partnerid"); //商户号
  26. req.prepayId = json.getString("prepayid"); //预支付交易会话id
  27. req.nonceStr = json.getString("noncestr"); //随机字符串,不超过32位
  28. req.timeStamp = json.getString("timestamp");//时间戳
  29. req.packageValue = json.getString("package");//扩展字段
  30. req.sign = json.getString("sign");//签名信息MD5加密后的
  31. ToastUtil.shortToastInBackgroundThread(getActivity(), "正常调起支付");
  32. toPay();
  33. } else {
  34. Log.d("PAY_GET", "返回错误" + json.getString("retmsg"));
  35. ToastUtil.shortToastInBackgroundThread(getActivity(), "返回错误" + json.getString("retmsg"));
  36. }
  37. } else {
  38. Log.d("PAY_GET", "服务器请求错误");
  39. ToastUtil.shortToastInBackgroundThread(getActivity(), "服务器请求错误");
  40. }
  41. } catch (Exception e) {
  42. Log.e("PAY_GET", "异常:" + e.getMessage());
  43. ToastUtil.shortToastInBackgroundThread(getActivity(), "异常:" + e.getMessage());
  44. }
  45. }
  46. }).start();}
  47. private void toPay() {
  48. // 在支付之前,如果应用没有注册到微信,应该先调用IWXMsg.registerApp将应用注册到微信
  49. api.registerApp(Constants.APP_ID);
  50. api.sendReq(req);
  51. Log.e("跳转结果--",api.sendReq(req)+"");
  52. }
  53. private Handler handler = new Handler() {
  54. @Override
  55. public void handleMessage(Message msg) {
  56. super.handleMessage(msg);
  57. toPay();
  58. }
  59. };

整个支付代码非常少,一共就这么多. 需要注意的一点是, 解析的 7个 参数中, sign 是我们自己服务端 二次加密得到的,这里容易出错, 当然我说的是接口里面容易出错,和我们app端没有半毛钱关系, 因为这个 加密,老版本微信是我们 app端生成的, 新版的 是由接口生成的,我们直接拿来用.

sign 生成过程是: 前6个参数+ 商家秘钥 经过MD5加密就得到了. (这个加密用的前6个参数,需要安装字母排序,不能乱用,商家秘钥不需排序放到最后即可,当然这都是接口的需要完成的,我们app端 了解即可)

如果这个sign 接口没有弄好, 极容易出错,无法调起支付页. 我在做的时候,就是因为接口的生成sign时,少写一个 = 连接符造成的.

-————————————————————————————————————————————————————————————————————-

三.支付宝支付

支付宝支付相对来说要简单的多了, 比较稳定写得清晰明了,一看即懂. 直接上代码吧 ,我去网络太卡了,刚才写了半天,图片无法加载,把我写得重新毁了,先保存吧,等会再上支付宝的.

流程如下:

1.申请账号,同样需要提交一些公司资料

2.下载需要资源,sdk ,文档什么的, 然后去配置工程

1>在工程中创建一个包存放支付宝所需资源,比如我的

538831-20160617085621792-2118012836.png

导入alipaySdk-20160427.jar , 目前最新版.

2>androidmainfest.xml中的配置

  1. <!--支付宝-->
  2. <activity
  3. android:name="com.alipay.sdk.app.H5PayActivity"  //H5页面, 如果手机没有安装支付宝,调起这个
  4. android:configChanges="orientation|keyboardHidden|navigation"
  5. android:exported="false"
  6. android:screenOrientation="behind" >
  7. </activity>
  8. <activity
  9. android:name="com.alipay.sdk.auth.AuthActivity"
  10. android:configChanges="orientation|keyboardHidden|navigation"
  11. android:exported="false"
  12. android:screenOrientation="behind" >
  13. </activity>
  14.    权限信息
  15.    <uses-permission android:name="android.permission.INTERNET" />
  16. <uses-permission android:name="android.permission.ACCESS_NETWORK_STATE" />
  17. <uses-permission android:name="android.permission.ACCESS_WIFI_STATE" />
  18. <uses-permission android:name="android.permission.READ_PHONE_STATE" />
  19. <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />

3.获取私钥, 公钥

命令如下:

  1. RSA密钥生成命令
  2. 生成RSA私钥
  3. openssl>genrsa -out rsa_private_key.pem 1024
  4. 生成RSA公钥
  5. openssl>rsa -in rsa_private_key.pem -pubout -out rsa_public_key.pem
  6. RSA私钥转换成PKCS8格式
  7. openssl>pkcs8 -topk8 -inform PEM -in rsa_private_key.pem -outform PEM -nocrypt

命令使用的工具在 这里: E:\支付宝钱包支付接口开发包2.0标准版(20160428)\DEMO\openssl\bin —> openssl.exe

双击启动,截图如下: 然后输入上面命令,就可以得到了.最后会生成相应的文件,但是后缀好像不是txt,你改成txt就能看了.具体过程文档说的很清楚,我也记不太清了,做了1周就忘记了,这记性 …

538831-20160617090323151-165396115.png538831-20160617090611385-1643159374.png

打开红框框中的2个文件,里面就用公钥和私钥信息了, 注意红色部分不能要,只要中间的, 你最好把中间部分回车去掉,改成1行,不然可能会因为回车造成出错

538831-20160617090729463-1966282395.png

538831-20160617090805448-1225815197.png

配置貌似也就这么多了吧. 比较少,文档也很清楚.

4.下面给出代码吧,截图了后面数据不能泄露了,公司的信息

538831-20160617091031463-1578940333.png

此处点击 : “支付宝”支付这个 线性布局,执行支付功能

  1. rlZfb.setOnClickListener(new View.OnClickListener() {
  2. @Override
  3. public void onClick(View v) {
  4. pay(v);
  5. }
  6. });
  7. /**
  8. * call alipay sdk pay. 调用SDK支付
  9. */
  10. public void pay(View v) {
  11. if (TextUtils.isEmpty(PARTNER) || TextUtils.isEmpty(RSA_PRIVATE) || TextUtils.isEmpty(SELLER)) {
  12. new AlertDialog.Builder(this).setTitle("警告").setMessage("需要配置PARTNER | RSA_PRIVATE| SELLER")
  13. .setPositiveButton("确定", new DialogInterface.OnClickListener() {
  14. public void onClick(DialogInterface dialoginterface, int i) {
  15. //
  16. finish();
  17. }
  18. }).show();
  19. return;
  20. }
  21. //得到订单信息
  22. String orderInfo = null;
  23. //待支付列表item对象
  24. orderInfo = getOrderInfo(orderItem.getServiceCode(), orderTotalPrice);
  25. String sign = sign(orderInfo);
  26. try {
  27. /**
  28. * 仅需对sign 做URL编码
  29. */
  30. sign = URLEncoder.encode(sign, "UTF-8");
  31. } catch (UnsupportedEncodingException e) {
  32. e.printStackTrace();
  33. }
  34. /**
  35. * 完整的符合支付宝参数规范的订单信息
  36. */
  37. final String payInfo = orderInfo + "&sign=\"" + sign + "\"&" + getSignType();
  38. Runnable payRunnable = new Runnable() {
  39. @Override
  40. public void run() {
  41. // 构造PayTask 对象--该对象主要为商户提供订单支付功能
  42. PayTask alipay = new PayTask(getActivity());
  43. // 调用支付接口,获取支付结果
  44. String result = alipay.pay(payInfo, true);
  45. Message msg = new Message();
  46. msg.what = SDK_PAY_FLAG;
  47. msg.obj = result;
  48. mHandler.sendMessage(msg);
  49. }
  50. };
  51. // 必须异步调用
  52. Thread payThread = new Thread(payRunnable);
  53. payThread.start();
  54. }
  55. /**
  56. * create the order info. 创建订单信息
  57. * subject 商品名字
  58. */
  59. private String getOrderInfo(String subject, float price) {
  60. // 签约合作者身份ID
  61. String orderInfo = "partner=" + "\"" + PARTNER + "\"";
  62. // 签约卖家支付宝账号
  63. orderInfo += "&seller_id=" + "\"" + SELLER + "\"";
  64. // 商户网站唯一订单号
  65. //orderInfo += "&out_trade_no=" + "\"" + getOutTradeNo() + "\"";
  66. orderInfo += "&out_trade_no=" + "\"" + out_trade_no + "\"";
  67. // 商品名称
  68. orderInfo += "&subject=" + "\"" + subject + "\"";
  69. // 商品详情
  70. //orderInfo += "&body=" + "\"" + body + "\"";
  71. // 商品金额
  72. orderInfo += "&total_fee=" + "\"" + price + "\"";
  73. // 服务器异步通知页面路径 --- 调用服务端地址
  74. //orderInfo += "&notify_url=" + "\"" + "http://notify.msp.hk/notify.htm" + "\"";
  75. //ToastUtils.show(getActivity(),subject+price);
  76. Log.e("xxxx", subject + price);
  77. orderInfo += "&notify_url=" + "\"" + "http://xxx.xxx.xxx/HomeWorkOrder.asmx/WorkOrderPayed" + "\"";//此处是接口端给的一个 地址,找他们要吧....
  78. // 服务接口名称, 固定值
  79. orderInfo += "&service=\"mobile.securitypay.pay\"";
  80. // 支付类型, 固定值
  81. orderInfo += "&payment_type=\"1\"";
  82. // 参数编码, 固定值
  83. orderInfo += "&_input_charset=\"utf-8\"";
  84. // 设置未付款交易的超时时间
  85. // 默认30分钟,一旦超时,该笔交易就会自动被关闭。
  86. // 取值范围:1m~15d。
  87. // m-分钟,h-小时,d-天,1c-当天(无论交易何时创建,都在0点关闭)。
  88. // 该参数数值不接受小数点,如1.5h,可转换为90m。
  89. orderInfo += "&it_b_pay=\"30m\"";
  90. // extern_token为经过快登授权获取到的alipay_open_id,带上此参数用户将使用授权的账户进行支付
  91. // orderInfo += "&extern_token=" + "\"" + extern_token + "\"";
  92. // 支付宝处理完请求后,当前页面跳转到商户指定页面的路径,可空
  93. orderInfo += "&return_url=\"m.alipay.com\"";
  94. // 调用银行卡支付,需配置此参数,参与签名, 固定值 (需要签约《无线银行卡快捷支付》才能使用)
  95. // orderInfo += "&paymethod=\"expressGateway\"";
  96. return orderInfo;
  97. }

大多数都是支付宝代码,贴着也没意思, 只不过修改了订单信息而已. 后面的不再贴了. demo和文档已经非常清楚了. 支付宝退款很简单,只需要发送订单号给我们自己的接口即可,其他的由接口去实现.

以上就是所有代码了.写得不好 ,勿喷!!

转载于:https://www.cnblogs.com/android-zcq/p/5589917.html

发表评论

表情:
评论列表 (有 0 条评论,108人围观)

还没有评论,来说两句吧...

相关阅读