이 글에서는 주로 WeChat js-sdk 결제 인터페이스 클래스의 PHP 버전 사용법을 소개하고, 도움이 필요한 친구의 경우 PHP 버전 WeChat js-sdk 결제 인터페이스 클래스의 정의 및 관련 사용 기술을 분석합니다. 이번 결제 수업 참고하세요
공식문서 기준으로 수정되었습니다! 주로 JS API, Native 패키지 서명 패키지 및 Native 응답 XML 형식 데이터 생성을 구현합니다. 주석에는 각 방법의 목적이 표시됩니다. 패키지 서명이 조금 복잡하기 때문에, 파일을 하나씩 비교하려면 파일을 읽고 로그하는 데 더 많은 시간을 투자해야 합니다! 물론 튜토리얼 클래스를 사용하고 해당 매개변수를 설정하면 패키지 매개변수 등을 올바르게 생성할 수 있습니다.
<?php if (isset($argc) && $argc >= 1 && $argv[0] == __FILE__) { //初始化pay的必要信息 $pay = new WechatPay(array( WechatPay::APPID => 'wx99dabzpiuq83985b8', WechatPay::APPSERCER => 'ac12e7e4abaer63hkoa0cc36a9663fa', WechatPay::PARTNERKEY => 'bae4sfa3562rsfaq23s2045', WechatPay::PARTNERID => '1268969802', WechatPay::PAYSIGNKEY => '9Fqsxb3PK4IVOCEc4yCquy5zecS9LeeMjF2Nn4B4YKoOxPwaQdFwMezKT8oNlBYaWcuT', WechatPay::SIGNTYPE => 'sha1', )); //设置package 必要的参数 jsapi native都通用 $pay->setParams(WechatPay::BANK_TYPE, "WX"); $pay->setParams(WechatPay::BODY, "test"); $pay->setParams(WechatPay::PARTNER, $pay->partnerid); $pay->setParams(WechatPay::OUT_TRADE_NO, commonUtil::createNoncestr()); $pay->setParams(WechatPay::TOTAL_FEE, "1"); $pay->setParams(WechatPay::FEE_TYPE, "1"); $pay->setParams(WechatPay::TIMESTAMP, time()); $pay->setParams(WechatPay::NOTIFY_URL, "http://www.demo.com/notify"); $pay->setParams(WechatPay::SPBILL_CREATE_IP, "127.0.0.1"); $pay->setParams(WechatPay::INPUT_CHARSET, "UTF-8"); //JSAPI的签名json print_r($pay->createJsApiPackage()); //生成native XML print_r($pay->createNativePackage()); //生成native URL print_r($pay->createNativeUrl("9701")); }
JS API에서 생성된 패키지 서명 패키지 매개변수:
{ "appId":"wx9998ff5f4dede5b7", "package":"bank_type=WX&body=test&fee_type=1&input_charset=UTF-8¬ify_url=http%3A%2F%2Fwww.demo.com%2Fnotify&out_trade_no=Vf5qsSwtu0hc2loH&partner=wx9998ff5f4dede5b7&spbill_create_ip=127.0.0.1×tamp=1409295711&total_fee=1&sign=FEE0167BD0D89A88BF6850590EA889B6", "timeStamp":1409295711, "nonceStr":"Vf5qsSwtu0hc2loH", "paySign":"f816264c750923863c370a1739640244b0c2d39c", "signType":"sha1" }
기본 응답 XML 형식:
<xml> <AppId><![CDATA[wx9998ff5f4dede5b7]]></AppId> <Package> <![CDATA[bank_type=WX&body=test&fee_type=1&input_charset=UTF-8¬ify_url=http%3A%2F%2Fwww.demo.com%2Fnotify&out_trade_no=GDl3what4sALDEAd&partner=wx9998ff5f4dede5b7&spbill_create_ip=127.0.0.1×tamp=1409296124&total_fee=1&sign=BF949B85570644B939B369FD44B0C4A9]]> </Package> <TimeStamp>1409296124</TimeStamp> <NonceStr><![CDATA[GDl3what4sALDEAd]]></NonceStr> <RetCode>0</RetCode> <RetErrMsg><![CDATA[ok]]></RetErrMsg> <AppSignature><![CDATA[ca4a2467b817a62c38a9801fcf451f51692027bf]]></AppSignature> <SignMethod><![CDATA[sha1]]></SignMethod> </xml>
기본 URL 링크:
weixin://wxpay/bizpayurl?appid=wx9998ff5f4dede5b7&noncestr=VY7cVA6mtVrc1BVq&productid=9701&sign=43508b65b50e1d e1089be66d55a709469155d73×tamp=1409296323
어쨌든 설정은 해야죠 setParams를 통해 필요한 초기화 매개변수, 상품 가격, 상태 등을!
WechatPay 수업:
<?php class WechatPay { const BANK_TYPE = 'bank_type', BODY = 'body', PARTNER = 'partner', OUT_TRADE_NO = 'out_trade_no', TIMESTAMP = 'timestamp', TOTAL_FEE = 'total_fee', FEE_TYPE = 'fee_type', NOTIFY_URL = 'notify_url', SPBILL_CREATE_IP = 'spbill_create_ip', INPUT_CHARSET = 'input_charset', APPID = 'appid', APPSERCER = 'appsercer', PAYSIGNKEY = 'appkey', PARTNERID = 'partnerid', PARTNERKEY = 'partnerkey', SIGNTYPE = 'signtype'; public $params = array(), $partnerid = ''; static protected $_instance; protected $_appid, $_appkey, $_signtype, $_partnerkey, $_appsercer; static public function getInstance(array $options = array()) { if (empty(self::$_instance)) { self::$_instance = new self ($options); } return self::$_instance; } public function __construct(array $options = array()){ $this->_appid = $options[self::APPID]; $this->_appkey = $options[self::PAYSIGNKEY]; $this->_signtype = $options[self::SIGNTYPE]; $this->_partnerkey = $options[self::PARTNERKEY]; $this->_appsercer = $options[self::APPSERCER]; $this->partnerid = $options[self::APPID]; } public function setParams($param, $paramValue) { $this->params[CommonUtil::trimString($param)] = CommonUtil::trimString($paramValue); } public function getParams($param) { return $this->params[$param]; } protected function createNoncestr( $length = 16 ) { $chars = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789"; $str =""; for ( $i = 0; $i < $length; $i++ ) { $str.= substr($chars, mt_rand(0, strlen($chars)-1), 1); } return $str; } public function checkParams(){ //必要的9个参与签名的参数 if($this->params[self::BANK_TYPE] == null || $this->params[self::BODY] == null || $this->params[self::PARTNER] == null || $this->params[self::OUT_TRADE_NO] == null || $this->params[self::TOTAL_FEE] == null || $this->params[self::FEE_TYPE] == null || $this->params[self::NOTIFY_URL] == null || $this->params[self::SPBILL_CREATE_IP] == null || $this->params[self::INPUT_CHARSET] == null ) { return false; } return true; } /* * 生成package包 * @params 初始化类时用setParams方法定义必要的9个参数 * 排序后格式化url query形式 再md5SignUtil类签名,再给合URL */ protected function getPackageSign(){ try { if (null == $this->_partnerkey || "" == $this->_partnerkey ) { throw new Exception("密钥不能为空!" . "<br>"); } $commonUtil = new CommonUtil(); ksort($this->params); $unSignParaString = $commonUtil->formatUrlQuery($this->params, false); $paraString = $commonUtil->formatUrlQuery($this->params, true); $md5SignUtil = new MD5SignUtil(); return $paraString . "&sign=" . $md5SignUtil->sign($unSignParaString,commonUtil::trimString($this->_partnerkey)); } catch (Exception $e) { echo ($e->getMessage()); } } /* * 生成签名方法 * @params appid appkey package timestamp noncestr 等参数而native事例代码中加上retcode reterrmsg两个参数 */ public function getPaySign($signObj){ foreach ($signObj as $k => $v){ $signParams[strtolower($k)] = $v; } try { if ($this->_appkey == "") { throw new Exception("APPKEY为空!" . "<br>"); } $signParams["appkey"] = $this->_appkey; ksort($signParams, SORT_STRING); $commonUtil = new CommonUtil(); $signString = $commonUtil->formatPayUrlQuery($signParams, false); return sha1($signString); } catch (Exception $e) { echo ($e->getMessage()); } } //JS API 签名 其中nonceStr是作为订单号 灌穿整个支付流程 public function createJsApiPackage(){ try { if($this->checkParams() == false) { throw new Exception("生成package参数缺失!" . "<br>"); } $payObj["appId"] = $this->_appid; $payObj["package"] = $this->getPackageSign(); $payObj["timeStamp"] = $this->getParams(self::TIMESTAMP); $payObj["nonceStr"] = $this->getParams(self::OUT_TRADE_NO); $payObj["paySign"] = $this->getPaySign($payObj); $payObj["signType"] = $this->_signtype; return json_encode($payObj); } catch (Exception $e) { die($e->getMessage()); } } /* * 构建发货状态数组 主要三个参数openid transid orderid */ public function createDeliverPost(Array $params) { $deliver = array(); $deliver['appid'] = $this->_appid; $deliver['openid'] = $params['openid']; $deliver['transid'] = $params['transid']; $deliver['out_trade_no'] = $params['out_trade_no']; $deliver['deliver_timestamp'] = current_time('timestamp'); $deliver['deliver_status'] = 1; $deliver['deliver_msg'] = 'OK'; $deliver['app_signature'] = $this->getPaySign($deliver); $deliver['sign_method'] = 'sha1'; return $deliver; } /* * 生成扫描或者点击原生URL后,响应的XML格式 * @params $retcode $reterrmsg 定义该商品的状态 */ public function createNativePackage($retcode = 0, $reterrmsg = "ok") { try { if ($this->checkParams() == false && $retcode == 0) { //如果是正常的返回, 检查财付通的参数 throw new Exception("生成package参数缺失!" . "<br>"); } $nativeObj["AppId"] = $this->_appid; $nativeObj["Package"] = $this->getPackageSign(); $nativeObj["TimeStamp"] = $this->getParams(self::TIMESTAMP); $nativeObj["NonceStr"] = $this->getParams(self::OUT_TRADE_NO); $nativeObj["RetCode"] = $retcode; $nativeObj["RetErrMsg"] = $reterrmsg; $nativeObj["AppSignature"] = $this->getPaySign($nativeObj); $nativeObj["SignMethod"] = $this->_signtype; $commonUtil = new CommonUtil(); $xml = $commonUtil->arrayToXml($nativeObj); exit($xml); }catch (Exception $e) { echo ($e->getMessage()); } } /* * 生成原生URL 以订单号为参数 这是灌穿整个支付流程 */ public function createNativeUrl($productid) { $commonUtil = new CommonUtil(); $nativeObj["appid"] = $this->_appid; $nativeObj["productid"] = urlencode($productid); $nativeObj["timestamp"] = time(); $nativeObj["nonceStr"] = commonUtil::createNoncestr(); $nativeObj["sign"] = $this->getPaySign($nativeObj); $nativeString = $commonUtil->formatPayUrlQuery($nativeObj, false); return "weixin://wxpay/bizpayurl?".$nativeString; } /* * 取IP地址 */ public function getIp(){ switch(true) { case !empty($_SERVER["HTTP_CLIENT_IP"]): $ip = $_SERVER["HTTP_CLIENT_IP"]; break; case !empty($_SERVER["HTTP_X_FORWARDED_FOR"]): $ip = $_SERVER["HTTP_X_FORWARDED_FOR"]; break; case !empty($_SERVER["REMOTE_ADDR"]): $ip = $_SERVER["REMOTE_ADDR"]; break; default: $ip = "127.0.0.1"; } return $ip; } } class MD5SignUtil { public function sign($content, $key) { try { if (null == $key) { throw new Exception("财付通签名key不能为空!" . "<br>"); } if (null == $content) { throw new Exception("财付通签名内容不能为空" . "<br>"); } $signStr = $content . "&key=" . $key; return strtoupper(md5($signStr)); } catch (Exception $e) { echo ($e->getMessage()); } } public static function verifySignature($content, $sign, $md5Key) { $signStr = $content . "&key=" . $md5Key; $calculateSign = strtolower(md5($signStr)); $tenpaySign = strtolower($sign); return $calculateSign == $tenpaySign; } } class CommonUtil { public function genAllUrl($toURL, $paras) { $allUrl = null; if (null == $toURL) { die("toURL is null"); } if (strripos($toURL,"?") =="") { $allUrl = $toURL . "?" . $paras; } else { $allUrl = $toURL . "&" . $paras; } return $allUrl; } //订单号,可根据实际自定义 static public function createOrderNo() { $nonce = CommonUtil::createNoncestr(4); return strtoupper(date('ymds').substr(microtime(),2,4).$nonce); } //随机字符串 static public function createNoncestr( $length = 16 ) { $chars = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789"; $str =""; for ( $i = 0; $i < $length; $i++ ) { $str.= substr($chars, mt_rand(0, strlen($chars)-1), 1); } return $str; } public function splitParaStr($src, $token) { $resMap = array(); $items = explode($token,$src); foreach ($items as $item){ $paraAndValue = explode("=",$item); if ($paraAndValue != "") { $resMap[$paraAndValue[0]] = $paraAndValue[1]; } } return $resMap; } static function trimString($value) { $ret = null; if (null != $value) { $ret = $value; if (strlen($ret) == 0) { $ret = null; } } return $ret; } public function formatUrlQuery($paraMap, $urlencode) { $buff = ""; ksort($paraMap, SORT_STRING); foreach ($paraMap as $k => $v) { if (null != $v && "null" != $v && "sign" != $k) { if($urlencode) { $v = urlencode($v); } $buff .= $k . "=" . $v . "&"; } } $reqPar = ''; if (strlen($buff) > 0) { $reqPar = substr($buff, 0, strlen($buff)-1); } return $reqPar; } public function formatPayUrlQuery($paraMap, $urlencode) { $buff = ""; ksort($paraMap, SORT_STRING); 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格式 */ public function arrayToXml($arr) { $xml = "<xml>"; foreach ($arr as $key=>$val) { if ($key == 'TimeStamp' || $key == 'RetCode') { $xml.="<".$key.">".$val."</".$key.">"; } else $xml.="<".$key."><![CDATA[".$val."]]></".$key.">"; } $xml .= "</xml>"; return $xml; } }
요약: 위 내용은 이 글의 전체 내용입니다. 모든 분들의 공부에 도움이 되었으면 좋겠습니다.
관련 권장 사항:
php 컬을 사용하여 json 객체를 가져와 배열로 변환하는 방법
PHP는 ID 번호를 정확하게 확인할 수 있는 도구 클래스를 구현합니다
Thinkphp 등록 기능 확인을 위한 SMS 방식 구현
위 내용은 PHP 버전의 WeChat js-sdk 결제 인터페이스 클래스 사용예에 대한 자세한 설명의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

로드 밸런싱은 세션 관리에 영향을 미치지 만 세션 복제, 세션 끈적임 및 중앙 집중식 세션 스토리지로 해결할 수 있습니다. 1. 세션 복제 복사 서버 간의 세션 데이터. 2. 세션 끈은 사용자 요청을 동일한 서버로 안내합니다. 3. 중앙 집중식 세션 스토리지는 Redis와 같은 독립 서버를 사용하여 세션 데이터를 저장하여 데이터 공유를 보장합니다.

SessionLockingIsateChniqueSureDureauser의 SessionLockingSsessionRemainSexclusivetoOneuseratatime.itiscrucialforpreptingdatacorruptionandsecurityBreachesInmulti-userApplications.sessionLockingSogingSompletEdusingserVerver-sidelockingMegynisms, unrasprantlockinj

PHP 세션의 대안에는 쿠키, 토큰 기반 인증, 데이터베이스 기반 세션 및 Redis/Memcached가 포함됩니다. 1. Cookies는 클라이언트에 데이터를 저장하여 세션을 관리합니다. 이는 단순하지만 보안이 적습니다. 2. Token 기반 인증은 토큰을 사용하여 사용자를 확인합니다. 이는 매우 안전하지만 추가 논리가 필요합니다. 3. Database 기반 세션은 데이터베이스에 데이터를 저장하여 확장 성이 좋지만 성능에 영향을 줄 수 있습니다. 4. Redis/Memcached는 분산 캐시를 사용하여 성능 및 확장 성을 향상하지만 추가 일치가 필요합니다.

SessionHijacking은 사용자의 SessionID를 얻음으로써 사용자를 가장하는 공격자를 말합니다. 예방 방법은 다음과 같습니다. 1) HTTPS를 사용한 의사 소통 암호화; 2) SessionID의 출처를 확인; 3) 보안 세션 생성 알고리즘 사용; 4) 정기적으로 SessionID를 업데이트합니다.

