JavaServer集成支付宝App支付、批量付款到支付宝账户简介文档

此文档写于2017年3月,只能说明此时该文档适用。使用前请查看以下接口支付宝是否提供。

  • 批量付款到支付宝账户
  • (链接如有发生变化,请在官方文档中寻找此产品,一般情况下,产品名不会发生改变)

1. App支付产品

  • 通俗上讲就是在App中使用支付宝付款,流程上就是:App请求接口(服务端),哪一个用户准备要买什么产品或者是要充多少钱,然后服务端拼接一些必要的参数返回给它,App端通过集成支付宝的SDK,根据接口返回的值去唤醒支付宝进行支付;与此同时,支付宝会异步通知服务端,哪一笔订单支付成功,服务端对充值后的逻辑做判断。(代码目前的做法)

  • 见图
    alipay
    alipay
  • 看完读可知,13,14步骤之前有10,图中的做法是在App端唤醒支付宝支付完成后的回调里请求接口,然后接口去请求支付宝,校验这笔订单完成与否,13,14的步骤为了完成服务端的逻辑。
    但实际上,支付宝是这样说的:

支付宝sdk对商户的请求支付数据处理完成后,会将结果同步反馈给商户app端。
同步返回的数据,商户可以按照下文描述的方式在服务端验证,验证通过后,可以认为本次用户付款成功。有些时候会出现商户app在支付宝付款阶段被关闭导致无法正确收到同步结果,此时支付结果可以完全依赖服务端的异步通知。
由于同步通知和异步通知都可以作为支付完成的凭证,且异步通知支付宝一定会确保发送给商户服务端。为了简化集成流程,商户可以将同步结果仅仅作为一个支付结束的通知(忽略执行校验),实际支付是否成功,完全依赖服务端异步通知

2. 批量付款到支付宝账户

  • 即提现;商户(业主)用自己的支付宝转给别人的支付宝账户。
  • 很多应用中涉及到提现,支付宝提现是一种普遍做法(知聊、美丽约等),但是该接口属于支付宝的历史接口,在很长的时间内没有更新,曾经问过客服,在不确定的某天,会将其重写。
  • 采用接口与后台相配合实现提现的功能,App端提交提现申请,后台(服务端)处理数据,跳往支付宝的逻辑,支付宝处理完毕后,同样给后台反馈,只会给成功与失败的反馈,这个地方有很多坑,具体请参见下面的描述。

强烈建议你(您),从官方的Api里寻求了解,获取帮助,再结合鄙人粗糙的代码进行完善或使用


3. App支付详解

1. 前期业主申请操作:
在支付宝 开放平台 注册账号:
  • 该账号必须为企业账号,部分功能只针对企业用户开放,有利于日后拓展;
  • 请业主提前准备:法定代表人信息、身份证照片、实际控制人信息、常用联系手机号码;企业证照(营业执照、组织机构代码证)照片;企业银行账号信息;
登录 开放平台 并创建应用:
  • 选择开发者中心—>概览—>创建支付应用—>创建—>填写应用名称[App名称] —>跳入完善信息页面;
  • 完善信息页面的[功能列表]选择我是商户,功能只留一个APP支付即可,将其他的删除[后期可根据需要实时添加,此处不必申请太多];
  • [开发配置]里的基础环境无需配置,接口加签方式的配置请参考同目录下的[支付宝接口加签方式说明];
  • 配置完这一步可以获得3个参数:支付宝的公钥商户私钥接口加签方式[RSA或RSA2];
  • 配置好后进行提交审核;这时返回应用列表会看到刚刚创建的应用,也有对应的APPID
应用审核的同时,需要签约APP支付产品:
  • 在应用列表里点开创建的应用,在功能列表APP支付右上方有立即签约字样,或者登录支付宝商家中心,单独点击签约APP支付产品;
  • 这里同微信支付类似,签约需要提供APP的信息,支付宝需要一个文档,把整个在APP中的支付流程说明。[需要特别注意自己的应用是否涉及到了网络文化经营许可证的范畴,人工审核会判别这一点,如果是需要提供此证。如申请不下这个证,则会有一定的法律风险,且不好通过]
