Java调用微信支付_迷途知返-的博客-程序员宅基地_java微信支付

技术标签: java  微信  微信支付  

Java 使用微信支付

前言百度搜了一下微信支付,都描述的不太好,于是乎打算自己写一个案例,希望以后拿来直接改造使用。
因为涉及二维码的前端显示,所以有前端的内容

一. 准备工作

所需微信公众号信息配置

  • APPID:绑定支付的APPID(必须配置)
  • MCHID:商户号(必须配置)
  • KEY:商户支付密钥,参考开户邮件设置(必须配置)
  • APPSECRET:公众帐号secert(仅JSAPI支付的时候需要配置)

我这个案例用的是尚硅谷一位老师提供的,这里不方便提供出来,需要大家自己找,或者公司提供

二. 构建项目架构

1.新建maven项目

在这里插入图片描述

2.导入依赖

    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>2.2.1.RELEASE</version>
    </parent>
    <dependencies>
        <!--spring boot -->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>
        <!--微信提供的sdk-->
        <dependency>
            <groupId>com.github.wxpay</groupId>
            <artifactId>wxpay-sdk</artifactId>
            <version>0.0.3</version>
        </dependency>
        <!--发送http请求-->
        <dependency>
            <groupId>org.apache.httpcomponents</groupId>
            <artifactId>httpclient</artifactId>
        </dependency>
        <!--模板引擎-->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-freemarker</artifactId>
        </dependency>
    </dependencies>

在这里插入图片描述
依赖中需要注意的是我导入了微信提供的sdk,以及freemarker模板引擎

3.编写配置文件application.properties

# 服务端口
server.port=8081
# 微信开放平台 appid
wx.pay.app_id=
#商户号
wx.pay.partner=
#商户key
wx.pay.partnerkey=
#回调地址
wx.pay.notifyurl:

spring.freemarker.tempalte-loader-path=classpath:/templates
#        关闭缓存,及时刷新,上线生产环境需要修改为true
spring.freemarker.cache=false
spring.freemarker.charset=UTF-8
spring.freemarker.check-template-location=true
spring.freemarker.content-type=text/html
spring.freemarker.expose-request-attributes=true
spring.freemarker.expose-session-attributes=true
spring.freemarker.request-context-attribute=request
spring.freemarker.suffix=.ftl
spring.mvc.static-path-pattern: /static/**

在这里插入图片描述

4.编写启动类

@SpringBootApplication
@ComponentScan(basePackages = {"com.haiyang.wxpay"})
public class Application {

    public static void main(String[] args) {
        SpringApplication.run(Application.class, args);
    }
}

在这里插入图片描述

5.创建常用包controller,service,impl,utils

在这里插入图片描述

6.创建两个前端需要的文件夹 static和templates

在这里插入图片描述

三. 代码实现

1. 创建工具类读取配置文件的参数

@Component
public class WxPayUtils implements InitializingBean {

    @Value("${wx.pay.app_id}")
    private String appId;

    @Value("${wx.pay.partner}")
    private String partner;

    @Value("${wx.pay.partnerkey}")
    private String partnerKey;
    @Value("${wx.pay.notifyurl}")
    private String notifyUrl;


    public static String WX_PAY_APP_ID;
    public static String WX_PAY_PARTNER;
    public static String WX_PAY_PARTNER_KEY;
    public static String WX_OPEN_NOTIFY_URL;

    @Override
    public void afterPropertiesSet() throws Exception {
        WX_PAY_APP_ID = appId;
        WX_PAY_PARTNER = partner;
        WX_PAY_PARTNER_KEY = partnerKey;
        WX_OPEN_NOTIFY_URL = notifyUrl;
    }

}

在这里插入图片描述

2. 构建工具类发送http请求

/**
 * http请求客户端
 * 
 * @author qy
 * 
 */
public class HttpClient {
	private String url;
	private Map<String, String> param;
	private int statusCode;
	private String content;
	private String xmlParam;
	private boolean isHttps;

	public boolean isHttps() {
		return isHttps;
	}

	public void setHttps(boolean isHttps) {
		this.isHttps = isHttps;
	}

	public String getXmlParam() {
		return xmlParam;
	}

