JavaWeb - 小米商城 :订单支付[沙箱支付]

1.功能描述

接上篇 JavaWeb - 小米商城:提交订单 订单创建后进入支付界面,整合支付宝沙箱支付。如 以下H5页面所示:


支付提示界面:

当点击提交订单, 此时将订单数据存储到数据表

java后端支付系统 javaweb支付项目实现_java后端支付系统

支付宝支付:

一般的开发中使用的是沙箱做模拟支付

java后端支付系统 javaweb支付项目实现_前端_02

支付过程提示:

java后端支付系统 javaweb支付项目实现_java后端支付系统_03

**使用沙箱app扫码支付成功后跳转到支付成功提示界面 **

java后端支付系统 javaweb支付项目实现_前端_04

我的订单列表:

支付成功可以通过点击我的订单查看订单列表

java后端支付系统 javaweb支付项目实现_前端_05

订单详情界面

通过点击订单列表中订单详情按钮打开该

java后端支付系统 javaweb支付项目实现_System_06

2.沙箱支付功能分析

2.1 在pom文件中添加依赖

<!--   支付宝沙箱支付的依赖     -->
    <dependency>
      <groupId>com.alipay.sdk</groupId>
      <artifactId>alipay-sdk-java</artifactId>
      <version>4.16.2.ALL</version>
    </dependency>

3.代码实现

3.1前端

3.1.1 修改链接地址
<html>
<head>
    <meta charset="utf-8">
    <meta http-equiv="X-UA-Compatible" content="IE=edge">
    <meta name="viewport" content="width=device-width, initial-scale=1">
    <title>订单成功</title>
    <link href="css/bootstrap.min.css" rel="stylesheet">
    <link rel="stylesheet" type="text/css" href="css/login2.css">
    <script type="text/javascript" src="js/jquery.min.js"></script>
    <script type="text/javascript" src="js/includeHeader.js"></script>
    <script>
        $(document).ready(function () {//当页面加载完成执行代码
            var oid = getParams("oid");
            var money = getParams("money");
            $("#oidh4").text("订单号:" + oid);
            $("#moneyh4").text("订单总金额:" + money);
            $("#payalink").attr("href", "pay.do?action=alipay&oid=" + oid + "&omoney=" + money);
            $("#Weixinlink").attr("href", "payWeixin.do?action=alipay&oid=" + oid + "&omoney=" + money);
        });
    </script>
</head>
<body>
<div id="headtop">
</div>

<div class="container">
    <hr>
    <div class="row" style="width: 30%;margin: 0 auto;padding-top: 20px">
        <div class="panel panel-success">
            <div class="panel-heading">
                <h3 class="panel-title">订单提示</h3>
            </div>
            <div class="panel-body">
                <h3 class="text-default"><span class="glyphicon glyphicon-ok-sign"></span>    订单添加成功!!
                </h3>
                <hr>
                <h4 id="oidh4"></h4>
                <h4 id="moneyh4"></h4>
                <span class="h3">立即支付</span>
                <a id="payalink" class="btn btn-warning">支付宝支付</a>    
                <a id="Weixinlink" class="btn btn-success">微信支付</a>
            </div>
        </div>

    </div>

</div>
<!--网站版权部分开始-->
//......
</body>
</html>

3.2后端

3.2.1 PayServlet

在 PayServlet类中首先设置如下私有属性

APP_ID (appId,从沙箱管理页面获得) APP_PRIVATE_KEY (应用私钥,最开始在密钥工具生成而 来) ALIPAY_PUBLIC_KEY (支付宝公钥,上一步骤获得) GATEWAY_URL (支付宝网关地址,在开发 平台沙箱管理页面中获得) SIGN_TYPE (签名类型) NOTIFY_URL (异步回调地址,须是公网IP,后 面再解释) RETURN_URL (同步回调地址,可以是私网IP)

@WebServlet("/pay.do")
public class PayServlet extends BaseServlet {
    //appid
    private final String APP_ID = "2021000119697420";
    //应用私钥
    private final String APP_PRIVATE_KEY = "MIIEvQIBADANBgkqhkiG9w0BAQEFAASCBKcwggSjAgEAAoIBAQ";
    private final String CHARSET = "UTF-8";
    // 支付宝公钥
    private final String ALIPAY_PUBLIC_KEY = "MIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEA6Gamayt9Z";
    //这是沙箱接口路径,正式路径为https://openapi.alipay.com/gateway.do
    private final String GATEWAY_URL = "https://openapi.alipaydev.com/gateway.do";

