APP服务端微信支付(PHP服务端)

<?php
/*
大概流程说明:
1、填写必填的参数到地址 微信统一地址 https://api.mch.weixin.qq.com/pay/unifiedorder
2、App端通过统一下单参数 返回一些数据 
3、通过得到的参数掉起微信支付,完成支付
4、此过程中需要做两次签名,
第一次签名是在上传request请求时,拼接的字符串,创建一次sign;
第二次签名是请求统一下单之后,主要是用返回的一个prepay_id的一个字符串,
根据appid、noncestr、package、partnerid、prepayid、timestamp这些字符串的内容,重新创建一次sign,
第二次的sign是为了调起微信支付用的。
*/


//填写必填的参数到地址 微信统一地址
$url = "https://api.mch.weixin.qq.com/pay/unifiedorder"; //微信统一地址

$body = "APP微信支付测试";
$appid = "wx111111111111";    //微信开放平台上的应用id
$mch_id = "2222222222"; //微信申请成功之后邮件中的商户id
$attach = "APP微信支付";
$total_fee = 1; //1分
$out_trade_no = rand(100000,999999);
$api_key = "as3aaaaaaaaaaaaaaaaaaaaaaaa"; //在微信商户平台上自己设定的api密钥 32位
$notify_url = "pay_weixin.php"; //自定义的回调程序地址

$nonce_str = $this->createNoncestr();


//获取签名必要参数
$data["appid"] = $appid;
$data["attach"] = $attach;
$data["body"] = $body;  
$data["mch_id"] = $mch_id;
$data["nonce_str"] = $nonce_str;
$data["notify_url"] = $notify_url;
$data["out_trade_no"] = $out_trade_no;
$data["spbill_create_ip"] = $this->get_client_ip();
$data["total_fee"] = $total_fee;
$data["trade_type"] = "APP";
   
//第一次签名 通过上传拼接的字符串 创建sign
$sign = $this->getSign($data);
$data["sign"] = $sign;


$xml = $this->arrayToXml($data);
$response = $this->postXmlCurl($xml, $url);


//将微信返回的结果xml转成数组
$response = $this->xmlToArray($response);


$new_arr=array();
$new_arr['appid'] = $appid;
$new_arr['partnerid'] = $mch_id;
$new_arr['prepayid'] = $response['prepay_id'];
$new_arr['package'] = "Sign=WXPay";
$new_arr['noncestr'] = $response['nonce_str'];
$new_arr['timestamp'] = time();


//第二次签名 调起微信支付
$sign2 = $this->getSign($new_arr);
$new_arr["sign"] = $sign2;


$xml2 = $this->arrayToXml($new_arr);
   
//将微信返回的结果xml转成数组
$response3 = $this->xmlToArray($xml2);


//数组转成json
echo json_encode($response3);
   


//微信支付 - 生成签名
public function getSign($Obj){
$api_key = "as3aaaaaaaaaaaaaaaaaaaaaaaa";


  foreach ($Obj as $k => $v){
      $Parameters[$k] = $v;
  }


  //签名步骤一:按字典序排序参数
  ksort($Parameters);
  $String = $this->formatBizQueryParaMap($Parameters, false);
  //echo '【string1】'.$String.'</br>';
  //签名步骤二:在string后加入KEY
  $String = $String."&key=".$api_key;
  //echo "【string2】".$String."</br>";
  //签名步骤三:MD5加密
  $String = md5($String);
  //echo "【string3】 ".$String."</br>";
  //签名步骤四:所有字符转为大写
  $result_ = strtoupper($String);
  //echo "【result】 ".$result_."</br>";
  return $result_;
}


//微信支付 - 产生随机字符串,不长于32位
public 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;
}


//微信支付 - 数组转xml
public 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;
}


//微信支付 - 将xml转为array
public function xmlToArray($xml){
  //将XML转为array
  $array_data = json_decode(json_encode(simplexml_load_string($xml, 'SimpleXMLElement', LIBXML_NOCDATA)), true);
  return $array_data;
}


//微信支付 - 以post方式提交xml到对应的接口url
public function postXmlCurl($xml,$url,$second=30){
  //初始化curl
  $ch = curl_init();
  //设置超时
  curl_setopt($ch, CURLOPT_TIMEOUT, $second);
  //这里设置代理,如果有的话
  //curl_setopt($ch,CURLOPT_PROXY, '8.8.8.8');
  //curl_setopt($ch,CURLOPT_PROXYPORT, 8080);
  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
  $data = curl_exec($ch);
  //返回结果


  if($data){
    curl_close($ch);
    return $data;
  }else{
    $error = curl_errno($ch);
    echo "curl出错,错误码:$error"."<br>";
    curl_close($ch);
    return false;
  }
}


//微信支付 - 获取当前服务器的IP
public function get_client_ip(){
  if ($_SERVER['REMOTE_ADDR']) {
     $cip = $_SERVER['REMOTE_ADDR'];
  } elseif (getenv("REMOTE_ADDR")) {
     $cip = getenv("REMOTE_ADDR");
  } elseif (getenv("HTTP_CLIENT_IP")) {
     $cip = getenv("HTTP_CLIENT_IP");
  } else {
     $cip = "unknown";
  }
  return $cip;
}


