本文讲解使用微信支付接口完成在android开发的原生态app中完成微信支付功能, 文章具体讲解了前端android如何集成微信支付功能以及后台如何组装前端需要支付信息, 话不多话, 具体看文章内容吧

本实例项目运行条件:

开发环境: 【Android Studio】

1. 到微信开放平台注册帐号并且创建移动应用

https://open.weixin.qq.com/cgi-bin/frame?t=home/app_tmpl&lang=zh_CN

2. 获得移动应用的权限【微信支付】

这个权限要求比较高,需要公司资质 并且 每年需要支付300元 才能开通 (这里不作讲解, 具体到官网上申请)

3. 配置应用签名, 这个签名通过 android打包文件jks生成或者keystore生成    【如何生成jks文件】

签名文件生成方法:

3.1  keytool -list -v -keystore jks文件(或者keystore文件)

3.2 获取指纹证书md5值, 将md5中的冒号去掉, 大写换成小写   (详情)

总结: 微信开放平台Android应用签名的本质便是我们签名文件jks(或者keystore)的MD5值

4. 配置支付密钥, 【如何配置密钥】

5. 应用程序开发完成后,debug模式是无法完成支付的,应用程序必须由相应的jks签名之后生成的apk包安装在手机上才能进行分支付(微信会校验应用签名)

2. 支付流程讲解

2.1 android程序启动后如下第一张图, 点击【确认支付】

2.1.1 android端向后台请求获得预支付信息

2.1.2 后台根据微信官网平台上的 配置信息 加上 订单信息 生成预支付信息

2.1.3 android端根据预支付信息 拉起微信支付页面进行支付(见下面第二张图)

3. 代码详解(Android端)

3.1 在android studio中引入 微信需要使用的jar包

3.2 在android工程对应的包名下面新建 包以及类, wxapi/WXPayEntryActivity

在AndroidManifest.xml中引用 WXPayEntryActivity

 <application
     android:allowBackup="true"
     android:icon="@drawable/desk"
     android:label="@string/app_name"
     android:theme="@style/AppTheme">
     <activity
         android:name=".activity.MainActivity"
         android:screenOrientation="portrait"
         android:label="@string/app_name">
         <intent-filter>
             <action android:name="android.intent.action.MAIN" />
             <category android:name="android.intent.category.LAUNCHER" />
          </intent-filter>
     </activity>

     <activity
         android:name=".wxapi.WXPayEntryActivity"
         android:screenOrientation="portrait"
         android:exported="true"
         android:launchMode="singleTop"/>
 </application>

WXPayEntryActivity 代码详情(支付完成后,onResp会被调用,BaseResp.ErrCode.ERR_OK 表明支付成功)

 public class WXPayEntryActivity extends Activity implements IWXAPIEventHandler {

     private static final String TAG = "WXPayEntryActivity";

     private IWXAPI api;

     @Override
     public void onCreate(Bundle savedInstanceState) {
         super.onCreate(savedInstanceState);

         Constant.wxApi.handleIntent(getIntent(), this);
     }

     @Override
     protected void onNewIntent(Intent intent) {
         super.onNewIntent(intent);
         setIntent(intent);
         api.handleIntent(intent, this);
     }

     @Override
     public void onReq(BaseReq req) {
     }

     /**
      * 得到支付结果回调
      */
     @Override
     public void onResp(BaseResp resp)
     {
         Log.i(TAG, "onPayFinish, errCode = " + resp.errCode);

         String strPayResult = "";
         switch (resp.errCode) {
             case BaseResp.ErrCode.ERR_OK:
                 Toast.makeText(this, "付款成功!", Toast.LENGTH_SHORT).show();
                 break;
             case BaseResp.ErrCode.ERR_USER_CANCEL:
                 //分享取消
                 //Toast.makeText(this, "付款取消!", Toast.LENGTH_SHORT).show();
                 //Constant.WEIXIN_PAY_STATUS = "PAY_CANCEL";
                 break;
             case BaseResp.ErrCode.ERR_AUTH_DENIED:
                 //分享拒绝
                 //Toast.makeText(this, "付款拒绝!", Toast.LENGTH_SHORT).show();
                 //Constant.WEIXIN_PAY_STATUS = "PAY_DENY";
                 break;
         }

         //向之前页面返回支付结果信息
         /*Intent intent = new Intent();
         intent.putExtra("payResult", strPayResult);
         setResult(100, intent);*/
         finish();
     }
 }

