【移动应用开发技术】php实现微信小程序支付及退款的方法_第1页
【移动应用开发技术】php实现微信小程序支付及退款的方法_第2页
【移动应用开发技术】php实现微信小程序支付及退款的方法_第3页
【移动应用开发技术】php实现微信小程序支付及退款的方法_第4页
【移动应用开发技术】php实现微信小程序支付及退款的方法_第5页
已阅读5页,还剩9页未读 继续免费阅读

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

【移动应用开发技术】php实现微信小程序支付及退款的方法

这篇文章给大家分享的是有关php实现微信小程序支付及退款的方法的内容。在下觉得挺实用的,因此分享给大家做个参考。一起跟随在下过来看看吧。一.支付支付主要分为几个步骤:前端携带支付需要的数据(商品id,购买数量等)发起支付请求后端在接收到支付请求后,处理支付数据,然后携带处理后的数据请求微信服务器的支付统一下单接口后端接收到上一步请求微信服务器的返回数据,再次处理,然后返回前端让前端可以开始支付。前端进行支付动作前端支付完成后,微信服务器会向后端发送支付通知(也就是微信要告诉你客户已经付过钱了),后端根据这个通知确定支付完成,然后就去做支付完成后的相应动作,比如修改订单状态,添加交易日志啊等等。从这几个步骤可以看出,后端主要的作用就是将支付需要的数据传给微信服务器,再根据微信服务器的响应确定支付是否完成。这个流程还是蛮容易理解的。形象的说,前端就是个顾客,后端就是店家,微信服务器的统一下单接口就像收银员。顾客跟店家说,我是谁谁谁,现在我要付多少多少钱给你买什么什么。店家就跟收银员说,那个谁谁谁要付多少钱,你准备收钱吧。收银员收到钱后,就去告诉店家,我已经收到钱了,你给他东西吧。下面就详细的说明一下各个步骤的具体实现。1.前端请求支付前端请求支付,就是简单的携带支付需要的数据,例如用户标识,支付金额,支付订单ID等等跟**你的业务逻辑有关**或者跟

**下一步请求微信服务器支付统一下单接口需要的数据有关**的相关数据,使用微信小程序的wx.request()去请求后端的支付接口。2.后端请求微信服务器后端接收到前端发送的支付请求后,可以进行一下相关验证,例如判断一下用户有没有问题,支付金额对不对等等。在验证没什么问题,可以向微信服务器申请支付之后,后端需要使用微信规定的数据格式去请求微信的支付统一下单接口。微信规定的请求数据:这需要较多代码实现。因为需要的数据个数较多,而且还需要加密并以XML

格式发送。首先,有以下数据是使用小程序支付必须提供给微信服务器的参数。小程序appid。写小程序的大概没有不知道这个的。。。用户标识openid。也就是用户的小程序标识,在我上篇博客中说明了如何获取。商户号mch_id。申请开通微信支付商户认证成功后微信发给你的邮件里有商户订单号out_trade_no。商户为这次支付生成的订单号总金额total_fee。订单总金额,很重要的一点是单位是分,要特别注意。微信服务器回调通知接口地址

notify_url。微信确认钱已经到账后,会往这个地址多次发送消息,告诉你顾客已经付完钱了,你需要返回消息给微信表示你已经收到了通知。。这个地址不能有端口号,同时要能直接接受POST方法请求。交易类型trade_type。微信小程序支付此值统一为JSAPI商品信息Body。类似"腾讯-游戏"这种格式终端IP地址spbill_create_ip。终端地址IP,也就是请求支付的IP地址。随机字符串nonce_str。需要后端随机生成的字符串用于保证数据安全。微信要求不长于32位。签名sign。使用上面的所有参数进行相应处理加密生成签名。(具体处理方式可见下文代码,可直接复用。)在处理好以上所有数据后,将这些数据以XML格式整理并以POST方法发送到微信支付统一下单接口3.后端接受微信服务器返回数据微信服务器在接收到支付数据之后,如果数据没有问题,其会返回用于支付的相应数据,其中非常重要的是名称为prepay_id

