PHP实现 APP端微信支付功能


Posted in PHP onJune 22, 2018

前面已经写了手机APP支付宝支付,今天再把手机APP微信支付补上,前期的准备工作在这里就不多说了,可以参考微信支付开发文档,一定要仔细阅读开发文档,可以让你少踩点坑;准备工作完成后就是配置参数,调用统一下单接口,支付后异步回调三部曲啦;

1.我封装好的一个支付类文件,多余的东西都去除掉了,并且把配置参数放到了这个支付类中,只需要修改Weixinpayandroid方法内的几个参数就可以直接复制使用:

class Wxpayandroid
{
 //参数配置
 public $config = array(
    'appid' => "", /*微信开放平台上的应用id*/
    'mch_id' => "", /*微信申请成功之后邮件中的商户id*/
    'api_key' => "", /*在微信商户平台上自己设定的api密钥 32位*/
   );
 //服务器异步通知页面路径(必填)
 public $notify_url = '';
 //商户订单号(必填,商户网站订单系统中唯一订单号)
 public $out_trade_no = '';
 //商品描述(必填,不填则为商品名称)
 public $body = '';
 //付款金额(必填)
 public $total_fee = 0;
 //自定义超时(选填,支持dhmc)
 public $time_expire = '';
 private $WxPayHelper;
 public function Weixinpayandroid($total_fee,$tade_no)
 {
  $this->total_fee = intval($total_fee * 100);//订单的金额 1元
  $this->out_trade_no = $tade_no;// date('YmdHis') . substr(time(), - 5) . substr(microtime(), 2, 5) . sprintf('%02d', rand(0, 99));//订单号
  $this->body = 'wxpay';//支付描述信息
  $this->time_expire = date('YmdHis', time() + 86400);//订单支付的过期时间(eg:一天过期)
  $this->notify_url = "http://www.ceshi.com/notifyandroid";//异步通知URL(更改支付状态)
  //数据以JSON的形式返回给APP
  $app_response = $this->doPay(); 
  if (isset($app_response['return_code']) && $app_response['return_code'] == 'FAIL') {
   $errorCode = 100;
   $errorMsg = $app_response['return_msg'];
   $this->echoResult($errorCode, $errorMsg);
  } else {
   $errorCode = 0;
   $errorMsg = 'success';
   $responseData = array(
    'notify_url' => $this->notify_url,
    'app_response' => $app_response,
   );
   $this->echoResult($errorCode, $errorMsg, $responseData);
  }
 }
 //接口输出
 function echoResult($errorCode = 0, $errorMsg = 'success', $responseData = array())
 {
  $arr = array(
   'errorCode' => $errorCode,
   'errorMsg' => $errorMsg,
   'responseData' => $responseData,
  );
   exit(json_encode($arr));  //exit可以正常发送给APP json数据
  // return json_encode($arr); //在TP5中return这个json数据,APP接收到的是null,无法正常吊起微信支付
 }
 function getVerifySign($data, $key)
 {
  $String = $this->formatParameters($data, false);
  //签名步骤二:在string后加入KEY
  $String = $String . "&key=" . $key;
  //签名步骤三:MD5加密
  $String = md5($String);
  //签名步骤四:所有字符转为大写
  $result = strtoupper($String);
  return $result;
 }
 function formatParameters($paraMap, $urlencode)
 {
  $buff = "";
  ksort($paraMap);
  foreach ($paraMap as $k => $v) {
   if($k=="sign"){
    continue;
   }
   if ($urlencode) {
    $v = urlencode($v);
   }
   $buff .= $k . "=" . $v . "&";
  }
  $reqPar;
  if (strlen($buff) > 0) {
   $reqPar = substr($buff, 0, strlen($buff) - 1);
  }
  return $reqPar;
 }
 /**
  * 得到签名
  * @param object $obj
  * @param string $api_key
  * @return string
  */
 function getSign($obj, $api_key)
 {
  foreach ($obj as $k => $v)
  {
   $Parameters[strtolower($k)] = $v;
  }
  //签名步骤一:按字典序排序参数
  ksort($Parameters);
  $String = $this->formatBizQueryParaMap($Parameters, false);
  //签名步骤二:在string后加入KEY
  $String = $String."&key=".$api_key;
  //签名步骤三:MD5加密
  $result = strtoupper(md5($String));
  return $result;
 }
 /**
  * 获取指定长度的随机字符串
  * @param int $length
  * @return Ambigous <NULL, string>
  */
 function getRandChar($length){
  $str = null;
  $strPol = "ABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789abcdefghijklmnopqrstuvwxyz";
  $max = strlen($strPol)-1;
  for($i=0;$i<$length;$i++){
   $str.=$strPol[rand(0,$max)];//rand($min,$max)生成介于min和max两个数之间的一个随机整数
  }
  return $str;
 }
 /**
  * 数组转xml
  * @param array $arr
  * @return string
  */
 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;
 }
 /**
  * 以post方式提交xml到对应的接口url
  *
  * @param string $xml 需要post的xml数据
  * @param string $url url
  * @param bool $useCert 是否需要证书,默认不需要
  * @param int $second url执行超时时间,默认30s
  * @throws WxPayException
  */
 function postXmlCurl($xml, $url, $second=30, $useCert=false, $sslcert_path='', $sslkey_path='')
 {
  $ch = curl_init();
  //设置超时
  curl_setopt($ch, CURLOPT_TIMEOUT, $second);
  curl_setopt($ch,CURLOPT_URL, $url);
  //设置header
  curl_setopt($ch, CURLOPT_HEADER, FALSE);
  //要求结果为字符串且输出到屏幕上
  curl_setopt($ch, CURLOPT_RETURNTRANSFER, TRUE);
  curl_setopt($ch,CURLOPT_SSL_VERIFYPEER,FALSE);
  curl_setopt($ch,CURLOPT_SSL_VERIFYHOST,FALSE);
  if($useCert == true){
   curl_setopt($ch,CURLOPT_SSL_VERIFYPEER,TRUE);
   curl_setopt($ch,CURLOPT_SSL_VERIFYHOST,2);//严格校验
   //设置证书
   //使用证书:cert 与 key 分别属于两个.pem文件
   curl_setopt($ch,CURLOPT_SSLCERTTYPE,'PEM');
   curl_setopt($ch,CURLOPT_SSLCERT, $sslcert_path);
   curl_setopt($ch,CURLOPT_SSLKEYTYPE,'PEM');
   curl_setopt($ch,CURLOPT_SSLKEY, $sslkey_path);
  }
  //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);
   curl_close($ch);
   return false;
  }
 }
 /**
  * 获取当前服务器的IP
  * @return Ambigous <string, unknown>
  */
 function get_client_ip()
 {
  if (isset($_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 = "127.0.0.1";
  }
  return $cip;
 }
 /**
  * 将数组转成uri字符串
  * @param array $paraMap
  * @param bool $urlencode
  * @return string
  */
 function formatBizQueryParaMap($paraMap, $urlencode)
 {
  $buff = "";
  ksort($paraMap);
  foreach ($paraMap as $k => $v)
  {
   if($urlencode)
   {
    $v = urlencode($v);
   }
   $buff .= strtolower($k) . "=" . $v . "&";
  }
  $reqPar;
  if (strlen($buff) > 0)
  {
   $reqPar = substr($buff, 0, strlen($buff)-1);
  }
  return $reqPar;
 }
 /**
  * XML转数组
  * @param unknown $xml
  * @return mixed
  */
 function xmlToArray($xml)
 {
  //将XML转为array
  $array_data = json_decode(json_encode(simplexml_load_string($xml, 'SimpleXMLElement', LIBXML_NOCDATA)), true);
  return $array_data;
 }
 public function chkParam()
 {
  //用户网站订单号
  if (empty($this->out_trade_no)) {
   die('out_trade_no error');
  } 
  //商品描述
  if (empty($this->body)) {
   die('body error');
  }
  if (empty($this->time_expire)){
   die('time_expire error');
  }
  //检测支付金额
  if (empty($this->total_fee) || !is_numeric($this->total_fee)) {
   die('total_fee error');
  }
  //异步通知URL
  if (empty($this->notify_url)) {
   die('notify_url error');
  }
  if (!preg_match("#^http:\/\/#i", $this->notify_url)) {
   $this->notify_url = "http://" . $_SERVER['HTTP_HOST'] . $this->notify_url;
  }
  return true;
 }
 /**
  * 生成支付(返回给APP)
  * @return boolean|mixed
  */
 public function doPay() {
  //检测构造参数
  $this->chkParam();
  return $this->createAppPara();
 }
 /**
  * APP统一下单
  */
 private function createAppPara()
 {
  $url = "https://api.mch.weixin.qq.com/pay/unifiedorder";
  $data["appid"]  = $this->config['appid'];//微信开放平台审核通过的应用APPID
  $data["body"]   = $this->body;//商品或支付单简要描述
  $data["mch_id"]  = $this->config['mch_id'];//商户号
  $data["nonce_str"] = $this->getRandChar(32);//随机字符串
  $data["notify_url"] = $this->notify_url;//通知地址
  $data["out_trade_no"] = $this->out_trade_no;//商户订单号
  $data["spbill_create_ip"] = $this->get_client_ip();//终端IP
  $data["total_fee"]  = $this->total_fee;//总金额
  $data["time_expire"]  = $this->time_expire;//交易结束时间
  $data["trade_type"]  = "APP";//交易类型
  $data["sign"]    = $this->getSign($data, $this->config['api_key']);//签名
  $xml  = $this->arrayToXml($data);
  $response = $this->postXmlCurl($xml, $url);
  //将微信返回的结果xml转成数组
  $responseArr = $this->xmlToArray($response);
  if(isset($responseArr["return_code"]) && $responseArr["return_code"]=='SUCCESS'){
   return $this->getOrder($responseArr['prepay_id']);
  }
  return $responseArr;
 }
 /**
  * 执行第二次签名,才能返回给客户端使用
  * @param int $prepayId:预支付交易会话标识
  * @return array
  */
 public function getOrder($prepayId)
 {
  $data["appid"]  = $this->config['appid'];
  $data["noncestr"] = $this->getRandChar(32);
  $data["package"] = "Sign=WXPay";
  $data["partnerid"] = $this->config['mch_id'];
  $data["prepayid"] = $prepayId;
  $data["timestamp"] = time();
  $data["sign"]  = $this->getSign($data, $this->config['api_key']);
  $data["packagestr"] = "Sign=WXPay";
  return $data;
 }
 /**
  * 异步通知信息验证
  * @return boolean|mixed
  */
 public function verifyNotify()
 {
  $xml = isset($GLOBALS['HTTP_RAW_POST_DATA']) ? $GLOBALS['HTTP_RAW_POST_DATA'] : ''; 
  if(!$xml){
   return false;
  }
  $wx_back = $this->xmlToArray($xml);
  if(empty($wx_back)){
   return false;
  }
  $checkSign = $this->getVerifySign($wx_back, $this->config['api_key']); 
  if($checkSign=$wx_back['sign']){
   return $wx_back;
  }else{
   return false;
  } 
 }
}