    private final String FORMAT = "JSON";
    //签名方式
    private final String SIGN_TYPE = "RSA2";
    //支付宝异步通知路径,付款完毕后会异步调用本项目的方法,必须为公网地址
    private final String NOTIFY_URL = "http://127.0.0.1/notifyUrl";
    //支付宝同步通知路径,也就是当付款完毕后跳转本项目的页面,可以不是公网地址
    private final String RETURN_URL = "http://localhost:8080/pay.do?action=returnUrl";

    /**
     * 支付方法
     */
    public String alipay(HttpServletRequest req, HttpServletResponse resp) {
        HttpSession session = req.getSession();
        String dona_id = req.getParameter("oid");
        String money = req.getParameter("omoney");
        Float dona_money = Float.parseFloat(money);
        //把dona_id项目id 放在session中
        session.setAttribute("dona_id", dona_id);

        //生成订单号(支付宝的要求?)
        String time = new SimpleDateFormat("yyyyMMddHHmmss").format(new Date());
        String user = UUID.randomUUID().toString().replace("-", "").toUpperCase();

        String OrderNum = time + user;

        //调用封装好的方法(给支付宝接口发送请求)
        try {
            return sendRequestToAlipay(OrderNum, dona_money, "小米商城");
        } catch (Exception e) {
            e.printStackTrace();
        }
        return Constants.REDIRECT + "/error.html";
    }

    /**
     * 参数1:订单号
     * 参数2:订单金额
     * 参数3:订单名称
     */
    //支付宝官方提供的接口
    private String sendRequestToAlipay(String outTradeNo, Float totalAmount, String subject) throws AlipayApiException {
        //获得初始化的AlipayClient
        AlipayClient alipayClient = new DefaultAlipayClient(GATEWAY_URL, APP_ID, APP_PRIVATE_KEY, FORMAT, CHARSET, ALIPAY_PUBLIC_KEY, SIGN_TYPE);

        //设置请求参数
        AlipayTradePagePayRequest alipayRequest = new AlipayTradePagePayRequest();
        alipayRequest.setReturnUrl(RETURN_URL);
        alipayRequest.setNotifyUrl(NOTIFY_URL);

        //商品描述(可空)
        String body = "小米商城模拟支付:" + subject;
        alipayRequest.setBizContent("{\"out_trade_no\":\"" + outTradeNo + "\","
                + "\"total_amount\":\"" + totalAmount + "\","
                + "\"subject\":\"" + subject + "\","
                + "\"body\":\"" + body + "\","
                + "\"product_code\":\"FAST_INSTANT_TRADE_PAY\"}");

        //请求
        String result = alipayClient.pageExecute(alipayRequest).getBody();
        return result;
    }

    /**
     * 支付宝的异步通知路径
     */
    public String returnUrl(HttpServletRequest req, HttpServletResponse resp) {
        //接收到付款成功的提示
        System.out.println("接收到付款成功的提示");
        //修改订单的状态 为 已支付

        System.out.println("=================================同步回调=====================================");
        try {

            // 获取支付宝GET过来反馈信息
            Map<String, String> params = new HashMap<String, String>();
            Map<String, String[]> requestParams = req.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);
            }

            System.out.println(params);//查看参数都有哪些
            //验证签名(支付宝公钥)
            boolean signVerified = AlipaySignature.rsaCheckV1(params, ALIPAY_PUBLIC_KEY, CHARSET, SIGN_TYPE); // 调用SDK验证签名
            //验证签名通过
            // 商户订单号
            String out_trade_no = new String(req.getParameter("out_trade_no").getBytes("ISO-8859-1"), "UTF-8");

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

            // 付款金额
            float money = Float.parseFloat(new String(req.getParameter("total_amount").getBytes("ISO-8859-1"), "UTF-8"));

            System.out.println("商户订单号=" + out_trade_no);
            System.out.println("支付宝交易号=" + trade_no);
            System.out.println("付款金额=" + money);

