用PHP开发APP端微信支付的一点个人心得
最近因为公司需求,要开发APP端上的微信支付,看了微信文档,感觉还不错,没有遇到太大的坑,需要注意的点不算太多。
写一个记事文档,作为备忘录。
APP支付流程
从上面的图片中,可以看出来,需要注意的流程是一共是3部分;
第一部分:调用下单API,返回预支付订单,签名之后再返回信息(4、5、6、7)
第二部分:异步通知(15、16)
第三部分:最后的判断支付结果
最需要注意的就是第一部分:调用下单API,返回预支付订单,签名之后再返回信息
微信文档中有详细的说明,这里不再赘述。
附录一下我的代码,伸手党,稍微改点代码就可以用了。
//入口函数
function weChatPay(){
$json = array();
//生成预支付交易单的必选参数:
$newPara = array();
//应用ID
$newPara["appid"] = "wx2421b1c4370ec43b";
//商户号
$newPara["mch_id"] = "10000100";
//设备号
$newPara["device_info"] = "WEB";
//随机字符串,这里推荐使用函数生成
$newPara["nonce_str"] = "1add1a30ac87aa2db72f57a2375d8fec";
//商品描述
$newPara["body"] = "APP支付测试";
//商户订单号,这里是商户自己的内部的订单号
$newPara["out_trade_no"] = "1415659990";
//总金额
$newPara["total_fee"] = 1;
//终端IP
$newPara["spbill_create_ip"] = $_SERVER["REMOTE_ADDR"];
//通知地址,注意,这里的url里面不要加参数
$newPara["notify_url"] = "http://wxpay.wxutil.com/pub_v2/pay/notify.v2.php";
//交易类型
$newPara["trade_type"] = "APP";
//第一次签名
$newPara["sign"] = produceWeChatSign($newPara);
//把数组转化成xml格式
$xmlData = getWeChatXML($newPara);
//利用PHP的CURL包,将数据传给微信统一下单接口,返回正常的prepay_id
$get_data = sendPrePayCurl($xmlData);
//返回的结果进行判断。
if($get_data['return_code'] == "SUCCESS" && $get_data['result_code'] == "SUCCESS"){
//根据微信支付返回的结果进行二次签名
//二次签名所需的随机字符串
$newPara["nonce_str"] = "5K8264ILTKCH16CQ2502SI8ZNMTM67VS";
//二次签名所需的时间戳
$newPara['timeStamp'] = time()."";
//二次签名剩余参数的补充
$secondSignArray = array(
"appid"=>$newPara['appid'],
"noncestr"=>$newPara['nonce_str'],
"package"=>"Sign=WXPay",
"prepayid"=>$get_data['prepay_id'],
"partnerid"=>$newPara['mch_id'],
"timestamp"=>$newPara['timeStamp'],
);
$json['datas'] = $secondSignArray;
$json['ordersn'] = $newPara["out_trade_no"];
$json['datas']['sign'] = weChatSecondSign($newPara,$get_data['prepay_id']);
$json['message'] = "预支付完成";
//预支付完成,在下方进行自己内部的业务逻辑
/*****************************/
return json_encode($json);
}
else{
$json['message'] = $get_data['return_msg'];
}
}
return json_encode($json);
}
//第一次签名的函数produceWeChatSign
function produceWeChatSign($newPara){
$stringA = self::getSignContent($newPara);
$stringSignTemp=$stringA."&key=192006250b4c09247ec02edce69f6a2d";
return strtoupper(MD5($stringSignTemp));
}
//生成xml格式的函数
public static function getWeChatXML($newPara){
$xmlData = "<xml>";
foreach ($newPara as $key => $value) {
$xmlData = $xmlData."<".$key.">".$value."</".$key.">";
}
$xmlData = $xmlData."</xml>";
return $xmlData;
}
//通过curl发送数据给微信接口的函数
function sendPrePayCurl($xmlData) {
$url = "https://api.mch.weixin.qq.com/pay/unifiedorder";
$header[] = "Content-type: text/xml";
$curl = curl_init();
curl_setopt($curl, CURLOPT_HTTPHEADER, $header);
curl_setopt($curl, CURLOPT_URL, $url);
curl_setopt($curl, CURLOPT_RETURNTRANSFER, true);
curl_setopt($curl, CURLOPT_POST, 1);
curl_setopt($curl, CURLOPT_POSTFIELDS, $xmlData);
$data = curl_exec($curl);
if (curl_errno($curl)) {
print curl_error($curl);
}
curl_close($curl);
return self::XMLDataParse($data);
}
//xml格式数据解析函数
public static function XMLDataParse($data){
$msg = array();
$msg = (array)simplexml_load_string($data, 'SimpleXMLElement', LIBXML_NOCDATA);
return $msg;
}
//二次签名的函数
function weChatSecondSign($newPara,$prepay_id){
$secondSignArray = array(
"appid"=>$newPara['appid'],
"noncestr"=>$newPara['nonce_str'],
"package"=>"Sign=WXPay",
"prepayid"=>$prepay_id,
"partnerid"=>$newPara['mch_id'],
"timestamp"=>$newPara['timeStamp'],
);
$stringA = self::getSignContent($secondSignArray);
$stringSignTemp=$stringA."&key=192006250b4c09247ec02edce69f6a2d";
return strtoupper(MD5($stringSignTemp));
}
两个注意点:
1.二次签名需要在后台完成,并且完成之后,连带着二次签名所用的所有信息一起传给前端,让前段唤起微信支付。这样不容易出现没法吊起微信支付的情况。
2.两次签名,用的是不同的随机字符串