的数据字段,需要将此数据返回前端,前端才能继续支付。因此,在后端接收到微信服务器的返回数据后,需要进行相应的处理,最终返回到前端如下数据:appid不需多说timeStamp当前时间戳nonceStr随机字符串package就是上面提到的prepay_id,不过切记格式如“prepay_id=prepay_id_item“。否则会导致错误。signType加密方式,一般应该是MD5paySign对以上数据进行相应处理并加密。到这里,后端的支付接口已经完成了接收前端支付请求,并返回了前端支付所需数据的功能。4.前端发起支付前端在接收到返回数据后,使用wx.requestPayment()来请求发起支付。此API

需要的对象参数各项值就是我们上一步返回的各个数据。5.后端接受微信服务器回调前端完成支付后,微信服务器确认支付已经完成。就会向第一步中设置的回调地址发送通知。后端的接收回调接口在接收到通知后,就可以判断支付是否完成,从而决定后续动作。需要注意的是,在接收到微信服务器的回调通知后,根据通知的result_code字段判断支付是否成功。在接受到成功的通知后,后端需要返回success数据向微信服务器告知已得到回调通知。否则微信服务器会不停的向后端发送消息。另外微信的通知是以XML格式发送的,在接受处理时需要注意。微信的大概支付流程就是这样。以下是PHP语法的微信支付类,可以比照上面的步骤介绍,加深理解。在需要支付时,直接传入参数实例化此类再调用类的pay

方法即可。//微信支付类

class