3.3 支付按钮的点击事件

通过http协议向后台请求相应的预支付信息, 根据这些信息组装相应的信息来调用微信接口, 拉起微信支付界面

 @OnClick(R.id.pay)
 public void pay(){
     String orderNum = OrderInfo.generateOutTradeNo();
     payService
             .wpay(orderNum, totalPrice, address.getText().toString() + "-外卖订单")
             .subscribe(new Action1<WeiXinPrePay>() {
                 @Override
                 public void call(WeiXinPrePay payInfo) {
                     if (Constant.wxApi != null) {
                         PayReq req = new PayReq();
                         req.appId = payInfo.getAppId();// 微信开放平台审核通过的应用APPID
                         req.partnerId = payInfo.getMchId();// 微信支付分配的商户号
                         req.prepayId = payInfo.getPrepayId();// 预支付订单号,app服务器调用“统一下单”接口获取
                         req.nonceStr = payInfo.getNonceStr();// 随机字符串,不长于32位,服务器小哥会给咱生成
                         req.timeStamp = payInfo.getTimeStamp();// 时间戳,app服务器小哥给出
                         req.packageValue = "WXPay";// 固定值Sign=WXPay,可以直接写死,服务器返回的也是这个固定值
                         req.sign = payInfo.getPaySign();// 签名,服务器小哥给出,他会根据:                                                   om/wiki/doc/api/app/app.php?chapter=4_3指导得到这个
                         Constant.wxApi.sendReq(req);
                     }
                 }
             }, new Action1<Throwable>() {
                 @Override
                 public void call(Throwable throwable) {
                     showToast(throwable.getMessage());
                 }
             });
 }

4 微信支付Java后台

4.1 后台代码结构图

4.2 微信配置信息 Config.properties

4.3 方法wxpay用于生成预支付订单信息

方法notifyWeiXinPay用于微信支付成功后的回调, 注意: 在手机端使用微信支付成功后,微信服务器会根据提供的回调地址进行回调, parameterMap.put("notify_url", wxnotify); (见下面代码)