이 기사는 PHP에 대해 설명하고, 전체 형식, 웹 개발의 주요 용도, Python 및 Java와의 비교 및 초보자를위한 학습 용이성을 자세히 설명합니다.

PHP는 유효성 검사, 소독 및 보안 데이터베이스 상호 작용을 통해 보안을 보장하면서 $ \ _ post 및 $ \ _를 사용하여 데이터 양식 데이터를 처리합니다.

이 기사는 PHP와 ASP.NET을 비교하여 대규모 웹 응용 프로그램, 성능 차이 및 보안 기능에 대한 적합성에 중점을 둡니다. 둘 다 대규모 프로젝트에서는 실용적이지만 PHP는 오픈 소스 및 플랫폼 독립적이며 ASP.NET,

PHP의 사례 감도는 다양합니다. 함수는 무감각하고 변수와 클래스는 민감합니다. 모범 사례에는 일관된 이름 지정 및 비교를위한 사례 감수 기능 사용이 포함됩니다.


핫 AI 도구

Undresser.AI Undress
사실적인 누드 사진을 만들기 위한 AI 기반 앱

AI Clothes Remover
사진에서 옷을 제거하는 온라인 AI 도구입니다.

Undress AI Tool
무료로 이미지를 벗다

Clothoff.io
AI 옷 제거제