            //TODO 在这里编写自己的业务代码(对数据库进行操作)
            //在这里编写自己的业务代码(对数据库的操作)
            String oid = req.getSession().getAttribute("dona_id").toString();
            IOrderService orderService = new OrderServiceImpl();
            boolean flag = orderService.updateOrderState(OrderStatus.PAY.ordinal(),oid);
            if (flag){
                System.out.println("订单状态修改成功");
            }else {
                System.out.println("订单状态修改失败");
            }

            System.out.println("修改订单的状态 为 已支付");
            return Constants.REDIRECT + "/message.html?msg=支付成功!";
        } catch (Exception e) {
            e.printStackTrace();
            System.out.println("支付失败");
            return Constants.REDIRECT + "/error.html?msg=支付失败!";
        }
    }
}
3.2.2 编写OrderServiceImpl码
public class OrderServiceImpl implements IOrderService {
    private IOrderDao orderDao = null;
    private ICartDao cartDao = null;
    private IOrderDetailDao orderDetailDao = null;

    public OrderServiceImpl() {
        orderDao = new OrderDaoImpl();
        cartDao = new CartDaoImpl();
        orderDetailDao = new OrderDetailDaoImpl();
    }
    
    //..........

    @Override
    public boolean updateOrderState(int status,String oid) {
        return orderDao.updateOrderState(status,oid);
    }

    @Override
    public List<OrdersView> queryAllDataByUid(int uid) {
        return orderDao.queryAllDataByUid(uid);
    }
}
3.2.3 编写OrderDaoImpl代码
public class OrderDaoImpl implements IOrderDao {
    private QueryRunner queryRunner = null;

    public OrderDaoImpl() {
        queryRunner = new QueryRunner();
    }

   //......

    @Override
    public boolean updateOrderState(int status, String oid) {
        String sql = "update tb_order set status=? where id=?";
        try {
            int n = queryRunner.update(DBUtils.getConn(), sql, status, oid);
            return n >= 1 ? true : false;
        } catch (Exception e) {
            e.printStackTrace();
        }
        return false;
    }
    
    //......
    
}

4. 我的订单列表码实现

4.1 前端

4.1.1 修改orderList.html文件
<html>
<head>
    <meta http-equiv="Content-Type" content="text/html; charset=utf-8">
    <title>订单列表</title>
    <link href="css/bootstrap.min.css" rel="stylesheet">
    <script type="text/javascript" src="js/jquery.min.js"></script>
    <link rel="stylesheet" type="text/css" href="css/login2.css">
    <link href="css/bootstrap.min.css" rel="stylesheet">
    <script type="text/javascript">
        function showOrder(orderId) {
            location.href = "orderDetail.html?oid=" + orderId;
        }
        function changeStatus(orderId) {
            location.href = "order.do?action=changeStatus&oid=" + orderId;
        }

        $(document).ready(function () {
            //获取购物车的session数据显示到页面
            $.get("order.do?action=showMyOrder", "", function (result) {
                if (result.flag === true) {
                    var index = 0;
                    for (var i in result.data) {
                        index++;
                        var odhtml = '      <tr>' +
                            '<td>' + index + '</td>' +
                            '<td>' + result.data[i].oid + '</td>' +
                            '<td>' + result.data[i].totalMoney + '</td>' +
                            '<td>' +
                            '<font color="red">';
                        var statu = "";
                        if (result.data[i].ostate === 1) {
                            statu = "未支付";
                        }
                        if (result.data[i].ostate === 2) {
                            statu = "已支付,待发货";
                        }
                        if (result.data[i].ostate === 3) {
                            statu = "已发货,待收货";
                        }
                        if (result.data[i].ostate === 4) {
                            statu = "订单完成";
                        }
                        if (result.data[i].ostate === 5) {
                            statu = "未支付";
                        }
                        odhtml += statu + '</font>' +
                            ' </td>' +
                            ' <td>' + result.data[i].orderTime + '</td>' +
                            ' <td>' + result.data[i].address + '</td>' +
                            '<td>' +
                            '<button type="button" class="btn btn-danger btn-sm" οnclick="showOrder(\'' + result.data[i].oid + '\')">订单详情</button>';
                        var qrstr = "";
                        if (result.data[i].ostate === 3) {
                            qrstr = '<button type="button" class="btn btn-warning btn-sm" οnclick="changeStatus(\'' + result.data[i].oid + '\')">确认收货</button>';
                        }
                        odhtml += qrstr + '</td>' + '</tr>';
                        $("#orderBody").append(odhtml);
                        $("#utip").text( result.data[i].userName);
                    }
                }else {
                    var odhtml = '<tr><td colspan="7">当前用户没有订单数据</td></tr>'
                    $("#orderBody").append(odhtml);
                }
                $("#sizetip").append(index);
            });
        });
    </script>
    <script type="text/javascript" src="js/includeHeader.js"></script>
