赞
踩
在许多网购商品项目中,不难会遇到用户支付的场景,但若要自己开发一个运用在支付场景的支付包,实属有些困难,但是也不乏一些大厂自己集成的环境。支付宝有一个供开发者测试使用的沙箱环境,会提供一个沙箱版的支付宝app、一个商家账户、一个买家账户。有了这个,可以让我们跳过商家入驻、企业资质审核等过程,开箱即用。首先进入到支付宝支付官网,点击“我是开发者”,在新的页面右上角,用你自己的支付宝扫码登录,再点击开发服务中的研发服务,获取APPID和公钥私钥。这里不再赘述。
首先导入依赖及配置支付包沙箱配置类:
<!-- 引入支付宝沙箱支付依赖-->
<dependency>
<groupId>com.alipay.sdk</groupId>
<artifactId>alipay-sdk-java</artifactId>
<version>4.23.26.ALL</version>
</dependency>
package com.cetus.computerstore.config; import lombok.Data; import org.springframework.stereotype.Component; /** * @author xiaojiang * @version 1.0 * @description: 支付宝沙箱支付配置类 * @date 2022/7/25 12:29 */ @Data @Component public class AlipayConfig { //自己的appId public static String appId = "2021000121634026"; //应用私有秘钥,这里删除了一些数据 public static String appPrivateKey = "MIIEvQIBADANBYu0s1MmIatigcJ6A7FyvJ+nnow+qr4tegi5CG76v+Ue/IrgZXJZyDqMBrHorn/SwRXleOj2gjfxkQ8mh0OkrIntbdD6SDnvtXkxcVkfX9ICY5VIA4DA="; //支付宝公钥 public static String alipayPublicKey = "MIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEAjtedJ8C1NIO3r4vuMThcvTMZxqO3Jki5VYPfkmnraA/PIKXXgsfOdoSWxCsiqPMIBMRT3Oyk1EsxgAeFBKTFaSM5LC8oinTXFbkv+3XEuOjtfqbp0oIgu9pWfJQDL2gIVSbm3VKmdE4UtJ36nu3hyuTT3U19QQsKVgxMDWHCOIw0eCHcJm1xDPj0zmagL3jC7576sXHcnFxEKARGugMpP9bkBgvFkjKrnkQfMAz3OO8vUSC0lCGo2UrSwhyD6zqXVz39sIduVpKTTg+wpAJQ/RhBhLXNw4JW3UaZpX2BZbmqEx91Hpr+O/95Z90cTqT+rwyu6uW612B5bCPnKa+BCQIDAQAB"; //异步回调地址 public static String notifyUrl = "http://1.14.176.219:8080/alipay/notifyNotice"; //同步回调地址 public static String returnUrl = "http://1.14.176.219:8080/alipay/returnNotice"; //推荐使用这个秘钥 public static String signType = "RSA2"; //使用的编码格式 public static String charset = "utf-8"; //支付宝默认网关 public static String gatewayUrl = "https://openapi.alipaydev.com/gateway.do"; }
再是编写controller层代码:
package com.cetus.computerstore.controller; import com.alipay.api.AlipayClient; import com.alipay.api.DefaultAlipayClient; import com.alipay.api.internal.util.AlipaySignature; import com.alipay.api.request.AlipayTradePagePayRequest; import lombok.extern.slf4j.Slf4j; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Controller; import org.springframework.web.bind.annotation.GetMapping; import org.springframework.web.bind.annotation.PostMapping; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.ResponseBody; import top.year21.computerstore.config.AlipayConfig; import top.year21.computerstore.service.IOrderService; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.net.URLDecoder; import java.net.URLEncoder; import java.util.HashMap; import java.util.Iterator; import java.util.Map; /** * @author xiaojiang * @version 1.0 * @description: 处理支付宝沙箱支付的接口 * @date 2022/7/25 1:42 */ @Slf4j @Controller @RequestMapping("/alipay") public class AliPayController { @Autowired private IOrderService orderService; /** * Description : 处理在线支付的请求 * @date 2022/7/25 * @param oid 订单id * @param totalPrice 订单总金额 * @return java.lang.String **/ @ResponseBody @GetMapping(value = "/pay",produces = "text/html;charset=UTF-8") public String goAlipay(String oid,String totalPrice,HttpServletRequest request) throws Exception { // 获得初始化的AlipayClient AlipayClient alipayClient = new DefaultAlipayClient(AlipayConfig.gatewayUrl, AlipayConfig.appId, AlipayConfig.appPrivateKey, "json", AlipayConfig.charset, AlipayConfig.alipayPublicKey, AlipayConfig.signType); // 设置请求参数 AlipayTradePagePayRequest alipayRequest = new AlipayTradePagePayRequest(); alipayRequest.setReturnUrl(AlipayConfig.returnUrl); alipayRequest.setNotifyUrl(AlipayConfig.notifyUrl); // 商户订单号,商户网站订单系统中唯一订单号,必填 String out_trade_no = oid; // 付款金额,必填 String total_amount = totalPrice; // 订单名称,必填 String subject = "支付宝沙箱测试商品支付"; // 商品描述,可空 String body = ""; // 该笔订单允许的最晚付款时间,逾期将关闭交易。取值范围:1m~15d。m-分钟,h-小时,d-天,1c-当天(1c-当天的情况下,无论交易何时创建,都在0点关闭)。 // 该参数数值不接受小数点, 如 1.5h,可转换为 90m。 String timeout_express = "1c"; //从session中取出异步需要用的uid Integer str = (Integer) request.getSession().getAttribute("uid"); String passback_params = String.valueOf(str); //对取得的数据进行URLEncoder编码,这一步不做会报错 String uidStr = URLEncoder.encode(passback_params,"UTF-8"); //将编码后的数据封装在alipayRequest对象中,参数名字一定要是passback_params才能在异步通知时返回 //至于参数的变量值这没有要求,只需要与上方编码后的一样即可 alipayRequest.setBizContent("{\"out_trade_no\":\"" + out_trade_no + "\"," + "\"total_amount\":\"" + total_amount + "\"," + "\"subject\":\"" + subject + "\"," + "\"body\":\"" + body + "\"," + "\"timeout_express\":\"" + timeout_express + "\"," + "\"passback_params\":\"" + uidStr + "\"," + "\"product_code\":\"FAST_INSTANT_TRADE_PAY\"}"); // 这个是请求支付宝后台获取的数据,实际上获取是一个表单,然后他会附带一个自动执行的js方法 //自动替你执行表单,然后进入表单中支付宝生成的页面 String result = alipayClient.pageExecute(alipayRequest).getBody(); log.info(result); return result; } /** * Description : 处理异步回调的方法 * @date 2022/7/25 * @param request 请求对象 * @param response 响应对象 * @return void **/ @PostMapping("/notifyNotice") @ResponseBody public void alipayNotifyNotice(HttpServletRequest request, HttpServletRequest response) throws Exception { log.info("支付成功, 进入异步通知接口..."); // 获取支付宝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.alipayPublicKey, AlipayConfig.charset, AlipayConfig.signType); // 调用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"); // 付款金额 String total_amount = new String(request.getParameter("total_amount").getBytes("ISO-8859-1"), "UTF-8"); //在支付接口中保存在session中uid String passback_params = new String(request.getParameter("passback_params").getBytes("ISO-8859-1"), "UTF-8"); //对传回的数据进行编码的逆过程,对参数进行解码 String uidStr = URLDecoder.decode(passback_params,"UTF-8"); if (trade_status.equals("TRADE_FINISHED")) { // 判断该笔订单是否在商户网站中已经做过处理 // 如果没有做过处理,根据订单号(out_trade_no)在商户网站的订单系统中查到该笔订单的详细,并执行商户的业务程序 // 如果有做过处理,不执行商户的业务程序 // 注意: 尚自习的订单没有退款功能, 这个条件判断是进不来的, 所以此处不必写代码 // 退款日期超过可退款期限后(如三个月可退款),支付宝系统发送该交易状态通知 } else if (trade_status.equals("TRADE_SUCCESS")) { // 判断该笔订单是否在商户网站中已经做过处理 // 如果没有做过处理,根据订单号(out_trade_no)在商户网站的订单系统中查到该笔订单的详细,并执行商户的业务程序 // 如果有做过处理,不执行商户的业务程序 // 注意: // 付款完成后,支付宝系统发送该交易状态通知 // 编写自己的订单支付成功的业务逻辑 //将上方取到的uid和将订单号转换为包装类 Integer uid = Integer.valueOf(uidStr); Integer oid = Integer.valueOf(out_trade_no); //调用订单业务层修改订单状态信息 orderService.updateOrderStatusByOid(oid,uid,1); log.info("********************** 支付成功(支付宝异步通知) **********************"); log.info("* 当前支付用户的id: {}", passback_params); log.info("* 订单号: {}", out_trade_no); log.info("* 支付宝交易号: {}", trade_no); log.info("* 实付金额: {}", total_amount); log.info("***************************************************************"); } log.info("支付成功..."); } else {// 验证失败 log.info("支付, 验签失败..."); } // return "success"; } /** * Description : 同步回调方法 * @date 2022/7/25 * @param request 请求对象 * @param response 响应对象 * @return void **/ @GetMapping("/returnNotice") public void alipayReturnNotice(HttpServletRequest request, HttpServletResponse response) throws Exception { log.info("支付成功, 进入同步通知接口..."); // 获取支付宝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.alipayPublicKey, AlipayConfig.charset, AlipayConfig.signType); // 调用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"); //同步调用方法中返回到指定的界面 //携带订单号并跳转到支付成功的界面 response.sendRedirect(request.getContextPath() + "/web/paySuccess.html?oid=" + out_trade_no); log.info("********************** 支付成功(支付宝同步通知) **********************"); log.info("* 订单号: {}", out_trade_no); log.info("* 支付宝交易号: {}", trade_no); log.info("* 实付金额: {}", total_amount); log.info("***************************************************************"); } else { log.info("支付, 验签失败..."); } } }
调用此接口时,会自动跳转到支付宝的支付页面。
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。