支付开发填坑记之支付宝( 二 )

将构造好的请求参数 , 进行处理 , 字典排序 , 拼接字符串 , 签名:
$para_filter = paraFilter($para_temp);$para_sort = argSort($para_filter);$mysign = $this->buildRequestMysign($para_sort);//签名结果与签名方式加入请求提交参数组中$para_sort['sign'] = $mysign;return $para_sort;处理:过滤值为空的数据 , 过滤签名类型和签名 。
function paraFilter($para) { $para_filter = array(); while (list ($key, $val) = each ($para)) { if($key == "sign" || $key == "sign_type" || $val == "")continue; else $para_filter[$key] = $para[$key]; } return $para_filter;}

  1. 字典排序:
/** * 对数组排序 * @param $para 排序前的数组 * return 排序后的数组 */function argSort($para) { ksort($para); reset($para); return $para;}签名:
/** * 生成签名结果 * @param $para_sort 已排序要签名的数组 * return 签名结果字符串 */function buildRequestMysign($para_sort) { //把数组所有元素 , 按照“参数=参数值”的模式用“&”字符拼接成字符串 $prestr = createLinkstring($para_sort); $mysign = ""; switch (strtoupper(trim($this->alipay_config['sign_type']))) { case "MD5" : // MD5直接将密钥拼接在字符串后面再进行MD5加密 。$mysign = md5Sign($prestr, $this->alipay_config['key']); break; case "RSA" : // RSA则是先读取商户的私钥 , 再用该密钥对字符串进行加密 。$mysign = rsaSign($prestr, $this->alipay_config['private_key_path']); break; case "0001" : $mysign = rsaSign($prestr, $this->alipay_config['private_key_path']); break; default : $mysign = ""; }return $mysign;}
  1. 用构造好的参数请求支付宝后台申请新订单:
  2. 注意:请求时 , 必须带上SSL证书 。
$sResult = getHttpResponsePOST($this->alipay_gateway_new, $this->alipay_config['cacert'],$request_data,trim(strtolower($this->alipay_config['input_charset'])));请求函数的实现:
/** * 远程获取数据 , POST模式 * 注意: * 1.使用Crul需要修改服务器中php.ini文件的设置 , 找到php_curl.dll去掉前面的";"就行了 * 2.文件夹中cacert.pem是SSL证书请保证其路径有效 , 目前默认路径是:getcwd().'\cacert.pem' * @param $url 指定URL完整路径地址 * @param $cacert_url 指定当前工作目录绝对路径 * @param $para 请求的数据 * @param $input_charset 编码格式 。默认值:空值 * return 远程输出的数据 */function getHttpResponsePOST($url, $cacert_url, $para, $input_charset = '') { if (trim($input_charset) != '') { $url = $url."_input_charset=".$input_charset; } $curl = curl_init($url); curl_setopt($curl, CURLOPT_SSL_VERIFYPEER, true);//SSL证书认证 curl_setopt($curl, CURLOPT_SSL_VERIFYHOST, 2);//严格认证 curl_setopt($curl, CURLOPT_CAINFO,$cacert_url);//证书地址 curl_setopt($curl, CURLOPT_HEADER, 0 ); // 过滤HTTP头 curl_setopt($curl,CURLOPT_RETURNTRANSFER, 1);// 显示输出结果 curl_setopt($curl,CURLOPT_POST,true); // post传输数据 curl_setopt($curl,CURLOPT_POSTFIELDS,$para);// post传输数据 $responseText = curl_exec($curl); //var_dump( curl_error($curl) );//如果执行curl过程中出现异常 , 可打开此开关 , 以便查看异常内容 curl_close($curl);return $responseText;}处理支付宝返回的数据 , 并获取token 。
//URLDECODE返回的信息$html_text = urldecode($html_text);//解析远程模拟提交后返回的信息$para_html_text = parseResponse($html_text);//获取request_token$request_token = $para_html_text['request_token'];
  1. parseResponse函数的实现:
/** * 解析远程模拟提交后返回的信息 * @param $str_text 要解析的字符串 * @return 解析结果 */function parseResponse($str_text) { //以“&”字符切割字符串 $para_split = explode('&',$str_text); //把切割后的字符串数组变成变量与数值组合的数组 foreach ($para_split as $item) { //获得第一个=字符的位置 $nPos = strpos($item,'='); //获得字符串长度 $nLen = strlen($item); //获得变量名 $key = substr($item,0,$nPos); //获得数值 $value = https://www.isolves.com/it/cxkf/bk/2019-09-19/substr($item,$nPos+1,$nLen-$nPos-1); //放入数组中 $para_text[$key] = $value; }if( ! empty ($para_text['res_data'])) { //解析加密部分字符串 if($this->alipay_config['sign_type'] == '0001') { $para_text['res_data'] = rsaDecrypt($para_text['res_data'], $this->alipay_config['private_key_path']); }//token从res_data中解析出来(也就是说res_data中已经包含token的内容) $doc = new DOMDocument(); $doc->loadXML($para_text['res_data']); $para_text['request_token'] = $doc->getElementsByTagName( "request_token" )->item(0)->nodeValue; }return $para_text;}


推荐阅读