微信小程序 在线支付功能的实现

发布时间 - 2026-01-11 00:10:03    点击率:

微信小程序 在线支付功能

最近需要在微信小程序中用到在线支付功能,于是看了一下官方的文档,发现要在小程序里实现微信支付还是很方便的,如果你以前开发过服务号下的微信支付,那么你会发现其实小程序里的微信支付和服务号里的开发过程如出一辙,下面我就具体说一下小程序里微信支付的开发流程和注意点。

1.开通微信支付和微信商户号

这个过程就和开通服务号的微信支付过程一样,没有什么可以说的。


2.获得用户的openid

首页我们需要在小程序的客户端js中获取当前用户的openid,通过调用wx.login方法可以得到用户的code,然后开发者服务器使用登录凭证 code 获取 openid。

wx.login({
   success: function(res) {
    if (res.code) {
     //发起网络请求
     wx.request({
      url: 'https://yourwebsit/onLogin',
      method: 'POST',
      data: {
       code: res.code
      },
      success: function(res) {
        var openid = res.data.openid;
      },
      fail: function(err) {
        console.log(err)
      }
     })
    } else {
     console.log('获取用户登录态失败!' + res.errMsg)
    }
   }
  });

var code = req.param("code");
    request({
      url: "https://api.weixin.qq.com/sns/jscode2session?appid="+appid+"&secret="+secret+"&js_code="+code+"&grant_type=authorization_code",
      method: 'GET'
    }, function(err, response, body) {
      if (!err && response.statusCode == 200) {
        res.json(JSON.parse(body));
      }
    });

3.获取prepay_id和支付签名验证paySign

这一步的过程就和服务号里的微信支付过程一样,分为客户端和服务器端

首先来看一下客户端js

在服务号里,我们是通过如下的代码来调起支付功能

function jsApiCall()
    {
      WeixinJSBridge.invoke(
        'getBrandWCPayRequest',
        {
          "appId":"",   //公众号名称,由商户传入   
          "timeStamp":"",     //时间戳,自1970年以来的秒数   
          "nonceStr":"", //随机串   
          "package":"prepay_id=<%=prepay_id%>",   
          "signType":"MD5",     //微信签名方式:   
          "paySign":"<%=_paySignjs%>" //微信签名
        },
        function(res){
          WeixinJSBridge.log(res.err_msg);
          if( res.err_msg =="get_brand_wcpay_request:ok"){
            alert("支付成功!");
          }else{
            alert("支付失败!");
          }
        }
      );
    }

在小程序里,我们是通过wx.requestPayment方法来调起支付功能,当然在这之前,我们先要获取prepay_id。

 wx.request({
          url: 'https://yourwebsit/service/getPay', 
          method: 'POST',
          data: {
           bookingNo:bookingNo, /*订单号*/
           total_fee:total_fee,  /*订单金额*/
           openid:openid
          },
          header: {
            'content-type': 'application/json'
          },
          success: function(res) {
            wx.requestPayment({
             'timeStamp':timeStamp,
             'nonceStr': nonceStr,
             'package': 'prepay_id='+res.data.prepay_id,
             'signType': 'MD5',
             'paySign': res.data._paySignjs,
             'success':function(res){
               console.log(res);
             },
             'fail':function(res){
               console.log('fail:'+JSON.stringify(res));
             }
            })
          },
          fail: function(err) {
            console.log(err)
          }
        })