其他说明:
  • 但凡需要人工审核的地方都需要时间,请业主提早申请,避免延误;
  • 另外还有3个参数,商户支付宝账户[一般为邮箱,开放平台的登录账号]、真实姓名[一般为xxx有限公司]、合作伙伴身份[开放平台右上角用户标识下的账号管理里的mapi网关产品密钥的PID,以2088开头的16位字符];
  • 以上加粗标识[除了标题]的关键值均需业主提供给程序员,在开发中会用到;
2. 支付流程示例:
  • 商户APP-----向商户服务端发起请求--1--商户服务端--2—返回商户APP;
  • 商户APP根据返回值调起支付宝支付,消费人支付成功;
  • 支付宝服务端-----3-----向商户服务端发起请求,商户服务端完成自身业务;
  • 其他:数字标志的含义,1代表处理自身业务,2代表按照支付宝要求调用SDK拼接参数返回APP,3代表按支付宝要求验证参数,完成自己业务。
3.代码结构以及使用说明:
下载:

包含了微信支付和支付宝支付,密码为phqa

结构:
示例代码结构
使用说明:
  • 操作者只需要填写Config所需的信息,根据自身业务抒写Controller、Service层即可,其他结构无需修改;在Service里调用ALiPayCore进行支付宝必要的步骤即可。
  • ALiPayConfig配置参数概览:均为字符串
参数名 释义或固定值 取得方式或固定值
partner 合作身份者ID以2088开头由16位纯数字组成的字符串 参考前期业主申请操作
appid 应用id 同上
seller_email 商户支付宝账号 同上
account_name 商户真实姓名 同上
private_key 商户的私钥RSA 同上
ali_public_key 支付宝的公钥 RSA 同上
pay_sign_type 签名方式 现在支付宝支持2个值RSA或RSA2,该值须和私钥公钥生成方式一致。
notify_url 支付宝服务器主动通知商户服务器里指定的页面http/https路径 参考外网映射的配置方式https://www.ngrok.cc/
subject 商品的标题/交易标题/订单标题/订单关键字等 栗子:索尼-相机购买
以下值为固定值 --- ---
method 接口名称 alipay.trade.app.pay
version version 1.0
product_code 销售产品码,商家和支付宝签约的产品码 QUICK_MSECURITY_PAY
input_charset 字符编码格式 目前支持 gbk 或 utf-8 utf-8
  • 在Controller层抒写两个业务,处理APP调起支付和支付宝的回调;调起App的接口可以自定义参数,接收支付宝的回调参数无需修改。在Service里抒写自己的具体业务,然后调用固定方法进行处理。

  • eg:A用户要购买100元的衣服,提供给App的接口需要接收用户的id,衣服的id,在Service层里处理订单的业务之后,调用ALiPayCore.createAliPayStr (out_trade_no,total_fee)方法,传入商户内部订单号和用户所要支付的金额,并将返回值返给前端即可;
    前端接收到返回值调起支付宝SDK,用户支付完毕;
    支付宝会异步通知在Config里配置的回调地址,在回调的Service层里,调用ALiPayCore.checkAliPayParam(request)进行验证,判断是否是支付宝发送来的通知以及是否支付成功,返回值不为NULL即可进行商品发货或者填写快递订单号等业务。

  • 两个核心方法使用介绍:

1.String orderString = ALiPayCore.createAliPayStr (String out_trade_no,String total_fee);

a) 拼接支付宝支付必要参数,加签,并返回字符串;
b) out_trade_no 传入商户内部订单号,必须唯一;total_fee 所要支付的金额,格式必须为0.00,单位为元;
c) 返回App端调起支付所需要的参数。

2.Map<String, String>  aliParam = ALiPayCore.checkAliPayParam (request);