</head>
<body style="background-color:#f5f5f5">
<div id="headtop">
</div>
<div class="container" style="background-color: white;">
    <div class="row" style="margin-left: 40px">
        <h3>我的订单列表  
            <small >温馨提示: <em id="utip"></em>有<font color="red" id="sizetip"></font>个订单</small></h3>
    </div>
    <div class="row" style="margin-top: 40px;">
        <div class="col-md-12">
            <table id="tb_list" class="table table-striped table-hover table-bordered table-condensed">
                <thead>
                <tr>
                    <th>序号</th>
                    <th>订单编号</th>
                    <th>总金额</th>
                    <th>订单状态</th>
                    <th>订单时间</th>
                    <th>收货地址</th>
                    <th>操作</th>
                </tr>
                </thead>
                <tbody id="orderBody">
                </tbody>
            </table>
        </div>
    </div>
</div>
<!-- 底部 -->
<!--网站版权部分开始-->
//.....
</body>
</html>

4.2 后端

4.2.1 OrderServlet
@WebServlet("/order.do")
public class OrderServlet extends BaseServlet {

    private ResultData resultData = new ResultData();
		
    //....
    
    /**
     * 显示我的订单信息
     */
    public String showMyOrder(HttpServletRequest req, HttpServletResponse resp) {
        HttpSession session = req.getSession();
        System.out.println(session.getId());
        resultData.setFlag(true);
        User user = (User) session.getAttribute(Constants.LOGINUSER);
        System.err.println(user);
        if (user == null) {
            resultData.setFlag(false);
        } else {
            OrderServiceImpl orderService = new OrderServiceImpl();
            List<OrdersView> ordersViews = orderService.queryAllDataByUid(user.getId());
            if (ordersViews!=null && ordersViews.size()!=0){
                resultData.setData(ordersViews);
            }else {
                resultData.setFlag(false);
            }
        }
        String json = JSON.toJSONString(resultData);
        resp.setContentType("application/json;charset=utf-8");
        return json;
    }
}
4.2.2 OrderServiceImpl
public class OrderServiceImpl implements IOrderService {
    private IOrderDao orderDao = null;
    private ICartDao cartDao = null;
    private IOrderDetailDao orderDetailDao = null;

    public OrderServiceImpl() {
        orderDao = new OrderDaoImpl();
        cartDao = new CartDaoImpl();
        orderDetailDao = new OrderDetailDaoImpl();
    }

	//........ 
    
    @Override
    public List<OrdersView> queryAllDataByUid(int uid) {
        return orderDao.queryAllDataByUid(uid);
    }
}
4.2.3 OrderDaompl
public class OrderDaoImpl implements IOrderDao {
    private QueryRunner queryRunner = null;

    public OrderDaoImpl() {
        queryRunner = new QueryRunner();
    }

	//.......

    @Override
    public List<OrdersView> queryAllDataByUid(int uid) {
        String sql = "select o.id as oid,o.uid as uid,u.username as userName,o.time as orderTime," +
                " a.phone as telephone,a.detail as address,o.`status` as ostate,o.money as totalMoney " +
                " from tb_user u,tb_order o, tb_address a " +
                " where u.id=? and u.id=o.uid and o.aid=a.id";
        List<OrdersView> ordersViews = null;
        QueryRunner qrun = new QueryRunner(DBUtils.getDataSource());
        try {
            ordersViews = qrun.query(sql, uid, new BeanListHandler<>(OrdersView.class));
        } catch (SQLException e) {
            e.printStackTrace();
        }
        return ordersViews;
    }
}