那在服务器端主要要实现的是prepay_id的获取和签名paySign

 var bookingNo = req.param("bookingNo");
    var total_fee = req.param("total_fee");
    var openid = req.param("openid");
    var body = "费用说明";
    var url = "https://api.mch.weixin.qq.com/pay/unifiedorder";
    var formData = "<xml>";
    formData += "<appid>appid</appid>"; //appid
    formData += "<attach>test</attach>";
    formData += "<body>" + body + "</body>";
    formData += "<mch_id>mch_id</mch_id>"; //商户号
    formData += "<nonce_str>nonce_str</nonce_str>";
    formData += "<notify_url>notify_url</notify_url>";
    formData += "<openid>" + openid + "</openid>";
    formData += "<out_trade_no>" + bookingNo + "</out_trade_no>";
    formData += "<spbill_create_ip>spbill_create_ip</spbill_create_ip>";
    formData += "<total_fee>" + total_fee + "</total_fee>";
    formData += "<trade_type>JSAPI</trade_type>";
    formData += "<sign>" + paysignjsapi(appid, attach, body, mch_id, nonce_str, notify_url, openid, bookingNo, spbill_create_ip, total_fee, 'JSAPI') + "</sign>";
    formData += "</xml>";
    request({
      url: url,
      method: 'POST',
      body: formData
    }, function(err, response, body) {
      if(!err && response.statusCode == 200) {
        var prepay_id = getXMLNodeValue('prepay_id', body.toString("utf-8"));
        var tmp = prepay_id.split('[');
        var tmp1 = tmp[2].split(']');
        //签名
        var _paySignjs = paysignjs(appid, mch_id, 'prepay_id=' + tmp1[0], 'MD5',timeStamp);
        var o = {
          prepay_id: tmp1[0],
          _paySignjs: _paySignjs
        }
        res.send(o);
      }
    });

下面是用到的函数

function paysignjs(appid, nonceStr, package, signType, timeStamp) {
  var ret = {
    appId: appid,
    nonceStr: nonceStr,
    package: package,
    signType: signType,
    timeStamp: timeStamp
  };
  var string = raw1(ret);
  string = string + '&key='+key;
  console.log(string);
  var crypto = require('crypto');
  return crypto.createHash('md5').update(string, 'utf8').digest('hex');
};

function raw1(args) {
  var keys = Object.keys(args);
  keys = keys.sort()
  var newArgs = {};
  keys.forEach(function(key) {
    newArgs[key] = args[key];
  });

  var string = '';
  for(var k in newArgs) {
    string += '&' + k + '=' + newArgs[k];
  }
  string = string.substr(1);
  return string;
};

function paysignjsapi(appid, attach, body, mch_id, nonce_str, notify_url, openid, out_trade_no, spbill_create_ip, total_fee, trade_type) {
  var ret = {
    appid: appid,
    attach: attach,
    body: body,
    mch_id: mch_id,
    nonce_str: nonce_str,
    notify_url: notify_url,
    openid: openid,
    out_trade_no: out_trade_no,
    spbill_create_ip: spbill_create_ip,
    total_fee: total_fee,
    trade_type: trade_type
  };
  var string = raw(ret);
  string = string + '&key='+key;
  var crypto = require('crypto');
  return crypto.createHash('md5').update(string, 'utf8').digest('hex');
};

function raw(args) {
  var keys = Object.keys(args);
  keys = keys.sort()
  var newArgs = {};
  keys.forEach(function(key) {
    newArgs[key.toLowerCase()] = args[key];
  });

  var string = '';
  for(var k in newArgs) {
    string += '&' + k + '=' + newArgs[k];
  }
  string = string.substr(1);
  return string;
};

function getXMLNodeValue(node_name, xml) {
  var tmp = xml.split("<" + node_name + ">");
  var _tmp = tmp[1].split("</" + node_name + ">");
  return _tmp[0];
}

这样简单3步,小程序的微信支付功能就接上了,下面是测试的支付效果图


感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!