2.创建控制器定义统一下单接口和支付后的异步回调接口:

//异步通知接口
 public function notifyandroid()
 {
  $wxpayandroid = new \Wxpayandroid;  //实例化微信支付类
  $verify_result = $wxpayandroid->verifyNotify();
  if ($verify_result['return_code']=='SUCCESS' && $verify_result['result_code']=='SUCCESS') {
  //商户订单号
  $out_trade_no = $verify_result['out_trade_no'];
  //交易号
  $trade_no  = $verify_result['transaction_id'];
  //交易状态
  $trade_status = $verify_result['result_code'];
  //支付金额
  $total_fee = $verify_result['total_fee']/100;
  //支付过期时间
  $pay_date  = $verify_result['time_end'];
  $order = new Order();
  $ret = $order->getOrderN2($out_trade_no); //获取订单信息
  $total_amount=$ret['money'];
  if ($total_amount==$total_fee) {
   // 验证成功 修改数据库的订单状态等 $result['out_trade_no']为订单号
   //此处写自己的逻辑代码
  }
   exit('<xml><return_code><![CDATA[SUCCESS]]></return_code><return_msg><![CDATA[OK]]></return_msg></xml>');
  }else{
   exit('<xml><return_code><![CDATA[FAIL]]></return_code><return_msg><![CDATA[ERROR]]></return_msg></xml>');
  }
 }
 
 //调用统一下单接口生成预支付订单并把数据返回给APP
 public function wxpayandroid(Request $request)
 {
  $param = $request->param(); //接收值
 
  $tade_no = $param['orderCode'];
  $order = new Order(); //实例化订单
  $ret = $order->getOrderN2($tade_no); //查询订单信息
  $total_fee = $ret['money']; //订单总金额
  
  $wxpayandroid = new \Wxpayandroid;  //实例化微信支付类
  $res = $wxpayandroid->Weixinpayandroid($total_fee,$tade_no); //调用weixinpay方法
  
 }