	public void setXmlParam(String xmlParam) {
		this.xmlParam = xmlParam;
	}

	public HttpClient(String url, Map<String, String> param) {
		this.url = url;
		this.param = param;
	}

	public HttpClient(String url) {
		this.url = url;
	}

	public void setParameter(Map<String, String> map) {
		param = map;
	}

	public void addParameter(String key, String value) {
		if (param == null)
			param = new HashMap<String, String>();
		param.put(key, value);
	}

	public void post() throws ClientProtocolException, IOException {
		HttpPost http = new HttpPost(url);
		setEntity(http);
		execute(http);
	}

	public void put() throws ClientProtocolException, IOException {
		HttpPut http = new HttpPut(url);
		setEntity(http);
		execute(http);
	}

	public void get() throws ClientProtocolException, IOException {
		if (param != null) {
			StringBuilder url = new StringBuilder(this.url);
			boolean isFirst = true;
			for (String key : param.keySet()) {
				if (isFirst)
					url.append("?");
				else
					url.append("&");
				url.append(key).append("=").append(param.get(key));
			}
			this.url = url.toString();
		}
		HttpGet http = new HttpGet(url);
		execute(http);
	}

	/**
	 * set http post,put param
	 */
	private void setEntity(HttpEntityEnclosingRequestBase http) {
		if (param != null) {
			List<NameValuePair> nvps = new LinkedList<NameValuePair>();
			for (String key : param.keySet())
				nvps.add(new BasicNameValuePair(key, param.get(key))); // 参数
			http.setEntity(new UrlEncodedFormEntity(nvps, Consts.UTF_8)); // 设置参数
		}
		if (xmlParam != null) {
			http.setEntity(new StringEntity(xmlParam, Consts.UTF_8));
		}
	}

	private void execute(HttpUriRequest http) throws ClientProtocolException,
			IOException {
		CloseableHttpClient httpClient = null;
		try {
			if (isHttps) {
				SSLContext sslContext = new SSLContextBuilder()
						.loadTrustMaterial(null, new TrustStrategy() {
							// 信任所有
							public boolean isTrusted(X509Certificate[] chain,
									String authType)
									throws CertificateException {
								return true;
							}
						}).build();
				SSLConnectionSocketFactory sslsf = new SSLConnectionSocketFactory(
						sslContext);
				httpClient = HttpClients.custom().setSSLSocketFactory(sslsf)
						.build();
			} else {
				httpClient = HttpClients.createDefault();
			}
			CloseableHttpResponse response = httpClient.execute(http);
			try {
				if (response != null) {
					if (response.getStatusLine() != null)
						statusCode = response.getStatusLine().getStatusCode();
					HttpEntity entity = response.getEntity();
					// 响应内容
					content = EntityUtils.toString(entity, Consts.UTF_8);
				}
			} finally {
				response.close();
			}
		} catch (Exception e) {
			e.printStackTrace();
		} finally {
			httpClient.close();
		}
	}

	public int getStatusCode() {
		return statusCode;
	}

	public String getContent() throws ParseException, IOException {
		return content;
	}

}

额~有点长就不放图片了 代码都一样

3. 新建controller

@Controller
@RequestMapping("/wxpay")
public class WxPayController {
    @RequestMapping("/pay")
    public String createPayQRcode(Model model) throws Exception{

        String price = "0.01";
        String no = getOrderNo();
        Map m = new HashMap();
        m.put("appid", WxPayUtils.WX_PAY_APP_ID);
        m.put("mch_id", WxPayUtils.WX_PAY_PARTNER);
        m.put("nonce_str", WXPayUtil.generateNonceStr());
        m.put("body","微信支付测试"); //主体信息
        m.put("out_trade_no", no); //订单唯一标识
        m.put("total_fee", getMoney(price));//金额
        m.put("spbill_create_ip", "127.0.0.1");//项目的域名
        m.put("notify_url", WxPayUtils.WX_OPEN_NOTIFY_URL);//回调地址
        m.put("trade_type", "NATIVE");//生成二维码的类型

        //3 发送httpclient请求,传递参数xml格式,微信支付提供的固定的地址
        HttpClient client = new HttpClient("https://api.mch.weixin.qq.com/pay/unifiedorder");
        //设置xml格式的参数
        //把xml格式的数据加密
        client.setXmlParam(WXPayUtil.generateSignedXml(m, WxPayUtils.WX_PAY_PARTNER_KEY));
        client.setHttps(true);
        //执行post请求发送
        client.post();
        //4 得到发送请求返回结果
        //返回内容,是使用xml格式返回
        String xml = client.getContent();
        //把xml格式转换map集合,把map集合返回
        Map<String,String> resultMap = WXPayUtil.xmlToMap(xml);
        //最终返回数据 的封装
        Map map = new HashMap();
        map.put("no", no);
        map.put("price", price);
        map.put("result_code", resultMap.get("result_code"));
        map.put("code_url", resultMap.get("code_url"));

        model.addAttribute("map",map);
        return "pay";

    }