# 微信小程序  # 在线支付  # 小程序在线支付功能  # 小程序  # 支付  # 微信小程序支付及退款流程详解  # 微信小程序 支付功能开发错误总结  # 微信小程序 支付简单实例及注意事项  # 微信小程序调用微信支付接口的实现方法  # 微信小程序微信支付接入开发实例详解  # 微信小程序进行微信支付的步骤昂述  # 详解微信小程序支付流程与梳理  # 微信小程序中实现微信支付完整步骤  # 商户  # 客户端  # 的是  # 和服务  # 我就  # 如果你  # 看了  # 在这  # 可以说  # 要在  # 没有什么  # 希望能  # 谢谢大家  # 可以得到  # 首页  # 用户登录  # 方法来  # 很方便  # 你会发现 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: Laravel如何使用Gate和Policy进行授权?(权限控制)  Laravel如何为API生成Swagger或OpenAPI文档  Laravel Admin后台管理框架推荐_Laravel快速开发后台工具  图册素材网站设计制作软件,图册的导出方式有几种?  Win11搜索不到蓝牙耳机怎么办 Win11蓝牙驱动更新修复【详解】  如何在万网利用已有域名快速建站?  广州网站制作公司哪家好一点,广州欧莱雅百库网络科技有限公司官网?  桂林网站制作公司有哪些,桂林马拉松怎么报名?  html如何与html链接_实现多个HTML页面互相链接【互相】  Laravel怎么判断请求类型_Laravel Request isMethod用法  如何在香港服务器上快速搭建免备案网站?  python中快速进行多个字符替换的方法小结  iOS正则表达式验证手机号、邮箱、身份证号等  HTML5建模怎么导出为FBX格式_FBX格式兼容性及导出步骤【指南】  Laravel如何部署到服务器_线上部署Laravel项目的完整流程与步骤  JS碰撞运动实现方法详解  北京网站制作费用多少,建立一个公司网站的费用.有哪些部分,分别要多少钱?  Linux虚拟化技术教程_KVMQEMU虚拟机安装与调优  Laravel如何使用withoutEvents方法临时禁用模型事件  西安专业网站制作公司有哪些,陕西省建行官方网站?  宙斯浏览器怎么屏蔽图片浏览 节省手机流量使用设置方法  如何实现javascript表单验证_正则表达式有哪些实用技巧  深圳防火门网站制作公司,深圳中天明防火门怎么编码?  node.js报错:Cannot find module &#39;ejs&#39;的解决办法  厦门模型网站设计制作公司,厦门航空飞机模型掉色怎么办?  Laravel怎么上传文件_Laravel图片上传及存储配置  网易LOFTER官网链接 老福特网页版登录地址  网站广告牌制作方法,街上的广告牌,横幅,用PS还是其他软件做的?  免费视频制作网站,更新又快又好的免费电影网站?  如何用好域名打造高点击率的自主建站?  ,网页ppt怎么弄成自己的ppt?  黑客入侵网站服务器的常见手法有哪些?  大连网站制作公司哪家好一点,大连买房网站哪个好?  学生网站制作软件,一个12岁的学生写小说,应该去什么样的网站?  Laravel怎么实现微信登录_Laravel Socialite第三方登录集成  Win11应用商店下载慢怎么办 Win11更改DNS提速下载【修复】  PHP的CURL方法curl_setopt()函数案例介绍(抓取网页,POST数据)  再谈Python中的字符串与字符编码(推荐)  Laravel怎么生成二维码图片_Laravel集成Simple-QrCode扩展包与参数设置【实战】  如何在 Go 中优雅地映射具有动态字段的 JSON 对象到结构体  Laravel如何连接多个数据库_Laravel多数据库连接配置与切换教程  Laravel Pest测试框架怎么用_从PHPUnit转向Pest的Laravel测试教程  如何在Tomcat中配置并部署网站项目?  JavaScript中的标签模板是什么_它如何扩展字符串功能  香港服务器WordPress建站指南:SEO优化与高效部署策略  悟空浏览器如何设置小说背景色_悟空浏览器背景色设置【方法】  什么是JavaScript解构赋值_解构赋值有哪些实用技巧  在线制作视频的网站有哪些,电脑如何制作视频短片?  如何在 Telegram Web View(iOS)中防止键盘遮挡底部输入框  Laravel如何发送系统通知_Laravel Notifications实现多渠道消息通知