基于 Spring Boot 框架实现微信支付接口调用及回调功能

实现微信支付接口调用及回调功能,以下是完整的步骤及代码实现,基于 Spring Boot 框架。

1. 微信支付开发准备

开通微信支付

  1. 确保已在 微信商户平台上注册并完成企业认证。
  2. 获取以下重要信息:
    • 商户号(mch_id
    • 商户密钥(key
    • 微信支付分配的公众账号 ID(appid
  3. 下载商户证书(如需处理退款)。

配置 API 安全密钥

前往商户平台的【账户设置】-【API安全】中配置 API 密钥。

2. 集成依赖

在 Spring Boot 项目中添加 HTTP 客户端依赖,例如 RestTemplateOkHttp。也可用支付宝/微信专用 SDK。

pom.xml 中添加以下依赖:

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
    <groupId>com.squareup.okhttp3</groupId>
    <artifactId>okhttp</artifactId>
    <version>4.11.0</version>
</dependency>
<dependency>
    <groupId>com.fasterxml.jackson.core</groupId>
    <artifactId>jackson-databind</artifactId>
</dependency>

3. 微信支付统一下单接口调用

核心代码实现

import com.fasterxml.jackson.databind.ObjectMapper;
import okhttp3.*;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Service;

import java.security.MessageDigest;
import java.util.Map;
import java.util.SortedMap;
import java.util.TreeMap;
import java.util.UUID;

@Service
public class WeChatPayService {

    @Value("${wechat.pay.appid}")
    private String appId;

    @Value("${wechat.pay.mch_id}")
    private String mchId;

    @Value("${wechat.pay.key}")
    private String apiKey;

    @Value("${wechat.pay.notify_url}")
    private String notifyUrl;

    private static final String UNIFIED_ORDER_URL = "https://api.mch.weixin.qq.com/pay/unifiedorder";

    public String createOrder(String orderNo, String description, int totalFee, String clientIp) throws Exception {
        SortedMap<String, String> params = new TreeMap<>();
        params.put("appid", appId);
        params.put("mch_id", mchId);
        params.put("nonce_str", UUID.randomUUID().toString().replaceAll("-", ""));
        params.put("body", description);
        params.put("out_trade_no", orderNo);
        params.put("total_fee", String.valueOf(totalFee)); // 金额,单位分
        params.put("spbill_create_ip", clientIp); // 调用支付的设备 IP
        params.put("notify_url", notifyUrl); // 异步回调地址
        params.put("trade_type", "NATIVE"); // 交易类型,NATIVE 表示扫码支付

        // 签名
        String sign = generateSign(params);
        params.put("sign", sign);

        // 构造 XML 请求
        String requestXml = mapToXml(params);

        // 使用 OkHttp 发起请求
        OkHttpClient client = new OkHttpClient();
        RequestBody body = RequestBody.create(MediaType.parse("application/xml; charset=utf-8"), requestXml);
        Request request = new Request.Builder().url(UNIFIED_ORDER_URL).post(body).build();
        Response response = client.newCall(request).execute();

        // 解析返回结果
        String responseXml = response.body().string();
        Map<String, String> result = xmlToMap(responseXml);

        if ("SUCCESS".equals(result.get("return_code")) && "SUCCESS".equals(result.get("result_code"))) {
            return result.get("code_url"); // 返回支付二维码链接
        } else {
            throw new Exception("微信支付下单失败:" + result.get("return_msg"));
        }
    }

    private String generateSign(SortedMap<String, String> params) throws Exception {
        StringBuilder sb = new StringBuilder();
        params.forEach((key, value) -> {
            if (value != null && !"".equals(value) && !"sign".equals(key)) {
                sb.append(key).append("=").append(value).append("&");
            }
        });
        sb.append("key=").append(apiKey);
        return md5(sb.toString()).toUpperCase();
    }

    private String md5(String str) throws Exception {
        MessageDigest md = MessageDigest.getInstance("MD5");
        byte[] array = md.digest(str.getBytes("UTF-8"));
        StringBuilder sb = new StringBuilder();
        for (byte b : array) {
            sb.append(String.format("%02x", b));
        }
        return sb.toString();
    }

    private String mapToXml(SortedMap<String, String> map) {
        StringBuilder xml = new StringBuilder("<xml>");
        map.forEach((key, value) -> xml.append("<").append(key).append(">").append(value)
                .append("</").append(key).append(">"));
        xml.append("</xml>");
        return xml.toString();
    }

    private Map<String, String> xmlToMap(String xml) throws Exception {
        // 使用 Jackson 或其他工具解析 XML 字符串为 Map
        ObjectMapper objectMapper = new ObjectMapper();
        return objectMapper.readValue(xml, Map.class);
    }
}

统一下单接口调用

@RestController
@RequestMapping("/api/pay")
public class PayController {

    @Autowired
    private WeChatPayService weChatPayService;

    @PostMapping("/createOrder")
    public String createOrder(@RequestParam String orderNo,
                              @RequestParam String description,
                              @RequestParam int totalFee,
                              @RequestParam String clientIp) {
        try {
            return weChatPayService.createOrder(orderNo, description, totalFee, clientIp);
        } catch (Exception e) {
            e.printStackTrace();
            return "Error: " + e.getMessage();
        }
    }
}

4. 微信支付回调处理

配置回调接口

回调地址需在统一下单时配置,示例代码如下:

import org.springframework.web.bind.annotation.*;

import javax.servlet.http.HttpServletRequest;
import java.util.Map;

@RestController
@RequestMapping("/api/pay")
public class PayCallbackController {

    @PostMapping("/notify")
    public String weChatNotify(HttpServletRequest request) {
        try {
            // 解析 XML 请求
            String requestBody = request.getReader().lines()
                .reduce("", (accumulator, actual) -> accumulator + actual);
            Map<String, String> resultMap = xmlToMap(requestBody);

            // 验签
            String receivedSign = resultMap.get("sign");
            resultMap.remove("sign");
            String calculatedSign = generateSign(resultMap);

            if (!calculatedSign.equals(receivedSign)) {
                return "<xml><return_code>FAIL</return_code><return_msg>签名失败</return_msg></xml>";
            }

            // 处理业务逻辑
            if ("SUCCESS".equals(resultMap.get("result_code"))) {
                String orderNo = resultMap.get("out_trade_no");
                // 更新订单状态
                System.out.println("支付成功,订单号:" + orderNo);
            }

            return "<xml><return_code>SUCCESS</return_code><return_msg>OK</return_msg></xml>";
        } catch (Exception e) {
            e.printStackTrace();
            return "<xml><return_code>FAIL</return_code><return_msg>处理失败</return_msg></xml>";
        }
    }
}

5. 注意事项

  1. 安全性:使用 HTTPS 确保传输安全。
  2. 验签:每次回调需要严格验签,防止伪造回调。
  3. 金额校验:确认支付金额与订单金额一致。
  4. 防止重复通知:回调接口需支持幂等性。

通过以上步骤,可以实现微信支付接口调用及回调处理功能。如果有更多需求(如退款或查询订单状态),可以扩展上述代码。

发布者:myrgd,转载请注明出处:https://www.object-c.cn/4432

Like (0)
Previous 2024年11月23日 下午1:49
Next 2024年11月23日 下午1:56

相关推荐

  • 2024年最新的 Node.js 安装与环境配置教程

    以下是2024年最新的 Node.js 安装与环境配置教程,适用于 Windows、macOS 和 Linux 系统。 一、什么是 Node.js? Node.js 是一个基于 V8 引擎的 JavaScript 运行环境,主要用于开发服务端应用程序。它的特点是异步事件驱动和非阻塞 I/O,适合高性能应用。 二、Node.js 安装 1. 下载 Node.j…

    2024年11月23日
    00
  • Jeewx-Api 1.3.1 发布:更简易的微信小程序开发 API 降低了开发成本

    Jeewx-Api 1.3.1 发布:更简易的微信小程序开发 API Jeewx-Api 是一款开源的微信开发 SDK,支持微信公众号、小程序、企业微信等全场景开发。1.3.1 版本针对小程序功能做了全面优化,为开发者提供了更简洁易用的 API,降低了开发成本。 1. 新版本亮点 支持微信小程序功能 更简洁的 API 支持企业微信与公众号 2. Jeewx-…

    2024年12月3日
    00
  • 在 Nuxt.js 应用中,webpack 的 compile 事件钩子构建过程

    在 Nuxt.js 应用中,webpack 的 compile 事件钩子通常用于在构建过程中处理或监听 Webpack 编译的状态。webpack 是 Nuxt.js 中的核心构建工具之一,而 Nuxt.js 本身是基于 Webpack 配置的,允许你通过扩展 Webpack 配置来进行自定义。要使用 webpack 的 compile 事件钩子,首先你需要…

    2024年11月29日
    00
  • 微信小程序中,通过Node.js连接本地 MySQL 实现数据的增删改查

    在微信小程序中,通过Node.js连接本地 MySQL 实现数据的增删改查的过程,可以分为以下几个步骤: 1. 准备工作 必要工具: 2. 搭建 Node.js 后端服务 安装依赖 在 Node.js 项目中,使用 npm init 初始化项目,并安装以下依赖: express: 用于搭建 Web 服务。 mysql: 用于连接和操作 MySQL 数据库。 …

    2024年12月9日
    00
  • 解决 Vue 3 应用部署到 GitHub Pages 后,遇到 404 错误问题

    在将 Vue 3 应用部署到 GitHub Pages 后,遇到 404 错误通常是由于 GitHub Pages 处理路由时的问题。Vue 3 使用 Vue Router 来管理前端路由,而 GitHub Pages 本身是静态托管服务,不支持处理 SPA(单页面应用)的客户端路由。因此,当你直接访问某个页面 URL(比如 https://youruser…

    2024年11月29日
    00
  • 微信小程序开发中使用 Tailwind CSS 提高开发效率和代码的可维护性

    Tailwind CSS 是一个利用原子化 CSS 类来构建用户界面的框架,在微信小程序开发中使用 Tailwind CSS 可以提高开发效率和代码的可维护性。以下是在微信小程序中使用 Tailwind CSS 进行原子 CSS 开发的具体步骤: 安装 Tailwind CSS 配置 Tailwind CSS 引入样式:在微信小程序的全局样式文件app.wx…

    2024年12月15日
    00
  • 在 Ant Design ProTable 中,如何设置不分页,依然显示分页信息,前端分页不触发

    在 Ant Design ProTable 中,默认情况下,分页是与数据请求(request)相关联的。也就是说,每当分页切换时,request 会被触发,重新请求新的数据。如果你希望在禁用分页的同时,依然显示分页控件,并且不触发 request 请求,可以通过以下方法进行配置。解决方案要在 Ant Design ProTable 中禁用分页的同时保留分页信…

    2024年11月29日
    00
  • 修复 Elementor 网站上出现的 HTTPS 400 错误请求(服务器错误)

    在修复 Elementor 网站上出现的 HTTPS 400 错误请求(服务器错误)时,您需要采取以下步骤来排查和解决问题。这类错误通常与服务器配置、插件冲突或 HTTPS 配置相关。 1. 检查 HTTPS 配置 2. 排查插件冲突 3. 检查主题兼容性 4. 调整服务器配置 URL 重写规则: 5. 清理缓存 6. 检查网络请求 7. 联系主机服务商 如…

    2024年12月9日
    00
  • 微信小程序使用 map 组件实现拖动地图并获取当前地图中心的经纬度

    在微信小程序中,使用 map 组件可以轻松实现拖动地图并获取当前地图中心的经纬度。以下是实现步骤和代码示例: 实现思路 代码实现 1. 页面 WXML 添加 map 组件并设置属性。 2. 页面 WXSS 定义地图样式和中心点标记样式。 3. 页面 JS 初始化地图中心点的经纬度,并监听地图拖动。 4. 中心点图标 将一个中心标记图标放置在地图中间。可以在小…

    2024年11月25日
    00
  • uni-app 一个使用 Vue.js 开发所有前端应用的框架跨端开发的优势

    uni-app 是一个使用 Vue.js 开发所有前端应用的框架,开发者编写一套代码,可发布到 iOS、Android、Web(响应式)、以及各种小程序(微信 / 支付宝 / 百度 / 头条 / QQ / 钉钉)等多个平台,其跨端开发具有以下优势: 开发效率高 跨平台兼容性好 性能优化 生态丰富 学习成本低

    2024年12月22日
    00
  • 把一个矩形div变成一个直角三角形

    要将一个矩形 div 变成一个直角三角形,可以通过 CSS 的一些属性进行实现。具体的方法是利用 border 属性来隐藏矩形的部分,留下一个直角三角形的形状。以下是实现的步骤:CSS 方法: 解释:width 和 height 设置为 0:通过将 div 的宽度和高度设置为 0,实际上把矩形的主体部分隐藏了。border-left 和 border-bot…

    2024年11月27日
    00
  • Python 的 json模块序列化数据从文件里读取出来或存入文件

    Python 的 json 模块用于处理 JSON 格式的数据,可以将 JSON 数据与 Python 数据结构之间相互转换。以下是具体用法,包括从文件读取 JSON 数据以及将数据写入文件: 1. 将 JSON 数据从文件中读取到 Python 数据结构 代码示例: 解析过程: 2. 将 Python 数据结构写入到文件中(序列化为 JSON) 代码示例:…

    2024年11月26日
    00
  • Flutter 在 PC 端多窗口支持方面的进展备受关注的功能,已在 Ubuntu/Canonical 展示

    Flutter 在 PC 端多窗口支持方面的进展是一个备受关注的功能,它的目标是进一步提升 Flutter 的跨平台能力。最近,Flutter 团队与 Canonical 合作,在 Ubuntu 平台上展示了多窗口功能的新成果。这一进展对开发者和终端用户来说都有重要意义,以下是相关细节分析: 展示细节多窗口功能亮相Flutter 的多窗口支持在 Ubuntu…

    2024年12月2日
    00
  • 如何用pbootcmsAPI接口开发微信小程序UNIAPP

    使用 PbootCMS 的 API 接口结合 UniApp 开发微信小程序,可以实现高效的内容管理和展示。以下是一个完整的开发流程,包括 API 接口设置、小程序功能设计和开发细节。 1. 准备工作1.1 配置 PbootCMS API 接口PbootCMS 提供 API 功能,需在后台开启并配置:登录 PbootCMS 后台管理。前往 系统管理 -> AP…

    2024年11月28日
    00
  • 在开发 Angular 项目时,使用 RangeSlider 控件时的常见问题

    在开发 Angular 项目时,使用 RangeSlider 控件时,可能会遇到一些常见的问题。以下是一些问题及其解决方案:1. 滑块值不更新问题描述:当用户拖动滑块时,滑块的值不会实时更新,或者显示的值不正确。解决方案:确保绑定的模型是双向绑定,使用 ngModel 或者监听 input 事件来确保滑块值能实时更新。例如: 确保你已经导入了 FormsMo…

    2024年11月27日
    00

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

联系我们

在线咨询: QQ交谈

邮件:723923060@qq.com

关注微信