Video Face Swap
완전히 무료인 AI 얼굴 교환 도구를 사용하여 모든 비디오의 얼굴을 쉽게 바꾸세요!

인기 기사

뜨거운 도구

SublimeText3 Linux 새 버전
SublimeText3 Linux 최신 버전

SecList
SecLists는 최고의 보안 테스터의 동반자입니다. 보안 평가 시 자주 사용되는 다양한 유형의 목록을 한 곳에 모아 놓은 것입니다. SecLists는 보안 테스터에게 필요할 수 있는 모든 목록을 편리하게 제공하여 보안 테스트를 더욱 효율적이고 생산적으로 만드는 데 도움이 됩니다. 목록 유형에는 사용자 이름, 비밀번호, URL, 퍼징 페이로드, 민감한 데이터 패턴, 웹 셸 등이 포함됩니다. 테스터는 이 저장소를 새로운 테스트 시스템으로 간단히 가져올 수 있으며 필요한 모든 유형의 목록에 액세스할 수 있습니다.

SublimeText3 중국어 버전
중국어 버전, 사용하기 매우 쉽습니다.

VSCode Windows 64비트 다운로드
Microsoft에서 출시한 강력한 무료 IDE 편집기

PhpStorm 맥 버전
최신(2018.2.1) 전문 PHP 통합 개발 도구
