Java之微信支付(扫码支付模式二)案例实战-之支付完成后页面跳转方案

本文主要是介绍Java之微信支付(扫码支付模式二)案例实战-之支付完成后页面跳转方案,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

摘要:做过微信扫码支付的同学都知道,微信扫码支付完成后,要跳转到指定的页面就比较麻烦了,这里我提供一种比较可行的方案,也许还有其他更多的方案,但是我这里分享下我们当时是怎么处理微信扫码支付完成后,跳转到指定页面的呢。

一:微信扫码支付流程:

       可以看前面写过的文章,具体地址:Java之微信支付(扫码支付模式二)案例实战 

二:微信扫码支付完成后页面跳转流程:

2.1.支付订单页面:

2.2.返回微信支付二维码页面:

2.3.扫码支付步骤一:

2.4.扫码支付步骤二:

2.5.扫码支付步骤三:

2.6.支付完成,跳转购买成功页面:

三:支付完成,跳转支付成功页面:

这里最关键的是就是调用微信的扫码支付接口后,接收到微信返回的二维码连接后,要同时输出一个页面,并且页面里面要写成一个模板,里面需要定时调用查询订单支付状态的接口,目前是每5s调用一次订单支付状态接口,发现支付状态返回成功,那么就跳转到页面里面配置的URL,也可以是动态URL,具体关键代码就是支付完成后,返回页面的代码,如下:

3.1.支付接口控制类:

@PostMapping("/create")public String h5Pay(HttpServletRequest request,HttpServletResponse response,ModelMap maps){String  payType = request.getParameter("payType");String  productCode = request.getParameter("productCode");Map<String, Object> map =new HashMap<String, Object>(); //参数校验R checkParams = PayServiceHelper.checkParams(request);if((Integer)checkParams.get("code")==500) {maps.addAttribute("error",checkParams.get("msg"));return "error";}//订单签名校验boolean checkOrder = paymentInfoService.checkOrder(request);if(!checkOrder) {maps.addAttribute("error","验证签名失败");return "error";}//request 参数封装Mapmap = PayServiceHelper.getPayMap(request);//订单校验是否存在,是否已支付PaymentInfoEntity queryOrder = paymentInfoService.queryStatus((String)map.get("orderId"));if(queryOrder!=null) {maps.addAttribute("error","订单已支付");return "error";}//增加交易记录String order = paymentInfoService.insert(map);map.put("order", order);//发起支付PayService payService = payServiceFactory.getPayService(payType, productCode);// 微信扫码支付,需要传客户端IP地址if (Constant.PayType.WXCODEPAY.getName().equals(payType) && Constant.ProductCode.PC.getName().equals(productCode)) {String ipAddr = IPUtils.getIpAddr(request);map.put("ipAddr", ipAddr);String orderId = request.getParameter("orderId");map.put("externalId", orderId);map.put("url", url);}payService.h5Pay(map,request, response);return "success";}

3.2.支付接口实现类:

@Overridepublic void h5Pay(Map<String, Object> map, HttpServletResponse response) throws Exception {BigDecimal bigDecimal = new BigDecimal(100);String externalId = String.valueOf(map.get("externalId"));String url = String.valueOf(map.get("url"));String returnUrl = String.valueOf(map.get("returnUrl"));String order = (String)map.get("order");String ipAddr = (String) map.get("ipAddr");PaymentInfoEntity paymentInfoEntity = paymentInfoService.queryOrder(map);if(null == paymentInfoEntity) {return;}// 订单实际支付金额BigDecimal amount = paymentInfoEntity.getAmount();WxPayUnifiedOrderRequest request = WxPayUnifiedOrderRequest.newBuilder().body("药品订单").totalFee(amount.multiply(bigDecimal).setScale(0,BigDecimal.ROUND_DOWN).intValue()).outTradeNo(order).spbillCreateIp(ipAddr).notifyUrl(wechatAccountConfig.getNotifyUrl()).tradeType(WxPayConstants.TradeType.NATIVE).productId("ydw").build();//1. 统一下单WxPayNativeOrderResult result = null;try {result = wxPayService.createOrder(request);String codeUrl = result.getCodeUrl();//增加网关支付请求记录payGatewayService.insert(order, url);//生成二维码的图片格式int width = 300;int height = 300;Hashtable hints = new Hashtable();// 内容所使用编码hints.put(EncodeHintType.CHARACTER_SET, "utf-8");BitMatrix bitMatrix = new MultiFormatWriter().encode(codeUrl, BarcodeFormat.QR_CODE, width, height, hints);BufferedImage bufferedImage = QRUtil.toBufferedImage(bitMatrix);byte[] data = imageToBytes(bufferedImage, "png");BASE64Encoder encoder = new BASE64Encoder();//转换成base64串String png_base64 =  encoder.encodeBuffer(data).trim();//删除 \r\npng_base64 = png_base64.replaceAll("\n", "").replaceAll("\r", "");// 输出二维码的Base64编码到页面response.setContentType("text/html;charset=" + "UTF-8");response.getWriter().write(PayServiceHelper.builderWxCodeForm(url, externalId, "data:image/png;base64, " + png_base64));response.getWriter().flush();response.getWriter().close();} catch (WxPayException e) {e.printStackTrace();} catch (WriterException e) {e.printStackTrace();} catch (IOException e) {e.printStackTrace();}}

3.3.支付接口输出页面的工具类:

public static String builderWxCodeForm(String url, String orderId, String imgUrl) {StringBuffer sBuffer= new StringBuffer();sBuffer.append("<!DOCTYPE html>\n" +"<html>\n" +"  <head> \n" +"    <meta charset=\"utf-8\">\n" +"    <meta http-equiv=\"X-UA-Compatible\" content=\"IE=edge\">\n" +"    <meta name=\"viewport\" content=\"width=750,user-scalable=no\">\n" +"    <link rel=\"icon\" href=\"static/ydw.ico\">\n" +"    <title>药兜网</title>\n" +"    <script>\n" +"      var lock=window.setInterval(function(){\n" +"        var ajax = new XMLHttpRequest();\n" +"        ajax.open('get','query?orderId='+"+orderId+");\n" +"        ajax.send();\n" +"        ajax.onreadystatechange = function () {\n" +"        console.log(ajax.status);\n" +"        if (ajax.status==200) {\n" +"            var resText = ajax.responseText;\n" +"            var data = eval('(' + resText + ')');\n" +"        		if (data['status']==1) {\n" +"               	window.location.href = \""+url+"/purchaseSuccess?orderId="+orderId+"\"\n" +"               	window.clearInterval(lock);\n" +"          	}\n" +"          }\n" +"        }\n" +"      },2000)\n" +"     \n" +"    </script>\n" +"     <style>\n" +"      #app {\n" +"          width: 100%;\n" +"          height: 100%;\n" +"          position: fixed;\n" +"          top: 0;\n" +"          left: 0;\n" +"          display: flex;\n" +"          flex-direction: row;\n" +"          justify-content: center;\n" +"          align-items: center;\n" +"      }\n" +"      #app .img {\n" +"          vertical-align: middle;\n" +"      }\n" +"\n" +"  </style>\n" +"\n" +"  </head>\n" +"  <body style=\"width:750px;margin:0 auto;background: black;\">\n" +"    <div id=\"app\" >\n" +"     <img src=\""+imgUrl+"\"/>\n" +"    </div>\n" +"  </body>\n" +"</html>");return sBuffer.toString();}

最关键是就是最后一个方法,这里就是输出一个页面,里面有个定时器定时调用订单支付状态接口,如果返回支付成功,那么就跳转的购买成功页面,以上就是关于微信扫码支付完成后,跳转购买成功页面的解决方案,欢迎大家一起讨论,如果大家有更好的方案,欢迎留言,谢谢!

如有问题可以加群讨论解决:QQ群好:278298761

这篇关于Java之微信支付(扫码支付模式二)案例实战-之支付完成后页面跳转方案的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/767687

相关文章

网页解析 lxml 库--实战

lxml库使用流程 lxml 是 Python 的第三方解析库,完全使用 Python 语言编写,它对 XPath表达式提供了良好的支 持,因此能够了高效地解析 HTML/XML 文档。本节讲解如何通过 lxml 库解析 HTML 文档。 pip install lxml lxm| 库提供了一个 etree 模块,该模块专门用来解析 HTML/XML 文档,下面来介绍一下 lxml 库

JVM 的类初始化机制

前言 当你在 Java 程序中new对象时,有没有考虑过 JVM 是如何把静态的字节码(byte code)转化为运行时对象的呢,这个问题看似简单,但清楚的同学相信也不会太多,这篇文章首先介绍 JVM 类初始化的机制,然后给出几个易出错的实例来分析,帮助大家更好理解这个知识点。 JVM 将字节码转化为运行时对象分为三个阶段,分别是:loading 、Linking、initialization

Spring Security 基于表达式的权限控制

前言 spring security 3.0已经可以使用spring el表达式来控制授权,允许在表达式中使用复杂的布尔逻辑来控制访问的权限。 常见的表达式 Spring Security可用表达式对象的基类是SecurityExpressionRoot。 表达式描述hasRole([role])用户拥有制定的角色时返回true (Spring security默认会带有ROLE_前缀),去

浅析Spring Security认证过程

类图 为了方便理解Spring Security认证流程,特意画了如下的类图,包含相关的核心认证类 概述 核心验证器 AuthenticationManager 该对象提供了认证方法的入口,接收一个Authentiaton对象作为参数; public interface AuthenticationManager {Authentication authenticate(Authenti

Spring Security--Architecture Overview

1 核心组件 这一节主要介绍一些在Spring Security中常见且核心的Java类,它们之间的依赖,构建起了整个框架。想要理解整个架构,最起码得对这些类眼熟。 1.1 SecurityContextHolder SecurityContextHolder用于存储安全上下文(security context)的信息。当前操作的用户是谁,该用户是否已经被认证,他拥有哪些角色权限…这些都被保

Spring Security基于数据库验证流程详解

Spring Security 校验流程图 相关解释说明(认真看哦) AbstractAuthenticationProcessingFilter 抽象类 /*** 调用 #requiresAuthentication(HttpServletRequest, HttpServletResponse) 决定是否需要进行验证操作。* 如果需要验证,则会调用 #attemptAuthentica

Spring Security 从入门到进阶系列教程

Spring Security 入门系列 《保护 Web 应用的安全》 《Spring-Security-入门(一):登录与退出》 《Spring-Security-入门(二):基于数据库验证》 《Spring-Security-入门(三):密码加密》 《Spring-Security-入门(四):自定义-Filter》 《Spring-Security-入门(五):在 Sprin

Java架构师知识体认识

源码分析 常用设计模式 Proxy代理模式Factory工厂模式Singleton单例模式Delegate委派模式Strategy策略模式Prototype原型模式Template模板模式 Spring5 beans 接口实例化代理Bean操作 Context Ioc容器设计原理及高级特性Aop设计原理Factorybean与Beanfactory Transaction 声明式事物

无人叉车3d激光slam多房间建图定位异常处理方案-墙体画线地图切分方案

墙体画线地图切分方案 针对问题:墙体两侧特征混淆误匹配,导致建图和定位偏差,表现为过门跳变、外月台走歪等 ·解决思路:预期的根治方案IGICP需要较长时间完成上线,先使用切分地图的工程化方案,即墙体两侧切分为不同地图,在某一侧只使用该侧地图进行定位 方案思路 切分原理:切分地图基于关键帧位置,而非点云。 理论基础:光照是直线的,一帧点云必定只能照射到墙的一侧,无法同时照到两侧实践考虑:关

Hadoop企业开发案例调优场景

需求 (1)需求:从1G数据中,统计每个单词出现次数。服务器3台,每台配置4G内存,4核CPU,4线程。 (2)需求分析: 1G / 128m = 8个MapTask;1个ReduceTask;1个mrAppMaster 平均每个节点运行10个 / 3台 ≈ 3个任务(4    3    3) HDFS参数调优 (1)修改:hadoop-env.sh export HDFS_NAMENOD