a) 支付成功的回调:检测支付宝异步反馈是否真实,不为null为真实,null为校验失败。
b) 返回的map中可以取得支付宝返回的参数,一般只取out_trade_no(商户内部订单号)、total_amount(充值的金额,单位为元)、trade_no(支付宝订单号)。
c) 判断aliParam是否不为NULL,不为NULL拿商户内部订单号去查出充值的订单数据,完成自己的业务。

  • 特别注意:在处理的回调里,处理成功后需要向支付宝反馈成功:
    Eg:out.print(“success”); *
    如果处理失败则无需反馈;支付宝如果接收不到正确反馈,则会按照一定策略重复通知,代码中应该有
    去重处理[判断该订单是否已经处理过*]
注意事项:

ALiPayCore类里的方法均不需要修改,和业务无关,直接调用即可。

引用jar:

集成支付宝接口需要引入的文件是:

  • alipay-sdk-java*.jar
  • commons-logging-1.1.1.jar

若进一步了解代码实现请引入文件:

  • alipay-sdk-java*-source.jar
  • commons-logging-1.1.1-sources.jar
4.备注以及拓展:
文档的第一步骤可以截取发给业主参考申请,让业主提供加粗的参数信息;
  • 本着程序员应全身心在开发上,避免不必要的时间浪费。
对ALiPayCore.createAliPayStr()方法的解释:
  • 方法的作用就是按照支付宝的要求返回字符串,APP端根据此返回值调起支付宝;
  • 字符串里包含着订单信息,[支付宝要扣多少钱、支付宝服务端给哪个地址发送支付成功的通知];
  • 不论是支付宝还是微信都有自己的一套规则,但大体上都很相似:
    1.将参数的key值按照ascii码的顺序进行排序,并转化成key=value&key=value的形式;
    2.然后支付宝会让调用各自语言的SHA加密算法,利用自己的私钥对刚刚的key=value进行加密得到sign参数,再将sign当作key,拼入字符串中;
    3.为了防止中文value乱码,需要对所有的value进行URLEncoder.encode。
  • 支付宝以上的3步是为了确保这个通知是商户的自身发送出来的,拿私钥加签实际上是采用了非对称性加密的方式,确保唯一性。[如果支付宝不这么做,订单信息被别人篡改了,对双方都将是最坏的结果]。
对ALiPayCore.checkAliPayParam()方法的解释:
  • 方法的作用就是检测是否是支付宝发送来的通知,根据此通知进行支付成功的业务;
  • 支付宝是不会发送支付失败的通知的;
  • 验证也有相应的规则:
    1.将支付宝的参数都取出[这里有个拓展,requesr.getgetParameterMap()就可以取出Map形式的参数];
    2.剔除sign和sign_type字段,变成按照ascii码排序的key=value&形式;
    3.调用各自语言对应的验签方法,这一步拿支付宝公钥进行验证返回布尔值;
    4.以上的步骤其实也是非对称性的验证方式。
  • 同理,支付宝需要这样的验证步骤,也是为了确保该通知是支付宝发出的,以防别人冒发送请求;
  • 支付宝防止意外,还让对订单号、商户的基本信息做验证。
拓展:
  • 关于数字签名这里有一篇文章介绍的很全面。
  • 也推荐关注该文章的翻译者:阮一峰。
新发现:

重新修改此文时,发现支付宝更新了SDK的介绍页面,在这个链接里,支付宝也介绍了如何通过调用SDK完成服务端的工作。


4. 批量付款到支付宝账户详解

1. 前期业主申请操作:
  • 找文档

这个藏得很深的,因为在支付宝的明面上看不到它,不像App支付,在类似产品大全的页面能找到;首先进入支付宝的 文档中心 在左侧找到 历史接口 ,但愿你看到的还是这个样子,如果在那一栏底下也标注了新的接口,那么你就重写它吧(笑)。

历史接口

  • 业主签约