WeiXinPay{

//=======【基本信息设置】=====================================

//微信公众号身份的唯一标识

protected

$APPID

=

appid;//填写您的appid。微信公众平台里的

protected

$APPSECRET

=

secret;

//受理商ID,身份标识

protected

$MCHID

=

'11111111';//商户id

//商户支付密钥Key

protected

$KEY

=

'192006250b4c09247ec02edce69f6a2d';

//回调通知接口

protected

$APPURL

=

'/receivesuc';

//交易类型

protected

$TRADETYPE

=

'JSAPI';

//商品类型信息

protected

$BODY

=

'wx/book';

//微信支付类的构造函数

function

__construct($openid,$outTradeNo,$totalFee){

$this->openid

=

$openid;

//用户唯一标识

$this->outTradeNo

=

$outTradeNo;

//商品编号

$this->totalFee

=

$totalFee;

//总价

}

//微信支付类向外暴露的支付接口

public

function

pay(){

$result

=

$this->weixinapp();

return

$result;

}

//对微信统一下单接口返回的支付相关数据进行处理

private

function

weixinapp(){

$unifiedorder=$this->unifiedorder();

$parameters=array(

'appId'=>$this->APPID,//小程序ID

'timeStamp'=>''.time().'',//时间戳

'nonceStr'=>$this->createNoncestr(),//随机串

'package'=>'prepay_id='.$unifiedorder['prepay_id'],//数据包

'signType'=>'MD5'//签名方式

);

$parameters['paySign']=$this->getSign($parameters);

return

$parameters;

}

/*

*请求微信统一下单接口

*/

private

function

unifiedorder(){

$parameters

=

array(

'appid'

=>

$this->APPID,//小程序id

'mch_id'=>

$this->MCHID,//商户id

'spbill_create_ip'=>$_SERVER['REMOTE_ADDR'],//终端ip

'notify_url'=>$this->APPURL,

//通知地址

'nonce_str'=>

$this->createNoncestr(),//随机字符串

'out_trade_no'=>$this->outTradeNo,//商户订单编号

'total_fee'=>floatval($this->totalFee),

//总金额

'open_id'=>$this->openid,//用户openid

'trade_type'=>$this->TRADETYPE,//交易类型

'body'

=>$this->BODY,

//商品信息

);

$parameters['sign']

=

$this->getSign($parameters);

$xmlData

=

$this->arrayToXml($parameters);

$xml_result

=

$this->postXmlCurl($xmlData,'/pay/unifiedorder',60);

$result

=

$this->xmlToArray($xml_result);

return

$result;

}

//数组转字符串方法

protected

function

arrayToXml($arr){

$xml

=

"<xml>";

foreach

($arr

as

$key=>$val)

{

if

(is_numeric($val)){

$xml.="<".$key.">".$val."</".$key.">";

}else{

$xml.="<".$key."><![CDATA[".$val."]]></".$key.">";

}

}

$xml.="</xml>";

return

$xml;

}

protected

function

xmlToArray($xml){

$array_data

=

json_decode(json_encode(simplexml_load_string($xml,

'SimpleXMLElement',

LIBXML_NOCDATA)),

true);

return

$array_data;

}

//发送xml请求方法

private

static

function

postXmlCurl($xml,

$url,

$second

=

30)

{

$ch

=

curl_init();

//设置超时

curl_setopt($ch,

CURLOPT_TIMEOUT,

$second);

curl_setopt($ch,

CURLOPT_URL,

$url);

curl_setopt($ch,

CURLOPT_SSL_VERIFYPEER,

FALSE);

curl_setopt($ch,

CURLOPT_SSL_VERIFYHOST,

FALSE);

//严格校验

//设置header

curl_setopt($ch,

CURLOPT_HEADER,

FALSE);

//要求结果为字符串且输出到屏幕上

curl_setopt($ch,

CURLOPT_RETURNTRANSFER,

TRUE);

//post提交方式

curl_setopt($ch,

CURLOPT_POST,

TRUE);

curl_setopt($ch,

CURLOPT_POSTFIELDS,

$xml);

curl_setopt($ch,

CURLOPT_CONNECTTIMEOUT,

20);

curl_setopt($ch,

CURLOPT_TIMEOUT,

40);

set_time_limit(0);

//运行curl

$data

=

curl_exec($ch);

//返回结果

if

($data)

{

curl_close($ch);

return

$data;

}

else

{

$error

=

curl_errno($ch);

curl_close($ch);

throw

new

WxPayException("curl出错,错误码:$error");

}

}

/*

*

对要发送到微信统一下单接口的数据进行签名

*/

protected

function

getSign($Obj){

foreach

($Obj

as

$k

=>

$v){

$Parameters[$k]

=

$v;

}

//签名步骤一:按字典序排序参数

ksort($Parameters);

$String

=

$this->formatBizQueryParaMap($Parameters,

false);

//签名步骤二:在string后加入KEY

$String

=

$String."&key=".$this->KEY;

//签名步骤三:MD5加密

$String

=

md5($String);

//签名步骤四:所有字符转为大写

$result_

=

strtoupper($String);

return

$result_;

}

/*

*排序并格式化参数方法,签名时需要使用

*/

protected

function

formatBizQueryParaMap($paraMap,

$urlencode)

{

$buff

=

"";

ksort($paraMap);

foreach

($paraMap

as

$k

=>

$v)

{

if($urlencode)

{

$v

=

urlencode($v);

}

//$buff

.=

strtolower($k)

.

"="

.

$v

.

"&";

$buff

.=

$k

.

"="

.

$v

.

"&";

}

$reqPar;

if

(strlen($buff)

>

0)

{

$reqPar

=

substr($buff,

0,

strlen($buff)-1);

}

return

$reqPar;

}

/*

*

生成随机字符串方法

*/

protected

function

createNoncestr($length

=

32

){

$chars

=

"abcdefghijklmnopqrstuvwxyz0123456789";

$str

="";

for

(

$i

=

0;

$i

<

$length;

$i++

)

{

$str.=

substr($chars,

mt_rand(0,

strlen($chars)-1),

1);

}

return

$str;

}

}以上就是微信支付的相关流程。在理清思路后,流程还是比较清晰和简单的。重点在于需要注意一些细节问题,例如数据格式,加密方法等。下面说一下微信小程序退款的具体实现二.退款小程序退款的流程和付款相似,但有一些细节上的不同。首先退款的步骤通常如下:用户前端点击退款按钮后,后端接收到用户的退款请求通过商城后台呈现给商户,商户确定允许退款后,后端再发起向微信退款接口的请求来请求退款。后端向微信退款接口发送请求后,得到响应信息,确定退款是否完成,根据退款是否完成再去进行改变订单状态等业务逻辑。退款的步骤相对微信支付来说比较简单。值得注意的有以下两点:1.向微信退款接口请求退款后,根据得到的响应是可以直接确定退款是否完成的。不再需要设置专门的回调接口等待微信通知。当然如果需要也是可以在微信商户平台设置回调接口接受从而接受微信回调的,但并不是必须的。2.退款请求需要在请求服务器安装微信提供的安全证书,也就是说,发起退款请求相比较支付请求在请求时请求方法不能复用,因为微信退款需要携带证书的请求,此证书可在申请微信商户号成功后从微信商户平台自行下载,Linux下的PHP开发环境的证书只需要放在网站根目录的cert文件夹中即可。其他开发环境可能需要导入操作。下面讲解一下退款的具体步骤一.用户发起退款请求用户在前端发起退款请求,后端接收到退款请求,将相应订单标记为申请退款,展示在后台.商户查看后,如果同意退款再进行相应操作.此后才进入真正的退款流程.二.商户发起退款请求商户同意退款后,后端即向微信提供的退款API发起请求.同请求微信支付API一样.退款请求也需要将需要的参数进行签名后以XML发送到微信的退款API退款请求需要的参数如下(多个参数在支付API请求时也有使用):小程序appid。商户号mch_id。申请开通微信支付商户认证成功后微信发给你的邮件里有商户订单号out_trade_no。退款订单在支付时生成的订单号退款订单号out_refund_no。由后端生成的退款单号,需要保证唯一,因为多个同样的退款单号只会退款一次。总金额total_fee。订单总金额,单位为分。退款金额refund_fee需要退款的金额,单位同样为分操作员op_user_id.与商户号相同即可随机字符串nonce_str。同支付请求签名sign。使用上面的所有参数进行相应处理加密生成签名。(具体处理方式与支付相同,可直接复用。)三.退款完成在发起退款请求后,就可以直接根据请求的响应XML中的