    @GetMapping("queryorder/{no}")
    @ResponseBody
    public String queryPayStatus(@PathVariable String no) throws Exception{
        //1、封装参数
        Map m = new HashMap<>();
        m.put("appid", WxPayUtils.WX_PAY_APP_ID);
        m.put("mch_id", WxPayUtils.WX_PAY_PARTNER);
        m.put("out_trade_no", no);
        m.put("nonce_str", WXPayUtil.generateNonceStr());

        //2 发送httpclient
        HttpClient client = new HttpClient("https://api.mch.weixin.qq.com/pay/orderquery");
        client.setXmlParam(WXPayUtil.generateSignedXml(m, WxPayUtils.WX_PAY_PARTNER_KEY));
        client.setHttps(true);
        client.post();

        //3.得到订单数据
        String xml = client.getContent();
        Map<String, String> resultMap = WXPayUtil.xmlToMap(xml);

        //4.判断是否支付成功
        if(resultMap.get("trade_state").equals("SUCCESS")) {
            /*
                  改变数据库中的数据等操作
             */
            return "支付成功";
        }
        return "支付中";
    }

    @GetMapping("success")
    public String success(){
        return "success";
    }
    @RequestMapping("test")
    public String test(){
        return "pay";
    }
    /**
     * 生成订单号
     * @return
     */
    public static String getOrderNo() {
        SimpleDateFormat sdf = new SimpleDateFormat("yyyyMMddHHmmss");
        String newDate = sdf.format(new Date());
        String result = "";
        Random random = new Random();
        for (int i = 0; i < 3; i++) {
            result += random.nextInt(10);
        }
        return newDate + result;
    }
    /**
     * 元转换成分
     * @param amount
     * @return
     */
    public static String getMoney(String amount) {
        if(amount==null){
            return "";
        }
        // 金额转化为分为单位
        // 处理包含, ¥ 或者$的金额
        String currency =  amount.replaceAll("\\$|\\¥|\\,", "");
        int index = currency.indexOf(".");
        int length = currency.length();
        Long amLong = 0l;
        if(index == -1){
            amLong = Long.valueOf(currency+"00");
        }else if(length - index >= 3){
            amLong = Long.valueOf((currency.substring(0, index+3)).replace(".", ""));
        }else if(length - index == 2){
            amLong = Long.valueOf((currency.substring(0, index+2)).replace(".", "")+0);
        }else{
            amLong = Long.valueOf((currency.substring(0, index+1)).replace(".", "")+"00");
        }
        return amLong.toString();
    }
}

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
值得一提的是 这里我们用的是controller而不是restcontroller,因为我们需要展示二维码

4. 在templates文件中新建 订单支付页面(二维码生成的页面)

注意:文件名必须和生成二维码方法中返回的字符串名称一样 我这里叫 pay

先新建html页面,然后再将后缀改成ftl(freemarker模板引擎的后缀名)

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    <script src="/static/qrcode.js"></script>
    <script src="https://cdn.bootcss.com/jquery/3.4.1/jquery.js"></script>
</head>
<center>
    <div id="qrcode"></div>
</center>
<script type="text/javascript">
    new QRCode(document.getElementById("qrcode"), "${map.code_url}");  // 设置要生成二维码的链接