由于属于历史接口,支付宝的考量可能是不建议新商家接入,所以签约的方式比较繁杂,需要App支付产品签约成功后,再找人工客服签约。
人工客服可以通过登录支付宝开放平台,如图1所示的地方唤醒;唤醒的是一只机器人,和它聊三句以上的话,会提醒接入人工,如图2。然后就可以和人工说我需要接入批量付款到支付宝账户这个有密接口,跟着她的节奏来进行了。这一步也是需要业主完成的,因为有业主需要考量的东西,程序员做不了主。

图1

图2

人工客服大概会说,该接口每笔会收取0.5%的手续费,不满1元按1元算,最高不会超过25元,您同意吗?,同意就签约了。一般这个费用是转嫁给用户的,看业主如何考量了。

2.程序员进行配置、抒写:
  • 获取PID、MD5Key

pid就不说了,以2088开头的数字;
配置密钥, 参照文档 上指引的地方,把MD5那个key保存下来,用于请求数据的签名和支付宝返回数据的验签(后期才注意到,在“提现”这一模块,签名是用MD5签名的)。这里就不放图了,支付宝的网页跳来跳去,而且经常改动,无法确切它的位置,文档里倒是更新开很快,这点比某信要强多了。

  • 简单综述一下:

用户,在App中填写好自己的支付宝真实姓名以及对应的支付宝账户,然后在提现那个地方,输上提现的金额,点击提交,接口就接收了这个数据,做一定的处理(比如转嫁手续费什么的),在提现表里有一个状态标识是提现申请、处理中、成功、失败;后台处理申请的提现,拼接一些参数给支付宝,处理成功后,支付宝给回馈,成功的成功,失败的失败。成功,失败拿系统消息作为提示,失败之后返回给用户金钱,可以重新提交申请。失败的原因可能是用户支付宝填的不正确,正常情况下是这样的。但是有很多意外情况,也就带来了很多坑,下面会拿粗体涉及。

  • App接口处理方面

对简单的参数做验证后,查看用户是否绑定了支付宝账户信息,以及用户的提现的钱是否符合够用(项目里是最低标准是5元)。

/**
     * 提现申请
     * @return
     */
    @ResponseBody
    @RequestMapping("applicationWithdraw")
    public Object applicationWithdraw(Integer pid, BigDecimal money){
        //验证参数
        if(ObjectUtil.isPassInteger(pid) && money != null && money.compareTo(new BigDecimal(5)) != -1){
            if(aliacount_dao.selectByPersionId(pid).size()!=0){//用户已绑定支付宝
                Persion persion = persion_dao.queryPersionByIdForIM(pid);
                if(money.compareTo(new BigDecimal(persion.getCapitalBalance())) != 1){
                    // 申请提现业务
                    boolean flag = wiService.insertApplication(persion, money);
                    return flag==true?"success":"error";
                }else{
                    return "error";//用户金额不足
                }
            }else{//用户未绑定支付宝
                return "fail";
            }
        }else{
            return "error";
        }
    }

下面这块给拼接了一个流水号,在后面会有介绍;然后转嫁手续费到用户身上,插记录,扣钱;以事务的方式进行控制。关于用户手续费方面,其实还可以更加灵活一些,无非就是多提少收,少提多收;这个也得业主考量了。

