支付宝接口的使用

异步通知和同步通知:

https://www.cnblogs.com/jhxk/articles/4826243.html

1:先创建测试环境(沙箱测试)。支付宝官网上有详细的步骤介绍。

2:下载官方 demo。

https://os.alipayobjects.com/rmsportal/kDBdHXwcnGxkVeBkomFL.zip

3:添加阿里巴巴支付接口的SDK

<!--阿里巴巴支付sdk-->
        <!-- https://mvnrepository.com/artifact/com.alipay.sdk/alipay-sdk-java -->
        <dependency>
            <groupId>com.alipay.sdk</groupId>
            <artifactId>alipay-sdk-java</artifactId>
            <version>3.1.0</version>
        </dependency>

4:把官方demo中 的AlipayConfig复制到自己的项目。然后把里面的参数修改成自己创建的测试环境的参数。

package com.betteryanwo.pay.config;


import java.io.FileWriter;
import java.io.IOException;

/* *
 *类名:AlipayConfig
 *功能:基础配置类
 *详细:设置帐户有关信息及返回路径
 *修改日期:2017-04-05
 *说明:
 *以下代码只是为了方便商户测试而提供的样例代码,商户可以根据自己网站的需要,按照技术文档编写,并非一定要使用该代码。
 *该代码仅供学习和研究支付宝接口使用,只是提供一个参考。
 */

public class AlipayConfig {

//↓↓↓↓↓↓↓↓↓↓请在这里配置您的基本信息↓↓↓↓↓↓↓↓↓↓↓↓↓↓↓

    // 应用ID,您的APPID,收款账号既是您的APPID对应支付宝账号
    public static String app_id = "2016091300504379";

    // 商户私钥,您的PKCS8格式RSA2私钥    (自己生成的私钥,记得去除空格,换行符)
    public static String merchant_private_key = "MIIEvwIBADANBgkqhkiG9w0BAQEFAASCBKkwggSlAgEAAoIBAQC1OQBO5rgfy+gGLHXfG3o68ZWGqRUhFAgtu0IJwTLk0DBH5tF2kzBJIe4shGW90eb0KUeiSkTKiaflzmj3DrxIuDODtYDijdsyQWJUU60HoiebMTkujDR19OWBZ8M8wOQY6LvSsxmh2Y6A27rUtgHhyAR5zapamEzFPipEemUudEQ3pJbBoD5HPgPbfF2XJfKnlKQZs20vWqn1NoKlVgFUjBGJUAgblX6GimDvDySoTuHotutekii54pWAFoFIspWJWE7m/V97V+MWOaftMrVQJFt5RoGawJ7xyD+wvL9A7PwV+C3yTTraGV4bqc1Tjf9DUC5GGtvKFKs8J53ksDNBAgMBAAECggEBAKQsJm5UL5uGkwT8xC/BacL6VrZueNjFl/8t9E53+s41GHgaz8l24Dhwh59GthD3lh29Q8rvM1C00iirDIY8kC/kx65bAI69akUl3Jl+UHNo4C6EskPL+j6eBEhuIv3n1PwH4xem7uKj/6gW5zOKSzwqgnuB6QE3ldzeS1ZL91vTqm1hpA13OIzwsCYQC9tldGqJ5zFXg5w8Eo80xDp7KZkfz9jLuGGZc34uKxXyW4FXdzPGS/s9NFuwEvcYygfI14ndB8wrQscV8k69EBl57sHcpBP9fMO6J3AAd0A5gN7FRayQNcx0ATxcl+gz0qYqCIubRlVmcDWr7y1q+nE9NB0CgYEA7O0tXEy6y1wx2azZhhdNEE5M7tCDWcTsrlPziqZhMlyHuY+jX3IvkhErPt8tQhrXHwLz2mTx9WYw5Gx1kLuekbW0YTNdLacxkh9Ue2wa1vB3pGPoKhMBaOpsw9cOBLODrbVpqYz3zoLou7MFQZCH96vvcO6cc/y5IaKn62Vay9sCgYEAw8/SrC+f+vT6CqaKK2x4/kopTDV5pkQjJIf3fe0DTfyTwJMGDT9kPMF116+k0ph+TwcliZsgROGKMTHLqljZW/Q355uug7xI38vlSWICQsfi8zs4UQxw1ET1B0NVQLGG5qJh2DqI8D5PrKVhg6a3rg+r06cdCVmm+WwPRkS+1hMCgYEAmFXfZotHR14eB1GmAxuURzmxKZQUAHIno+cCnlFgCVuJQPxkFQh8IbS8U453sRtE2gGx/OgO0rREF3rNFKQtzo5ATocSEDqCGuveDAV0NGMk6iP6sKLLs0OXb0wlDUzHC7erGoMzCisNrTHr3T4qzkpUiA5Dtif2ePP2d9oRSSUCgYEAvoFmNRGMsys+TbhjuwWo3bYnYbaxKRsnmbYTCtfaHDi9Q2GHRMJE8ntB/Fstn5qvYJHSaoObLIjF20DYJl6U8kqzTUmAyzgXKm0EIZYSHwi7++rEys2wxERmo+9VdUCCv8aCLU4dxqbI+25XZi+Aiv9CLARtUph/xDDm13WwuTMCgYAseLlZ6arMkwbE+tWlHeVB02jRhPKdXkZ/XdeZqZWq9cyWKggH/L7mKpY0h8NSEVv8tLcZ01p0JzZnBfAHIiqsC6HoH1Tz9Qqmzlg6TStnI8QiiNToJThQSzqWj4sm81j+0sR4oiUKcHGg762Q/g+8p7iJTh3bH2qUUpK55JJG/Q==";