在局域网是无法进行回调的,必须将你的服务端放在公网上进行测试, 回调函数会被多次调用,如果第一次成功后,你可以将业务数据状态标志为已处理, 对于相同订单的其它回调就不需要再次处理了

 @Controller
 @RequestMapping("/pay")
 public class PayController {

     String timeMillis = String.valueOf(System.currentTimeMillis() / 1000);
     String randomString = PayCommonUtil.getRandomString(32);
     //支付成功后的回调函数
     public static String wxnotify = "http://gepanjiang.hk1.tunnelfrp.cc/WxPay/pay/notifyWeiXinPay.htm";

     public PayController() {
         System.out.println("MainController构造函数");
     }

     /**
      * @param totalAmount    支付金额
      * @param description    描述
      * @param request
      * @return
      */
     @RequestMapping(value = "/wxpay.htm", produces = MediaType.APPLICATION_JSON_VALUE)
     @ResponseBody
     public Result wxpay(HttpServletRequest request) {
         Result result = new Result();
         Long userId = new Long(1);//baseController.getUserId();

         BigDecimal totalAmount = new BigDecimal(request.getParameter("totalPrice"));
         String trade_no = "";
         String description="";
         try {
             trade_no = new String(request.getParameter("orderNum").getBytes("ISO-8859-1"),"UTF-8");
             description = request.getParameter("description");
         } catch (UnsupportedEncodingException e) {
             // TODO Auto-generated catch block
             e.printStackTrace();
         }
         String openId = "";

         Map<String, String> map = weixinPrePay(trade_no,totalAmount,description,openId,request);
         SortedMap<String, Object> finalpackage = new TreeMap<String, Object>();
         finalpackage.put("appId", ConfigManager.getInstance().getConfigItem("WXAppID")/*PayCommonUtil.APPID*/);
         finalpackage.put("mchId", ConfigManager.getInstance().getConfigItem("MCH_ID"));
         Long time = (System.currentTimeMillis() / 1000);
         finalpackage.put("timeStamp", time.toString());
         finalpackage.put("nonceStr", map.get("nonce_str"));
         finalpackage.put("prepayId", map.get("prepay_id"));
         finalpackage.put("package", "Sign=WXPay");
         finalpackage.put("signType", "MD5");
         String sign = PayCommonUtil.createSign("UTF-8", finalpackage);
         finalpackage.put("paySign", sign);//官方文档上是sign,当前示例代码是paySign 可能以前的

         WeiXinPrePay prePay = new WeiXinPrePay();
         prePay.setAppId(ConfigManager.getInstance().getConfigItem("WXAppID"));
         prePay.setMchId(ConfigManager.getInstance().getConfigItem("MCH_ID"));
         prePay.setTimeStamp(time.toString());
         prePay.setNonceStr(map.get("nonce_str"));
         prePay.setPrepayId(map.get("prepay_id"));
         prePay.setSignType("MD5");
         prePay.setPaySign("paySign");
         result.setData(prePay);
         result.setStateCode(GeneralConstant.SUCCESS);
         result.setDesc("微信支付加载成功");

         return result;
     } 

     /**
      * 统一下单
      * 应用场景:商户系统先调用该接口在微信支付服务后台生成预支付交易单,返回正确的预支付交易回话标识后再在APP里面调起支付。
      * @param trade_no
      * @param totalAmount
      * @param description
      * @param openid
      * @param sym
      * @param request
      * @return
      */
     @SuppressWarnings("unchecked")
     public Map<String, String> weixinPrePay(String trade_no,BigDecimal totalAmount,
             String description, String openid, HttpServletRequest request) {
         SortedMap<String, Object> parameterMap = new TreeMap<String, Object>();
         parameterMap.put("appid", ConfigManager.getInstance().getConfigItem("WXAppID"));  //应用appid
         parameterMap.put("mch_id", ConfigManager.getInstance().getConfigItem("MCH_ID")/*PayCommonUtil.MCH_ID*/);  //商户号
         //parameterMap.put("device_info", "WEB");
         parameterMap.put("nonce_str", randomString);
         parameterMap.put("body", description);
         parameterMap.put("out_trade_no", trade_no);
         parameterMap.put("fee_type", "CNY");
         System.out.println("jiner");
         BigDecimal total = totalAmount.multiply(new BigDecimal(100));  //接口中参数支付金额单位为【分】,参数值不能带小数,所以乘以100
         java.text.DecimalFormat df=new java.text.DecimalFormat("0");
         parameterMap.put("total_fee", df.format(total));
         System.out.println("jiner2");
         parameterMap.put("spbill_create_ip", PayCommonUtil.getRemoteHost(request));
         parameterMap.put("notify_url", wxnotify);
         parameterMap.put("trade_type", "APP");//"JSAPI"
         //trade_type为JSAPI是 openid为必填项
         //parameterMap.put("openid", openid);
         System.out.println("");
         String sign = PayCommonUtil.createSign("UTF-8", parameterMap);
         System.out.println("jiner2");
         parameterMap.put("sign", sign);
         String requestXML = PayCommonUtil.getRequestXml(parameterMap);
         System.out.println(requestXML);
         String result = PayCommonUtil.httpsRequest(
                 "https://api.mch.weixin.qq.com/pay/unifiedorder", "POST",
                 requestXML);
         System.out.println(result);
         Map<String, String> map = null;
         try {
             map = PayCommonUtil.doXMLParse(result);
         } catch (JDOMException e) {
             // TODO Auto-generated catch block
             e.printStackTrace();
         } catch (IOException e) {
             // TODO Auto-generated catch block
             e.printStackTrace();
         }
         return map;
     }

     /**
      * 此函数会被执行多次,如果支付状态已经修改为已支付,则下次再调的时候判断是否已经支付,如果已经支付了,则什么也执行
      * @param request
      * @param response
      * @return
      * @throws IOException
      * @throws JDOMException
      */
     @RequestMapping(value = "notifyWeiXinPay.htm", produces = MediaType.APPLICATION_JSON_VALUE)
    // @RequestDescription("支付回调地址")
     @ResponseBody
     public String notifyWeiXinPay(HttpServletRequest request, HttpServletResponse response) throws IOException, JDOMException {
         System.out.println("微信支付回调");
         InputStream inStream = request.getInputStream();
         ByteArrayOutputStream outSteam = new ByteArrayOutputStream();
         byte[] buffer = new byte[1024];
         int len = 0;
         while ((len = inStream.read(buffer)) != -1) {
             outSteam.write(buffer, 0, len);
         }
         String resultxml = new String(outSteam.toByteArray(), "utf-8");
         Map<String, String> params = PayCommonUtil.doXMLParse(resultxml);
         outSteam.close();
         inStream.close();

         Map<String,String> return_data = new HashMap<String,String>();
         if (!PayCommonUtil.isTenpaySign(params)) {
             // 支付失败
             return_data.put("return_code", "FAIL");
             return_data.put("return_msg", "return_code不正确");
             return StringUtil.GetMapToXML(return_data);
         } else {
             System.out.println("===============付款成功==============");
             // ------------------------------
             // 处理业务开始
             // ------------------------------
             // 此处处理订单状态,结合自己的订单数据完成订单状态的更新
             // ------------------------------

             String total_fee = params.get("total_fee");
             double v = Double.valueOf(total_fee) / 100;
             String out_trade_no = String.valueOf(Long.parseLong(params.get("out_trade_no").split("O")[0]));
             Date accountTime = DateUtil.stringtoDate(params.get("time_end"), "yyyyMMddHHmmss");
             String ordertime = DateUtil.dateToString(new Date(), "yyyy-MM-dd HH:mm:ss");
             String totalAmount = String.valueOf(v);
             String appId = params.get("appid");
             String tradeNo = params.get("transaction_id");

             return_data.put("return_code", "SUCCESS");
             return_data.put("return_msg", "OK");
             return StringUtil.GetMapToXML(return_data);
         }
     }

 }