**
     * 申请提现业务
     * @return
     */
    @Transactional
    public boolean insertApplication(Persion persion,BigDecimal money){
        String onceId = UUIDHashCode.getOrderIdByUUId();//获得一个随机唯一标识
        //扣除支付宝的手续费 0.5% 最高不会超过25元 最低不会超过1元
        BigDecimal platform_cost =  money.multiply(new BigDecimal(0.005)).setScale(2, BigDecimal.ROUND_HALF_UP);
        if(platform_cost.compareTo(new BigDecimal(25)) == 1){
            platform_cost = new BigDecimal(25);
        }else if(platform_cost.compareTo(new BigDecimal(1)) == -1){
            platform_cost = new BigDecimal(1);
        }
        BigDecimal getMoney = money.subtract(platform_cost);//扣除手续费之后的金额
        int num = 0 ;
        if(getMoney.compareTo(new BigDecimal(0)) == 1){
            // 新增一条提现申请记录
            num = withMa.insertSelective(new Withdraw(onceId,persion.getId(),getMoney,1,platform_cost));
        }
        if(num == 1){
            // 修改该用户信息中可提现金额信息
            persion.setCapitalBalance(money.doubleValue());
            Persion queryPersion = perMa.queryPersionByIdForIM(persion.getId());
            if(queryPersion != null){
                persion.setVersion(queryPersion.getVersion());
                int withdr = perMa.cutPersionCap(persion);
                return withdr>0?true:false;
            }else{
                num = 1/0;
                return false;
            }
        }else{
            num = 1/0;
            return false;
        }
    }
  • 后台进行处理

因为这个提现还是需要人工操作的,比如输支付密码什么的,支付宝的考量也可能是为了安全吧,所以需要有后台的一系列逻辑。像App支付一样,先从基本信息配置入手。
AlipayConfig 这里关键在于私钥和公钥,可以看一看 阮一峰的数字签名是什么 支付宝App支付的公钥和这里提现所用的公钥是不一致的,对应的RSA的与RSA2的公钥也是不同的。之前不了解公钥的意义,这里的公钥和App支付的共用了,也算是无知的坑吧。商户安全校验码就是上面让保存的那个MD5的key值。

public class AlipayConfig {

    //↓↓↓↓↓↓↓↓↓↓请在这里配置您的基本信息↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓
    // 合作身份者ID,以2088开头由16位纯数字组成的字符串
    public static String partner = "2088xxxxxxx";
    //商户支付宝账号
    public static String seller_email = "";
    //商户真实姓名
    public static String account_name = " XXXXX科技有限公司";
    // 支付宝用于提现的公钥,一般情况下无需修改该值  (提现),也可与支付宝map网关产品密钥的支付宝公钥做对比,应该是一致的。
    public static String ali_public_key = "MIGfMA0GCSqGSIb3DQEBAQUAA4GNADCBiQKBgQCnxj/9qwVfgoUh/y2W89L6BkRAFljhNhgPdyPuBV64bfQNN1PjbCzkIM6qRdKBoLPXmKKMiFYnkd6rAoprih3/PrQEB/VsW8OoM8fxn67UDYuyBTqA23MML9q1+ilIZwBC2AQ2UBVOrFXfFl75p6/B5KsiNG9zpgmLCUYuLkxpLQIDAQAB";
    //商户安全校验码
    public static String key = "MD5key";
    //支付宝异步通知地址    需http://格式的完整路径,不允许加?id=123这类自定义参数
    public static String notify_url = PathUtil.GetDemain() + "/WoBanAdmin/ALiPay/TransNotify";
    //public static String notify_url = "http://xiaofanfight.viphk.ngrok.org/WoBanAdmin/ALiPay/TransNotify";
    
    //↑↑↑↑↑↑↑↑↑↑请在这里配置您的基本信息↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑

    // 调试用,创建TXT日志文件夹路径
    public static String log_path = "D:\\";

    // 字符编码格式 目前支持 gbk 或 utf-8
    public static String input_charset = "utf-8";

    // 签名方式 不需修改(退款签名方式)
    public static String sign_type = "MD5";
}

代码里有很多是运用支付宝提供的Demo,支付宝并没有对这个接口提供jar包公司的前辈也对此进行了开创性的研究,而我只是完善了一下,并写成了拙劣的文档。
后台的流程及页面展示:
页面是位于pages - recharge 里的。

提现申请列表 money_drawing.jsp

这里每进一次页面会有提示,因为很重要,所以每次都提醒;提示框的蓝色按钮可以跳往支付宝对证书的提示,它会说,目前只支持IE 32位部分浏览器以及UC浏览器,Mac下只支持safari;但我实际测试的时候,只支持火狐......支付宝需要证书是为了后面你输入支付宝支付密码时候的安全。
提现申请详情 BalanceDetail.jsp