</script>
<script type="text/javascript">
    var int=self.setInterval("querystatus()",3000);
    function querystatus() {
        $.get("/wxpay/queryorder/${map.no}",function(data,status){
            if (data==="支付中"){
                console.log("支付中");
            } else {
                clearInterval(int)
                window.location.href="/wxpay/success"
            }
        })
    }
</script>
</body>
</html>

在这里插入图片描述
再创建支付成功跳转的页面 文件名要与支付成功方法返回的文件名一样

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
</head>
<body>
<h1>支付成功</h1>
</body>
</html>

在这里插入图片描述
引入 qrcode 生成二维码的依赖,放入static文件中
这里我提供下载链接
链接: https://pan.baidu.com/s/15-E3KpRCenAewh0ZaBLnjQ 提取码: xhs9 复制这段内容后打开百度网盘手机App,操作更方便哦

引入完成后
在这里插入图片描述

最后 我们启动项目来测试一下

浏览器输入地址
http://localhost:8081/wxpay/pay
发现二维码生成成功,并且定时器也没问题
在这里插入图片描述
之后我们扫码支付
成功跳转到支付成功页面 ~nice
在这里插入图片描述

四. 总结

  1. 首先就是生成二维码,需要的几个主要的参数,订单号,金额,购买的信息(主体信息),其余的参数除了一些可以不写的都是固定的
  2. 生成二维码然后展示在页面上,用的qrcode插件,生成
  3. 然后设置定时器,来实时查询订单是否支付
  4. 查询订单信息的写法和生成二维码的方式差不多 无非就是请求时少了几个参数,必须得带上订单号
  5. 微信提供的查询订单接口返回数据中 trade_state 代表支付状态 notpay没有支付,seccess表示已成功
  6. 定时器检测到订单支付成功就清除定时器,并且执行支付成功之后的操作

实际项目中远没有这么简单,并且所有的数据都要从数据库中获取,在这里我为了方便把价格固定写死的

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/haiyanghan/article/details/106792920

智能推荐

小程序的图片预加载_aaron9185的博客-程序员宅基地_小程序图片预加载

通过搜索资料找出三种方式一:加载多张图片    可以先加载一张图片并监听图片控件的bindload(图片加载完成后触发),当加载完一张图片后再将其余图片进行显示。二:将较大的图片  先进行加载缩略图  后加载整张图片。三:采用wx:if对图片进行加载但是不显示,并对每张图片进行监听直至加载图片完成,再通过wx:if进行显示github地址:https://github.com/o2team/wxa...

在Linux系统下通过TFTP或NFS烧写内核_multimicro的博客-程序员宅基地

注:我之前在cnblog上发布的这篇文章,现在把这篇教程发布到CSDN上面一直想直接通过Linux系统往JZ2440开发板中烧写内核,但网上的教程千篇一律都是借助Windows平台上的TFTP工具烧写的,十分不爽,因为我不喜欢在虚拟机上玩Linux!摸索了一下午,终于搞定了,下来记录一下烧录过程。本文主要讲诉TFTP烧写内核的过程,NFS的先挖个坑。开发环境Linux 16.04 L...

FSM----C语言_Rookie on the road的博客-程序员宅基地

有限状态机(finite state machine)简称FSM,表示有限个状态及在这些状态之间的转移和动作等行为的数学模型,在计算机领域有着广泛的应用。FSM是一种逻辑单元内部的一种高效编程方法。状态机实现的方式有多种,下面讲述三种.1.使用if/else if语句实现的FSM使用if/else if语句是实现的FSM最简单最易懂的方法,我们只需要通过大量的if /else if语句来判断...

windows、ubuntu双系统下扩容挂载点,完整教程_丶ershiyi的博客-程序员宅基地

    由于笔者刚开始安装ubuntu,给每个挂载点的大小不太合理,很快就提示内存不足。所以查看了网上的很多教程,却不太完整,所以在这里给大家整理下。    笔者原来给挂载点的大小是: 一共30G,  /  6G  ,  /boot  200M  ,  /swap    4G  ,  /home  剩下的所有空间。 为了配置开发环境,结果很快就提示 / (根目录)的内存大小不足。    扩容挂载点...

使用vue,获取input输入值(不使用双向数据绑定),参数e的问题_Gabriel_wei的博客-程序员宅基地