    // 支付宝公钥,查看地址:https://openhome.alipay.com/platform/keyManage.htm 对应APPID下的支付宝公钥。
    public static String alipay_public_key = "MIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEAp2vE/iQ01uIH+IyML07iARpglpWuJUIsQZAesj8vh0MXxTBzOSrMUCndrA8lrSoY61Jni921K3jH8gH0p9iGZlFbm8uSAsXnHH1jhLWgSAxzBa2+wiAJ3A0v2CIypONSmzrdtgUBD6A4PMwJOBN1H73U7GUVQpctobM3sVJ9AtWnuxMls/9Naox/I3ANGzZlTPxFvxxQjBwUSXI8nY8rjJOXjdb3CFurMMP/dV7I23tyaFFUQeKICJNWMKzDILJ7xntvkwM20zY8dwSiYISc1waU4BAIHJcDtvQsY/d3zaWviLCk+pFy7j9qhM2N4rr+M7He4Cbd7nqaz1O2F86adQIDAQAB";

    // 服务器异步通知页面路径  需http://格式的完整路径,不能加?id=123这类自定义参数,必须外网可以正常访问
    public static String notify_url = "http://localhost:9000/alipay/notify_url";

    // 页面跳转同步通知页面路径 需http://格式的完整路径,不能加?id=123这类自定义参数,必须外网可以正常访问
    public static String return_url = "http://localhost:9000/alipay/return_url";

    // 签名方式
    public static String sign_type = "RSA2";

    // 字符编码格式
    public static String charset = "utf-8";

    // 支付宝网关
    public static String gatewayUrl = "https://openapi.alipaydev.com/gateway.do";

    // 支付宝网关
    public static String log_path = "/home/luo/logs";


//↑↑↑↑↑↑↑↑↑↑请在这里配置您的基本信息↑↑↑↑↑↑↑↑↑↑↑↑↑↑↑