一个的时候展示一个的详情,多人的时候展示多人的信息,点击结算后的页面是以弹出层展示的
点击结算后 HandleClearing.jsp

这里的话其实不加验证也可以的,毕竟支付宝那边还要验证支付密码,或者这里也可以结合一下管理员的手机号,进行处理。
输入密码,点击确定后,就走Controller了,从控制层里拼接一些参数,跳往支付宝页面。下面详述ALiPayController
实际上,在上一张图我们可知,能传过来的数据无非就几个:
1.哪个管理员操作的;
2.哪些人申请的提现。
因为像申请的钱什么的是依赖表的而不是页面上的数据。
参数ids实际上控制着三个选择,全部处理、按照选择处理、单个处理;然后一些基础信息的取得,计算提现的总额(依靠SQL),最为关键就是拼接参数,按照文档进行,参数中最为关键的就是 付款详细数据 以及 批次号流水号,批次号每次都是随机生成的,因为支付宝将相同的批次号视作同一笔申请,而流水号在App端提交的时候就做了处理;这里不需要关注一些细节,在支付宝提供的Demo里有涉及,有兴趣可以点进去看一看。

 /**
     * 
     * <b>Method ALiPayTrans</b>
     * <dd>方法作用:请求批量转账前一步
     * <dd>适用条件:提现
     * <dd>使用方法:请参照最新的支付宝批量转账有密接口文档
     * @param aid    管理员id
     * @param ids     提现申请人id的集合
     * @param out     用于给支付宝回馈(大概吧)
     * @since Met 2.0
     */
    @RequestMapping(value = "ALiPayTrans", method = RequestMethod.GET)
    public void ALiPayTrans(Integer aid, String ids,PrintWriter out) {
        
        DecimalFormat df = new DecimalFormat(".00");//将double类型的数据保留两位数
        List<Withdraw> wlist = new ArrayList<Withdraw>();
        if(ids!=null && !ids.equals("")){
            if(ids.equals("all")){//全部处理
                wlist = withdrawMapper.selectWithdraw(new Withdraw());
            }else{
                if(ids.contains(",")){
                    String Ids=ids.trim().substring(0, ids.length() - 1);
                    
                    Withdraw withdraw = new Withdraw();
                    withdraw.setId(0);
                    withdraw.setBatchNo(Ids);
                    wlist = withdrawMapper.selectWithdraw(withdraw);
                }else{//单个流水
                    wlist = withdrawMapper.selectWithdraw(new Withdraw(Integer.parseInt(ids)));
                }
            }
            //服务器异步通知页面路径   通知提现成功与失败
            String notify_url = AlipayConfig.notify_url;
            //付款账号
            String email = AlipayConfig.seller_email;
            //付款账户名
            String account_name = AlipayConfig.account_name;
            //必填,个人支付宝账号是真实姓名公司支付宝账号是公司名称

            //付款当天日期
            String pay_date = DateUtil.getDays();
            //必填,格式:年[4位]月[2位]日[2位],如:20100801
            //批次号
            String batch_no = DateUtil.getDays() + DateUtil.getThree() + DateUtil.getThree();
            //必填,格式:当天日期[8位]+序列号[3至16位],如:201008010000001
            //付款总金额
            //计算总额
            BigDecimal summoney = withdrawMapper.querySumByGet(wlist);
            
            if(summoney != null && summoney.compareTo(new BigDecimal(10000000)) == -1){
                String batch_fee = df.format(summoney).toString();
                 //必填,即参数detail_data的值中所有金额的总和
                //付款笔数
                Integer num = wlist.size();
                String batch_num = num.toString();
                //必填,即参数detail_data的值中,“|”字符出现的数量加1,最大支持1000笔(即“|”字符出现的数量999个)

                /*
                 * 下列付款详细数据说明以及示例
                 * String detail_data = batch_no + "^" + "zhangsan@qq.com" + "^" + "张三" + "^" + batch_fee + "^备注说明";
                 * 解释:其中batch_no为上面生成的转账批次号;zhangsan@qq.com为需要转账的支付宝账户;张三为转账支付宝账户的真实姓名;
                 * batch_fee为转账金额,最后的参数为附加参数,可以对本次转账备注说明,只要是字符串就可以,但长度不宜过长。请根据需要以此替换
                 */

                //必填,即参数detail_data的值中,“|”字符出现的数量加1,最大支持1000笔(即“|”字符出现的数量999个)
                //付款详细数据
                String detail_data = "";
                for(int i=0;i<wlist.size();i++){
                    detail_data += wlist.get(i).getOnceid() + "^" + wlist.get(i).getAccount() + "^" + wlist.get(i).getRealname() + "^" + wlist.get(i).getMoney() + "^平台手续费"+wlist.get(i).getPlatform_cost()+"元"+"|";
                }
                detail_data = detail_data.substring(0,detail_data.length()-1);
                //batch_no + "^" + "收款方账号" + "^" + "收款方真实姓名" + "^" + batch_fee + "^备注说明";
                //必填,格式:流水号1^收款方帐号1^真实姓名^付款金额1^备注说明1|流水号2^收款方帐号2^真实姓名^付款金额2^备注说明2....
                
                //修改提现信息
                for(int j=0;j<wlist.size();j++){
                    Withdraw mapper = new Withdraw(wlist.get(j).getId(), batch_no, aid);
                    mapper.setState(2);
                    withdrawMapper.updateWithdraw(mapper);
                }

                //把请求参数打包成数组
                Map<String, String> sParaTemp = new HashMap<String, String>();
                sParaTemp.put("service", "batch_trans_notify");
                sParaTemp.put("partner", AlipayConfig.partner);
                sParaTemp.put("_input_charset", AlipayConfig.input_charset);
                sParaTemp.put("notify_url", notify_url);
                sParaTemp.put("email", email);
                sParaTemp.put("account_name", account_name);
                sParaTemp.put("pay_date", pay_date);
                sParaTemp.put("batch_no", batch_no);
                sParaTemp.put("batch_fee", batch_fee);
                sParaTemp.put("batch_num", batch_num);
                sParaTemp.put("detail_data", detail_data);
                //建立请求
                //
                Log log = new Log(aid,"处理提现申请操作,处理额度为" + summoney + "元");
                logMa.insert(log);
                
                String sHtmlText = AlipaySubmit.buildRequest(sParaTemp, "post", "确认");
                out.println(sHtmlText);
            }else{
                out.print("<script>alert('处理批次额度超限。最高不超过1000万元。');</script>");
            }
        }
    }