封装一个支付类文件,并把配置参数放到支付类内,再定义控制器创建两个方法,这样两步就可以把手机APP微信支付搞定啦。

总结

以上所述是小编给大家介绍的PHP实现 APP端微信支付功能,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对三水点靠木网站的支持!

PHP 相关文章推荐
多文件上传的例子
Oct 09 PHP
PHP foreach循环使用详解与实例代码
May 08 PHP
PHP简洁函数小结
Aug 12 PHP
关于php程序报date()警告的处理(date_default_timezone_set)
Oct 22 PHP
php缩小png图片不损失透明色的解决方法
Dec 25 PHP
PHP实现的简单mock json脚本分享
Feb 10 PHP
PHP实现对png图像进行缩放的方法(支持透明背景)
Jul 15 PHP
PHP中addcslashes与stripcslashes函数用法分析
Jan 07 PHP
YII CLinkPager分页类扩展增加显示共多少页
Jan 29 PHP
php禁用函数设置及查看方法详解
Jul 25 PHP
php实现的中文分词类完整实例
Feb 06 PHP
Thinkphp框架使用list_to_tree 实现无限级分类列出所有节点示例
Apr 04 PHP
thinkPHP实现基于ajax的评论回复功能
Jun 22 #PHP
php strftime函数的详细用法
Jun 21 #PHP
PHP获取本周所有日期或者最近七天所有日期的方法
Jun 20 #PHP
ThinkPHP5.0 图片上传生成缩略图实例代码说明
Jun 20 #PHP
thinkPHP框架实现的短信接口验证码功能示例
Jun 20 #PHP
thinkPHP3.2.2框架行为扩展及demo示例
Jun 19 #PHP
Laravel框架实现的记录SQL日志功能示例
Jun 19 #PHP
You might like
PHP 单引号与双引号的区别
2009/11/24 PHP
getJSON跨域SyntaxError问题分析
2014/08/07 PHP
thinkPHP实现递归循环栏目并按照树形结构无限极输出的方法
2016/05/19 PHP
PHP网站自动化配置的实现方法(必看)
2017/05/27 PHP
PHP实现腾讯与百度坐标转换
2017/08/05 PHP
laravel 实现登陆后返回登陆前的页面方法
2019/10/03 PHP
Prototype Function对象 学习
2009/07/12 Javascript
JQuery给元素绑定click事件多次执行的解决方法
2014/05/29 Javascript
基于jquery实现表格内容筛选功能实例解析
2016/05/09 Javascript
js关于getImageData跨域问题的解决方法
2016/10/14 Javascript
jquery.Callbacks的实现详解
2016/11/30 Javascript
JavaScript事件发布/订阅模式原理与用法分析
2018/08/21 Javascript
微信小程序 如何保持登录状态
2019/08/16 Javascript
JS轮播图的实现方法
2020/08/24 Javascript
javascript使用canvas实现饼状图效果
2020/09/08 Javascript
JavaScript缓动动画函数的封装方法
2020/11/25 Javascript
Python使用logging结合decorator模式实现优化日志输出的方法
2016/04/16 Python
Python 自动刷博客浏览量实例代码
2017/06/14 Python
Python 编码规范(Google Python Style Guide)
2018/05/05 Python
详解如何为eclipse安装合适版本的python插件pydev
2018/11/04 Python
Python 限制线程的最大数量的方法(Semaphore)
2019/02/22 Python
Python使用贪婪算法解决问题
2019/10/22 Python
pytorch中的卷积和池化计算方式详解
2020/01/03 Python
如何理解Python中包的引入
2020/05/29 Python
Python+Selenium随机生成手机验证码并检查页面上是否弹出重复手机号码提示框
2020/09/21 Python
pycharm 实现光标快速移动到括号外或行尾的操作
2021/02/05 Python
css3 clip实现圆环进度条的示例代码
2018/02/07 HTML / CSS
基于canvas的骨骼动画的示例代码
2018/06/12 HTML / CSS
捷克移动配件网上商店:ProMobily.cz
2019/03/15 全球购物
介绍一下Make? 为什么使用make
2016/07/31 面试题
销售人员自我评价怎么写
2013/09/19 职场文书
学校运动会开幕演讲稿
2014/01/04 职场文书
办理生育手续介绍信
2014/01/14 职场文书
教师节倡议书2015
2015/04/27 职场文书
红色故事汇观后感
2015/06/18 职场文书
女方家长婚礼答谢词
2015/09/29 职场文书