本文demo下载

微信app支付(android端+java后台)的更多相关文章

  1. 微信app支付android客户端以及.net服务端实现

    由于公司运营需要,需要在客户端(android/ios)增加微信以及支付宝支付,在调用微信app支付时遇到一些问题,也算是一些踩过的坑,记录下来 ,希望能对.net开发者服务端网站更快的集成微信app ...

  2. 微信APP支付服务端开发Java版(一)

    一.准备工作 去微信开发者中心下载(扫码支付,里面的大部分代码是可以用的) https://pay.weixin.qq.com/wiki/doc/api/micropay.php?chapter=11 ...

  3. 微信支付-微信公众号支付,微信H5支付,微信APP支付,微信扫码支付

    在支付前,如果使用第三方MVC框架,则使用重写模式,服务器也需要配置该项 if (!-e $request_filename){ rewrite ^/(.*)$ /index.php/$ last; ...

  4. Android版-微信APP支付

    首发地址: Android版-微信APP支付 欢迎留言.转发 微信极速开发系列文章(微信支付.授权获取用户信息等):点击这里 目录 1.注册账号.开发者认证 2.添加应用 3.申请微信支付 4.技术开 ...

  5. .Net后台实现微信APP支付

    上一节分享了微信小程序支付的后台,这一节来分享一下微信APP支付的后台.微信APP支付和微信小程序差别不大,微信APP支付后台不需要微信登录凭证.后台下单时交易类型(trade_type)不再是&qu ...

  6. 微信APP支付整体流程记录备忘

      支付整体流程见文档:https://pay.weixin.qq.com/wiki/doc/api/app.php?chapter=8_3   商户系统和微信支付系统主要交互说明:     步骤1: ...

  7. php开发微信APP支付接口

    之前在开发APP中用到了微信支付,因为是第一次用,所以中途也遇到了好多问题,通过查看文档和搜集资料,终于完成了该功能的实现.在这里简单分享一下后台php接口的开发实例. 原文地址:代码汇个人博客 ht ...

  8. nodejs+koa2微信app支付,小程序支付

    企业付款到零钱文档:https://pay.weixin.qq.com/wiki/doc/api/tools/mch_pay.php?chapter=14_2 1,搞微信支付,先看流程图 https: ...

  9. H5使用codovar插件实现微信支付(微信APP支付模式,前端)

    H5打包的app实现微信支付及支付宝支付,本章主要详解微信支付,支付宝支付请查看另一篇“H5使用codovar插件实现支付宝支付(支付宝APP支付模式,前端)” ps:本文只试用H5开发的,微信 AP ...

随机推荐

  1. 如何修改maven默认仓库(即repository)的路径

    如何修改maven默认仓库(即repository)的路径 1  在maven的安装目录下,修改Eclipse(或IntelliJ IDEA)的MAVEN的存储位置,点击Browser按钮,选择set ...

  2. 理解spread运算符与rest参数

    理解spread运算符与rest参数 spread运算符与rest参数 是ES6的新语法.它们的作用是什么?能做什么事情? 1. rest运算符用于获取函数调用时传入的参数. function tes ...

  3. 在 ReactNative 的 App 中,集成 Bugly 你会遇到的一些坑

    一.前言 最近开新项目,准备尝试一下 ReactNative,所以前期做了一些调研工作,ReactNative 的优点非常的明显,可以做到跨平台,除了少部分 UI 效果可能需要对不同的平台进行单独适配 ...

  4. APP Store开发指南

    App Store 审核指南 iOS App打包上架超详细流程 ---2017.03 苹果对开发者提交的应用的审核之严格是出了名的,了解苹果的审核标准对于开发者防止应用被拒有着十分重要的意义.几天前苹 ...

  5. Android Looper原理分析

    实际业务使用场景: 某业务场景需要将本地数据传递到服务端,服务端再返回传递成功或者失败的信息. 1. 失败时: 重传5次 2.设置客户端请求的最小时间间隔,这个间隔内最多请求1次 具体逻辑如下:(这里 ...

  6. 解题思路:house robber i && ii && iii

    这系列题的背景:有个小偷要偷钱,每个屋内都有一定数额的钱,小偷要发家致富在北京买房的话势必要把所有屋子的钱都偷了,但是屋子之内装了警报器,在一定条件下会触发朝阳群众的电话,所以小偷必须聪明一点,才能保 ...

  7. sublime学习笔记

    学习课程地址:快乐的sublime编辑器_sublime编辑器使用 另可参考笔记地址:http://c.haoduoshipin.com/happysublime/ PS:博主的一些文章地址:http ...

  8. bzoj 4196: [Noi2015]软件包管理器

    Description Linux用户和OSX用户一定对软件包管理器不会陌生.通过软件包管理器,你可以通过一行命令安装某一个软件包,然后软件包管理器会帮助你从软件源下载软件包,同时自动解决所有的依赖( ...

  9. 怎么选择公司???MVC加jquery-easyui 后端工程师

    代码管理 Git 架构 MVC 这样的项目扩展性强,维护性强!!! 别很老的asp.net    不用框架!!!

  10. python net-snmp 的使用

    这一年一直在做一个综合管控平台的项目,用python写的,项目春节前可能就要进行实际部署了和测试,趁着这个空闲期,回顾一下项目中用到的一些技术,第一个就是SNMP协议. 项目结构主要是实现对ipran ...