顺利就跳到了以下的页面,值得注意的是,在这个Controller内,将这笔提现申请改成了处理中,避免其他人会重复处理。请注意这个状态。

支付宝页面

支付宝提示当前操作环境不支持支付宝控件,因为我是谷歌浏览器打开的,这种情况支付宝不会给异步通知,因为它不知道是这种问题,它会通知的情况只有两种:
1.支付成功;
2.支付失败,给转账的支付宝用户信息不正确。
所以处理中的作用就是存放意外情况下的申请,管理员可以在合适的环境下,将处理中的申请再次转化为申请中,再次提交。
申请中列表

支付失败之一 收款支付宝账户信息校验不通过

以下为出现意外的情况截图:
不会通知的情况 未安装证书

上一步安装了之后有刷新选项,但是支付宝会这样说,也不需要管,因为刚刚那笔订单已经在提现申请中里了

不会通知的情况 安装了证书之后未安装电子证书

正常情况下:
安装了电子证书 正常的情况1

下面这个图就说明了,支付宝会给打款成功的回调。
安装了电子证书 正常的情况2

支付宝的异步通知

刚刚问了半天客服,客服也无法说清楚,建议是在IE8 32位浏览器下进行的,可是IE8会和layui会有冲突,这就有点尴尬了。以上的截图是在火狐下进行的。客服也不置可否。
完成了支付宝付款的逻辑,就剩下接收回调的处理了,也在Controller层,对于成功走成功逻辑,失败走失败逻辑。
这里关注一下支付宝的说明即可:如果成功的信息为空,证明都失败了,反之;如果两者都不为空,就需要各自走各自的逻辑了,根据流水号查出提现的详情,成功推送信息,失败返回资金并推送
回调处理的关键