&lt;input class="ItemInput" type="number" @input="firstInput"&gt;firstInput: function(e) { console.log(e) console.log(e.target) console.log(e.target.value) console.log(e.currentTarget);}...

JavaScript 实用工具函数_|恰同学少年|的博客-程序员宅基地

1. 数字操作(1)数字操作export const randomNum = (min, max) =&gt; Math.floor(Math.random() * (max - min + 1)) + min;(2)数字千分位分隔export const format = (n) =&gt; { let num = n.toString(); let len = num.length; if (len &lt;= 3) { return num;

随便推点

express跳转,重定向(及传参)的实现_hh3167253066的博客-程序员宅基地_express模块中哪个方法可以实现响应的重定向

res.location()和res.redirect(),使用它们可以实现URL的301或302重定向。locationres.location('/foo/bar');res.location('http://example.com');res.location('back');路径值back涉及到请求头Referer中指定的URL,如果Referer头没有指定,将会设置为’/’。Express通过Location头将指定的URL字符串传递给浏览器,它并不会对指定的字符串进行验证(除’ba

node操作excel5 node-xlsx设置字体、文字颜色、下划线、斜体_独行侠_阿涛的博客-程序员宅基地

背景《node操作excel》系列里头第二篇《node操作excel2 利用node_xlsx设置单元格边宽》,我们已经讲解了如何设置单元格的边框,过程我们已经对node-xlsx进行了修改。后续所有的样式设置都依赖已经修改过的node-xlsx来进行操作的!设置字体、文字颜色、下划线、斜体const xlsx = require('node-xlsx');const fs = require('fs')let s = { font: { name: '隶书',

《图形编程技术学习》(一)计算机图形学与图形流水线_小水VV的博客-程序员宅基地_图形编程技术学习

这个系列是学习北京林业大学林刚教授的课程时的学习笔记,课程简洁易懂又含以重要知识,谨以此分享出来。欢迎勘误~一.计算机图形学的概念及主要研究内容计算机图形学是作什么的计算机图形学(Computer Graphics,简称CG)1.计算机图形学是一门研究如何利用计算机进行的计算、处理和显示的学科。2.简单地说:是一种使用数学算法将二维或三维图形转化为计算机显示器所能显示的二维栅...

Leetcode——合并K个升序链表问题_Purple.''的博客-程序员宅基地

题目:给你一个链表数组,每个链表都已经按升序排列。请你将所有链表合并到一个升序链表中,返回合并后的链表。示例 1:输入:lists = [[1,4,5],[1,3,4],[2,6]]输出:[1,1,2,3,4,4,5,6]解释:链表数组如下:[1-&gt;4-&gt;5,1-&gt;3-&gt;4,2-&gt;6]将它们合并到一个有序链表中得到。1-&gt;1-&gt;2-&gt;3-&gt;4-&gt;4-&gt;5-&gt;6示例 2:输入:lists = []

微信小程序用<cover-view>自定义tooltip样式覆盖canvas 且带有点击过几秒tooltip消失功能_赫塔的博客-程序员宅基地_微信小程序tooltip

wxml&lt;cover-view wx:if="{{isShowImg}}" class='rich-textq' style='top:{{topL}}px;left:{{leftL}}px;background-color :rgba(50, 50, 50, 0.7);padding:10px;font-size:12px;color:#fff;'&gt; &lt;cover-view wx:for='{{itemList}}' style="line-height: 20px;" wx:key

iWebshop二次开发2_码农老黄牛的博客-程序员宅基地

搭建iwebshop服务器,常见的也就那么几个云服务商,阿里云,腾讯云。。。,这里选用阿里云,按需购买服务器的空间,还有带宽,如果是初期流量不大的,可以用按流量付费,会比较省钱。一般的一个服务器空间,一年差不多1000多起步差不多。域名几十块钱一年,首选.com的,其次,.cn的,备案这块按要求提交资料即可免费的。iwebshop使用的php+mysql,系统这块,可以选用ubuntu 18,默认的未安装图形化界面,可以安装上图形化界面,操作更方便。搭建服务器环境,可以用xampp,安装后搭建比较容

推荐文章

热门文章

相关标签