result_code字段来判断退款是否成功,从而对订单状态进行处理和后续操作。不需要像支付那样等待另一个接口的通知来确定请求状态。当然如上文所说,如果需要微信服务器发送通知到后端的话,可以到微信商户平台进行设置。退款因为流程与支付大同小异,因此退款的PHP类我选择了直接继承支付类,代码如下,注意区分退款请求方法postXmlSSLCurl和支付请求方法postXmlCurl的区别,这也就是上文提到的退款需要的双向证书的使用。````

class

WinXinRefund

extends

WeiXinPay{

protected

\$SSLCERT_PATH

=

'cert/apiclient_cert.pem';//证书路径

protected

\$SSLKEY_PATH

=

'cert/apiclient_key.pem';//证书路径

protected

\$opUserId

=

'1234567899';//商户号

function

__construct($openid,$outTradeNo,$totalFee,$outRefundNo,$refundFee){

//初始化退款类需要的变量

$this->openid

=

$openid;

$this->outTradeNo

=

$outTradeNo;

$this->totalFee

=

$totalFee;

$this->outRefundNo

=

$outRefundNo;

$this->refundFee

=

$refundFee;

}

public

function

refund(){

//对外暴露的退款接口

$result

=

$this->wxrefundapi();

return

$result;

}

private

function

wxrefundapi(){

//通过微信api进行退款流程

$parma

=

array(

'appid'=>

$this->APPID,

'mch_id'=>

$this->MCHID,

'nonce_str'=>

$this->createNoncestr(),

'out_refund_no'=>

$this->outRefundNo,

'out_trade_no'=>

$this->outTradeNo,

'total_fee'=>

$this->totalFee,

'refund_fee'=>

$this->refundFee,

'op_user_id'

=>

$this->opUserId,

);

$parma['sign']

=

$this->getSign($parma);

$xmldata

=

$this->arrayToXml($parma);

$xmlresult

=

$this->postXmlSSLCurl($xmldata,'/secapi/pay/refund');

$result

=

$this->xmlToArray($xmlresult);

return

$result;

}

//需要使用证书的请求

function

postXmlSSLCurl($xml,$url,$second=30)

{

$ch

=

curl_init();

//超时时间

curl_setopt($ch,CURLOPT_TIMEOUT,$second);

//这里设置代理,如果有的话

//curl_setopt($ch,CURLOPT_PROXY,

'');

//curl_setopt($ch,CURLOPT_PROXYPORT,

8080);

curl_setopt($ch,CURLOPT_URL,

$url);

curl_setopt($ch,CURLOPT_SSL_VERIFYPEER,FALSE);

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

评论

0/150

提交评论