@RequestMapping(value = "TransNotify", method = RequestMethod.POST)
    /**
     * 批量付款数据中转账成功的详细信息 String success_details 
     * 批量付款数据中转账失败的详细信息 String fail_details
     * 批量付款数据中转账批次号 String batch_no
     */
    public void TransNotify(HttpServletRequest request, String success_details, String batch_no,
            String fail_details, PrintWriter out) {
        //获取支付宝POST过来反馈信息
        Map<String, String> params = GetInfoFromALiPay(request);
        
        boolean flag = true;
        
        if(AlipayNotify.verify(params)){
             //验证成功
            //判断是否在商户网站中已经做过了这次通知返回的处理;如果没有做过处理,那么执行商户的业务程序;如果有做过处理,那么不执行商户的业务程序
            //可以判断success_details是否为null来标识转账是否成功,支付宝方面明确说明如果转账成功success_details不为null,fail_details则
            //为null;若转账失败success_details为null而fail_details不为null,同样根据batch_no来查询转账对象并更新转账状态
            
            if(fail_details == null){
                //提现全部成功 处理相关业务 看是否已经处理过了 改状态  推送
                withdrawService.getMoneySuccess(batch_no);
            }else if(success_details == null){
                //提现全部失败
                //返还未提现用户的金币 推送
                String info = withdrawService.getMoneyError(batch_no);
                if("error".equals(info)){
                     flag = false;//自己sql出错,请求支付宝再次发送验证
                }
            }else{
                //转账部分成功/部分失败
                String info = withdrawService.getMoneySuccessOrError(batch_no, success_details, fail_details);
                if("error".equals(info)){
                 flag = false;//自己sql出错,请求支付宝再次发送验证
                }
            }
            
            if(flag){
                 out.print("success");//请勿修改该值!
            }else{
                out.print("fail");//自己sql出错,请求支付宝再次发送验证
            }
           
        }else{
            //验证失败 
            //程序执行完后必须打印输出“success”(不包含引号)。如果商户反馈给支付宝的字符不是success这7个字符,支付宝服务器会不断重发通知,直到超过24小时22分钟
            out.print("fail");
        }
    }

    @SuppressWarnings("unchecked")
    public Map<String, String> GetInfoFromALiPay(HttpServletRequest request) {
        Map<String, String> params = new HashMap<String, String>();
        Map requestParams = request.getParameterMap();
        for (Iterator iter = requestParams.keySet().iterator(); iter.hasNext();) {
            String name = (String) iter.next();
            String[] values = (String[]) requestParams.get(name);
            String valueStr = "";
            for (int i = 0; i < values.length; i++) {
                valueStr = (i == values.length - 1) ? valueStr + values[i] : valueStr + values[i]
                        + ",";
            }
            //乱码解决,这段代码在出现乱码时使用。如果mysign和sign不相等也可以使用这段代码转化
            //valueStr = new String(valueStr.getBytes("ISO-8859-1"), "gbk");
            params.put(name, valueStr);
        }
        return params;
    }

5. 结束

第一次总结,文字还是很冗余的。在这里还是需要感谢一下ydm公司的前辈。写于2017-03-17。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 206,126评论 6 481
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 88,254评论 2 382
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 152,445评论 0 341
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 55,185评论 1 278
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 64,178评论 5 371
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,970评论 1 284
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,276评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,927评论 0 259
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,400评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,883评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,997评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,646评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,213评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,204评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,423评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,423评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,722评论 2 345

推荐阅读更多精彩内容