    /**
     * 写日志,方便测试(看网站需求,也可以改成把记录存入数据库)
     * @param sWord 要写入日志里的文本内容
     */
    public static void logResult(String sWord) {
        FileWriter writer = null;
        try {
            writer = new FileWriter(log_path + "alipay_log_" + System.currentTimeMillis()+".txt");
            writer.write(sWord);
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
            if (writer != null) {
                try {
                    writer.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }
}

5:写controller层。

@Autowired
    OrderService orderService;
    private final Logger logger = LoggerFactory.getLogger(this.getClass());

    /**
     * 快捷支付调用支付宝支付接口
     *
     * @return
     * @throws IOException
     */


    @RequestMapping(value = "/aliPay/{orderSerial}")
    public String goPay(ModelMap map,
                        HttpSession session,
                        @PathVariable("orderSerial") String orderSerial){
        try{
            //获得初始化的AlipayClient
            AlipayClient alipayClient = new DefaultAlipayClient(AlipayConfig.gatewayUrl, AlipayConfig.app_id, AlipayConfig.merchant_private_key, "json", AlipayConfig.charset, AlipayConfig.alipay_public_key, AlipayConfig.sign_type);

            //设置请求参数
            AlipayTradePagePayRequest alipayRequest = new AlipayTradePagePayRequest();
            alipayRequest.setReturnUrl(AlipayConfig.return_url);
            alipayRequest.setNotifyUrl(AlipayConfig.notify_url);
            Thread.sleep(2000);
            Order order = orderService.getOrderByOrderSerial(orderSerial);
            // 封装请求支付信息
            AlipayTradePagePayModel model=new AlipayTradePagePayModel();
            //订单号
            model.setOutTradeNo(orderSerial);
            Users user = (Users) session.getAttribute("user");
            String subject=user.getUserName();
            //订单名称
            model.setSubject(subject);
            //价格:保留3位小数点
            model.setTotalAmount(order.getPrice().toString());
            //描述
            model.setBody(order.getDescription());
            model.setTimeoutExpress("");
            // 销售产品码 必填
            model.setProductCode("FAST_INSTANT_TRADE_PAY");
            alipayRequest.setBizModel(model);
            // 调用SDK生成表单
            String result = alipayClient.pageExecute(alipayRequest).getBody();
            map.put("result",result);
        }catch (Exception e){
            e.printStackTrace();
        }

        return "/pay/alipay";
    }

对应的JSP

<%@ page language="java" contentType="text/html; charset=utf-8" pageEncoding="utf-8"%>
<%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core"%>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html>
<head>
    <meta http-equiv="Content-Type" content="text/html; charset=utf-8">
    <title>付款</title>
</head>
<body>
${result}
</body>
</html>

支付宝同步通知。

/* *
     * 功能:支付宝服务器同步通知页面
     * 日期:2017-03-30
     * 说明:
     * 以下代码只是为了方便商户测试而提供的样例代码,商户可以根据自己网站的需要,按照技术文档编写,并非一定要使用该代码。
     * 该代码仅供学习和研究支付宝接口使用,只是提供一个参考。


     *************************页面功能说明*************************
     * 该页面仅做页面展示,业务逻辑处理请勿在该页面执行
     */
    @RequestMapping(value = "/return_url")
    public String returnUrl1(ModelMap map,HttpServletRequest request){
        try {
            //获取支付宝GET过来反馈信息
            Map<String,String> params = new HashMap<String,String>();
            Map<String,String[]> requestParams = request.getParameterMap();
            for (Iterator<String> 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] + ",";
                }
                //乱码解决,这段代码在出现乱码时使用
                valueStr = new String(valueStr.getBytes("ISO-8859-1"), "utf-8");
                params.put(name, valueStr);
            }
            boolean signVerified = AlipaySignature.rsaCheckV1(params, AlipayConfig.alipay_public_key, AlipayConfig.charset, AlipayConfig.sign_type); //调用SDK验证签名
            //——请在这里编写您的程序(以下代码仅作参考)——
            if(signVerified) {
                //商户订单号
                String out_trade_no = new String(request.getParameter("out_trade_no").getBytes("ISO-8859-1"),"UTF-8");
                //支付宝交易号
                String trade_no = new String(request.getParameter("trade_no").getBytes("ISO-8859-1"),"UTF-8");
                //付款金额
                String total_amount = new String(request.getParameter("total_amount").getBytes("ISO-8859-1"),"UTF-8");
                System.out.println("trade_no:"+trade_no+"<br/>out_trade_no:"+out_trade_no+"<br/>total_amount:"+total_amount);
                map.put("alipayResult", "支付宝充值成功");
            }else {
                System.out.println("验签失败");
                map.put("alipayResult", "支付宝充值失败");
            }

        }catch (Exception e){
            e.printStackTrace();
        }
        return "/pay/return_url";
    }

异步通知

/* *
     * 功能:支付宝服务器异步通知页面
     * 日期:2017-03-30
     * 说明:
     * 以下代码只是为了方便商户测试而提供的样例代码,商户可以根据自己网站的需要,按照技术文档编写,并非一定要使用该代码。
     * 该代码仅供学习和研究支付宝接口使用,只是提供一个参考。


     *************************页面功能说明*************************
     * 创建该页面文件时,请留心该页面文件中无任何HTML代码及空格。
     * 该页面不能在本机电脑测试,请到服务器上做测试。请确保外部可以访问该页面。
     * 如果没有收到该页面返回的 success
     * 建议该页面只做支付成功的业务逻辑处理,退款的处理请以调用退款查询接口的结果为准。
     */
    @RequestMapping(value = "/notify_url")
    public String notifyUrl(ModelMap map,HttpServletRequest request){
    try{
        //获取支付宝POST过来反馈信息
        Map<String,String> params = new HashMap<String,String>();
        Map<String,String[]> requestParams = request.getParameterMap();
        for (Iterator<String> 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] + ",";
            }
            //乱码解决,这段代码在出现乱码时使用
            valueStr = new String(valueStr.getBytes("ISO-8859-1"), "utf-8");
            params.put(name, valueStr);
        }
        boolean signVerified = AlipaySignature.rsaCheckV1(params, AlipayConfig.alipay_public_key, AlipayConfig.charset, AlipayConfig.sign_type); //调用SDK验证签名
        //——请在这里编写您的程序(以下代码仅作参考)——
        /* 实际验证过程建议商户务必添加以下校验:
        1、需要验证该通知数据中的out_trade_no是否为商户系统中创建的订单号,
        2、判断total_amount是否确实为该订单的实际金额(即商户订单创建时的金额),
        3、校验通知中的seller_id(或者seller_email) 是否为out_trade_no这笔单据的对应的操作方(有的时候,一个商户可能有多个seller_id/seller_email)
        4、验证app_id是否为该商户本身。
        */
        if(signVerified) {//验证成功
            //商户订单号
            String out_trade_no = new String(request.getParameter("out_trade_no").getBytes("ISO-8859-1"),"UTF-8");

            //支付宝交易号
            String trade_no = new String(request.getParameter("trade_no").getBytes("ISO-8859-1"),"UTF-8");

            //交易状态
            String trade_status = new String(request.getParameter("trade_status").getBytes("ISO-8859-1"),"UTF-8");

            if(trade_status.equals("TRADE_FINISHED")){
                //判断该笔订单是否在商户网站中已经做过处理
                //如果没有做过处理,根据订单号(out_trade_no)在商户网站的订单系统中查到该笔订单的详细,并执行商户的业务程序
                //如果有做过处理,不执行商户的业务程序

                //注意:
                //退款日期超过可退款期限后(如三个月可退款),支付宝系统发送该交易状态通知
            }else if (trade_status.equals("TRADE_SUCCESS")){
                //判断该笔订单是否在商户网站中已经做过处理
                //如果没有做过处理,根据订单号(out_trade_no)在商户网站的订单系统中查到该笔订单的详细,并执行商户的业务程序
                //如果有做过处理,不执行商户的业务程序

                //注意:
                //付款完成后,支付宝系统发送该交易状态通知
            }

            System.out.println("success");
            map.put("alipayResult","成功");
        }else {//验证失败
            System.out.println("fail");
            map.put("alipayResult","失败");
            //调试用,写文本函数记录程序运行情况是否正常
            //String sWord = AlipaySignature.getSignCheckContentV1(params);
            //AlipayConfig.logResult(sWord);
        }
    }catch (Exception e){

    }

        return "/pay/notify_url";
    }

对应的JSP   notify_url.jsp和return_url.jsp的内容是一样的。

<%@ page language="java" contentType="text/html; charset=UTF-8" pageEncoding="UTF-8"%>
<%@taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c"%>
<!DOCTYPE html>
<html>
    <head>
        <meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
        <title>在线扫码支付</title>
    </head>
    <body>
        ${alipayResult}
    </body>
</html>

 

posted @ 2018-07-05 15:01  陆伟  阅读(4578)  评论(0编辑  收藏  举报