//微信支付 - 格式化参数,签名过程需要使用
public function formatBizQueryParaMap($paraMap, $urlencode){
  $buff = "";
  ksort($paraMap);
  foreach ($paraMap as $k => $v){
    if($urlencode){
        $v = urlencode($v);
    }
    $buff .= $k . "=" . $v . "&";
  }
  $reqPar;
  if (strlen($buff) > 0){
    $reqPar = substr($buff, 0, strlen($buff)-1);
  }
  return $reqPar;
}
     
  • 2
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
2018-09-03 php服务端微信支付整理SDK,封装,如果喜欢请给个好评!谢谢 说明: 配置在 WeChatConfig文件下 环境:php5.6,基于tp5开发 调用统一下单接口: include_once EXTEND_PATH . 'WeChatSDK/WeChatSDK.php'; $data = $this->getOrderInfo($pay_sn); if (!$data) { return $this->resultCode(-2019, '订单不存在或已支付'); } $WeixinPay = new \WeChatSDK(); if ($trade_type == 'JSAPI') { //目前未有此功能 $openid = ''; $product_id = ''; } if ($trade_type == 'NATIVE') { $openid = ''; $product_id = $pay_sn; } if ($trade_type == 'MWEB') { $openid = ''; $product_id = $pay_sn; } if ($trade_type == 'APP') { $openid = ''; $product_id = $pay_sn; } $out_trade_no = $pay_sn; $result = $WeixinPay->setWeiXinPay($data['pay_body'], $data['pay_detail'], $data['pay_money'] * 100, $out_trade_no, $red_url, $trade_type, $openid, $product_id); APP加密:$WeChatSDK->GetAppParameters($result['data']); web编码 $WeChatSDK->GetMwebApiParameters(); jsapi:WeChatSDK-> GetJsApiParameters(); 回调调用: include_once EXTEND_PATH . 'WeChatSDK/WeChatSDK.php'; Log::write("gwgwgwgw---------------------------------进入异步回掉"); $postStr = file_get_contents('php://input'); Log::write("gwgwgwgw---------------------------------" . $postStr); $WeChatSDK = new \WeChatSDK(); if (!empty($postStr)) { $postObj = simplexml_load_string($postStr, 'SimpleXMLElement', LIBXML_NOCDATA); $check_sign = $WeChatSDK->checkSign($postObj, $postObj->sign); Log::write('-----check_sign-------' . $check_sign . '------------check_sign--------------'); if ($postObj->result_code == 'SUCCESS' && $check_sign == 1) { model('order', 'service')->affirmPayment($postObj->out_trade_no); $xml = "<xml> <![CDATA[SUCCESS]]></return_co
以下是使用微信支付服务的示例代码: 1. 引入SDK ```java // 引入微信支付SDK import com.github.wxpay.sdk.WXPay; import com.github.wxpay.sdk.WXPayConstants; import com.github.wxpay.sdk.WXPayUtil; import java.util.HashMap; import java.util.Map; ``` 2. 初始化配置 ```java // 配置微信支付参数 Map<String, String> config = new HashMap<>(); config.put("appid", "微信开放平台审核通过的应用APPID"); config.put("mch_id", "商户号"); config.put("key", "API密钥"); config.put("notify_url", "支付结果通知的回调地址"); config.put("trade_type", "APP"); WXPay wxPay = new WXPay(config, WXPayConstants.SignType.MD5); ``` 3. 统一下单 ```java // 统一下单 Map<String, String> data = new HashMap<>(); data.put("body", "商品描述"); data.put("out_trade_no", "商户订单号"); data.put("total_fee", "支付金额"); data.put("spbill_create_ip", "用户IP地址"); Map<String, String> resp = wxPay.unifiedOrder(data); ``` 4. 处理支付结果 ```java // 处理支付结果 if ("SUCCESS".equals(resp.get("return_code")) && "SUCCESS".equals(resp.get("result_code"))) { // 获取预支付交易会话标识 String prepayId = resp.get("prepay_id"); // 签名参数 Map<String, String> signData = new HashMap<>(); signData.put("appid", config.get("appid")); signData.put("partnerid", config.get("mch_id")); signData.put("prepayid", prepayId); signData.put("package", "Sign=WXPay"); signData.put("noncestr", WXPayUtil.generateNonceStr()); signData.put("timestamp", String.valueOf(System.currentTimeMillis() / 1000)); // 生成签名 String sign = WXPayUtil.generateSignature(signData, config.get("key")); // 返回APP端需要的参数 Map<String, String> payParams = new HashMap<>(); payParams.put("appid", config.get("appid")); payParams.put("partnerid", config.get("mch_id")); payParams.put("prepayid", prepayId); payParams.put("package", "Sign=WXPay"); payParams.put("noncestr", signData.get("noncestr")); payParams.put("timestamp", signData.get("timestamp")); payParams.put("sign", sign); } else { // 支付失败 String errorCode = resp.get("err_code"); String errorMsg = resp.get("err_code_des"); } ``` 以上就是使用微信支付服务的示例代码,你可以根据自己的具体需求进行修改。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值