Heim Backend-Entwicklung PHP-Tutorial WeChat-Zahlungsmethode über die PHP-APP

WeChat-Zahlungsmethode über die PHP-APP

Jul 04, 2018 pm 03:54 PM

Dieser Artikel stellt hauptsächlich die WeChat-Zahlungsmethode in der PHP-APP vor. Er hat einen gewissen Referenzwert. Jetzt kann ich ihn mit allen teilen, die ihn benötigen.

Ich habe bereits über die mobile APP geschrieben Alipay-Zahlung, heute werde ich auf die vorbereitenden Arbeiten für die mobile App nicht eingehen. Lesen Sie dazu unbedingt das Entwicklungsdokument Fallstricke. Nachdem die Vorbereitungsarbeiten abgeschlossen sind, sind es die Parameter, der Aufruf der einheitlichen Bestellschnittstelle und der asynchrone Rückruf nach der Zahlung Um die Konfigurationsparameter in diese Zahlungsklasse einzufügen, müssen Sie nur einige Parameter in der Weixinpayandroid-Methode ändern und können sie direkt kopieren und verwenden:

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=&#39;&#39;, $sslkey_path=&#39;&#39;)
    {
        $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,&#39;PEM&#39;);
            curl_setopt($ch,CURLOPT_SSLCERT, $sslcert_path);
            curl_setopt($ch,CURLOPT_SSLKEYTYPE,&#39;PEM&#39;);
            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[&#39;REMOTE_ADDR&#39;])) {
            $cip = $_SERVER[&#39;REMOTE_ADDR&#39;];
        } 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, &#39;SimpleXMLElement&#39;, LIBXML_NOCDATA)), true);
        return $array_data;
    }
    public function chkParam() 
    {
        //用户网站订单号
        if (empty($this->out_trade_no)) {
            die(&#39;out_trade_no error&#39;);
        }   
        //商品描述
        if (empty($this->body)) {
            die(&#39;body error&#39;);
        }
        if (empty($this->time_expire)){
            die(&#39;time_expire error&#39;);
        }
        //检测支付金额
        if (empty($this->total_fee) || !is_numeric($this->total_fee)) {
            die(&#39;total_fee error&#39;);
        }
        //异步通知URL
        if (empty($this->notify_url)) {
            die(&#39;notify_url error&#39;);
        }
        if (!preg_match("#^http:\/\/#i", $this->notify_url)) {
            $this->notify_url = "http://" . $_SERVER[&#39;HTTP_HOST&#39;] . $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[&#39;appid&#39;];//微信开放平台审核通过的应用APPID
        $data["body"]         = $this->body;//商品或支付单简要描述
        $data["mch_id"]       = $this->config[&#39;mch_id&#39;];//商户号
        $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[&#39;api_key&#39;]);//签名
        $xml        = $this->arrayToXml($data);
        $response   = $this->postXmlCurl($xml, $url);
        //将微信返回的结果xml转成数组
        $responseArr = $this->xmlToArray($response);
        if(isset($responseArr["return_code"]) && $responseArr["return_code"]==&#39;SUCCESS&#39;){
            return  $this->getOrder($responseArr[&#39;prepay_id&#39;]);
        }
        return $responseArr;
    }
    
    /**
     * 执行第二次签名,才能返回给客户端使用
     * @param int $prepayId:预支付交易会话标识
     * @return array
     */
    public function getOrder($prepayId)
    {
        $data["appid"]      = $this->config[&#39;appid&#39;];
        $data["noncestr"]   = $this->getRandChar(32);
        $data["package"]    = "Sign=WXPay";
        $data["partnerid"]  = $this->config[&#39;mch_id&#39;];
        $data["prepayid"]   = $prepayId;
        $data["timestamp"]  = time();
        $data["sign"]       = $this->getSign($data, $this->config[&#39;api_key&#39;]);
        $data["packagestr"] = "Sign=WXPay";
        return $data;
    }
    
    /**
     * 异步通知信息验证
     * @return boolean|mixed
     */
    public function verifyNotify()
    {
        $xml = isset($GLOBALS[&#39;HTTP_RAW_POST_DATA&#39;]) ? $GLOBALS[&#39;HTTP_RAW_POST_DATA&#39;] : &#39;&#39;;  
        if(!$xml){
            return false;
        }
        $wx_back = $this->xmlToArray($xml);
        if(empty($wx_back)){
            return false;
        }
        $checkSign = $this->getVerifySign($wx_back, $this->config[&#39;api_key&#39;]);     
        if($checkSign=$wx_back[&#39;sign&#39;]){
            return $wx_back;
        }else{
            return false;
        }   
    }
}  
Nach dem Login kopieren

2. Erstellen Sie einen zu definierenden Controller eine einheitliche Bestellschnittstelle und eine asynchrone Rückrufschnittstelle nach der Zahlung:

vendor(&#39;Wxpayandroid&#39;);//引入支付类
class Wxpayandroid extends Controller 
{
//异步通知接口
    public function notifyandroid()
    {
        $wxpayandroid = new \Wxpayandroid;     //实例化微信支付类
        $verify_result = $wxpayandroid->verifyNotify(); 
        if ($verify_result[&#39;return_code&#39;]==&#39;SUCCESS&#39; && $verify_result[&#39;result_code&#39;]==&#39;SUCCESS&#39;) {
          //商户订单号
          $out_trade_no = $verify_result[&#39;out_trade_no&#39;];
          //交易号
          $trade_no     = $verify_result[&#39;transaction_id&#39;];
          //交易状态
          $trade_status = $verify_result[&#39;result_code&#39;];
          //支付金额
          $total_fee    = $verify_result[&#39;total_fee&#39;]/100;
          //支付过期时间
          $pay_date     = $verify_result[&#39;time_end&#39;];
          $order = new Order();
          $ret = $order->getOrderN2($out_trade_no); //获取订单信息
          $total_amount=$ret[&#39;money&#39;];
          if ($total_amount==$total_fee) {
              // 验证成功 修改数据库的订单状态等 $result[&#39;out_trade_no&#39;]为订单号
              //此处写自己的逻辑代码
          }
            exit(&#39;<xml><return_code><![CDATA[SUCCESS]]></return_code><return_msg><![CDATA[OK]]></return_msg></xml>&#39;);
        }else{
            exit(&#39;<xml><return_code><![CDATA[FAIL]]></return_code><return_msg><![CDATA[ERROR]]></return_msg></xml>&#39;);
        }
    }

    //调用统一下单接口生成预支付订单并把数据返回给APP
    public function wxpayandroid(Request $request)
    {
        $param = $request->param(); //接收值

        $tade_no = $param[&#39;orderCode&#39;];
        $order = new Order();   //实例化订单
        $ret = $order->getOrderN2($tade_no);  //查询订单信息
        $total_fee = $ret[&#39;money&#39;]; //订单总金额
        
        $wxpayandroid = new \Wxpayandroid;     //实例化微信支付类
        $res = $wxpayandroid->Weixinpayandroid($total_fee,$tade_no);    //调用weixinpay方法
    }
}
Nach dem Login kopieren

Kapseln Sie eine Zahlungsklassendatei, fügen Sie die Konfigurationsparameter in die Zahlungsklasse ein und definieren Sie dann die Controller, um zwei Methoden zu erstellen, sodass Sie in zwei Schritten die mobile APP WeChat-Zahlung durchführen können.

Das Obige ist der gesamte Inhalt dieses Artikels. Ich hoffe, dass er für das Studium aller hilfreich ist. Weitere verwandte Inhalte finden Sie auf der chinesischen PHP-Website.

Verwandte Empfehlungen:

So importieren Sie Excel-Daten in PHPExcel


Einführung in die Verwendung von OpenSSL als Ersatz für mcrypt in PHP7. 1

Das obige ist der detaillierte Inhalt vonWeChat-Zahlungsmethode über die PHP-APP. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Erklärung dieser Website
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn

Heiße KI -Werkzeuge

Undresser.AI Undress

Undresser.AI Undress

KI-gestützte App zum Erstellen realistischer Aktfotos

AI Clothes Remover

AI Clothes Remover

Online-KI-Tool zum Entfernen von Kleidung aus Fotos.

Undress AI Tool

Undress AI Tool

Ausziehbilder kostenlos

Clothoff.io

Clothoff.io

KI-Kleiderentferner

AI Hentai Generator

AI Hentai Generator

Erstellen Sie kostenlos Ai Hentai.

Heißer Artikel

R.E.P.O. Energiekristalle erklärten und was sie tun (gelber Kristall)
3 Wochen vor By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O. Beste grafische Einstellungen
3 Wochen vor By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O. So reparieren Sie Audio, wenn Sie niemanden hören können
3 Wochen vor By 尊渡假赌尊渡假赌尊渡假赌
WWE 2K25: Wie man alles in Myrise freischaltet
4 Wochen vor By 尊渡假赌尊渡假赌尊渡假赌

Heiße Werkzeuge

Notepad++7.3.1

Notepad++7.3.1

Einfach zu bedienender und kostenloser Code-Editor

SublimeText3 chinesische Version

SublimeText3 chinesische Version

Chinesische Version, sehr einfach zu bedienen

Senden Sie Studio 13.0.1

Senden Sie Studio 13.0.1

Leistungsstarke integrierte PHP-Entwicklungsumgebung

Dreamweaver CS6

Dreamweaver CS6

Visuelle Webentwicklungstools

SublimeText3 Mac-Version

SublimeText3 Mac-Version

Codebearbeitungssoftware auf Gottesniveau (SublimeText3)

Arbeiten mit Flash -Sitzungsdaten in Laravel Arbeiten mit Flash -Sitzungsdaten in Laravel Mar 12, 2025 pm 05:08 PM

Laravel vereinfacht die Behandlung von temporären Sitzungsdaten mithilfe seiner intuitiven Flash -Methoden. Dies ist perfekt zum Anzeigen von kurzen Nachrichten, Warnungen oder Benachrichtigungen in Ihrer Anwendung. Die Daten bestehen nur für die nachfolgende Anfrage standardmäßig: $ Anfrage-

Curl in PHP: So verwenden Sie die PHP -Curl -Erweiterung in REST -APIs Curl in PHP: So verwenden Sie die PHP -Curl -Erweiterung in REST -APIs Mar 14, 2025 am 11:42 AM

Die PHP Client -URL -Erweiterung (CURL) ist ein leistungsstarkes Tool für Entwickler, das eine nahtlose Interaktion mit Remote -Servern und REST -APIs ermöglicht. Durch die Nutzung von Libcurl, einer angesehenen Bibliothek mit Multi-Protokoll-Dateien, erleichtert PHP Curl effiziente Execu

Vereinfachte HTTP -Reaktion verspottet in Laravel -Tests Vereinfachte HTTP -Reaktion verspottet in Laravel -Tests Mar 12, 2025 pm 05:09 PM

Laravel bietet eine kurze HTTP -Antwortsimulationssyntax und vereinfache HTTP -Interaktionstests. Dieser Ansatz reduziert die Code -Redundanz erheblich, während Ihre Testsimulation intuitiver wird. Die grundlegende Implementierung bietet eine Vielzahl von Verknüpfungen zum Antworttyp: Verwenden Sie Illuminate \ Support \ facades \ http; Http :: fake ([ 'Google.com' => 'Hallo Welt',, 'github.com' => ['foo' => 'bar'], 'Forge.laravel.com' =>

12 Beste PHP -Chat -Skripte auf Codecanyon 12 Beste PHP -Chat -Skripte auf Codecanyon Mar 13, 2025 pm 12:08 PM

Möchten Sie den dringlichsten Problemen Ihrer Kunden in Echtzeit und Sofortlösungen anbieten? Mit Live-Chat können Sie Echtzeitgespräche mit Kunden führen und ihre Probleme sofort lösen. Sie ermöglichen es Ihnen, Ihrem Brauch einen schnelleren Service zu bieten

Erklären Sie das Konzept der späten statischen Bindung in PHP. Erklären Sie das Konzept der späten statischen Bindung in PHP. Mar 21, 2025 pm 01:33 PM

In Artikel wird die in PHP 5.3 eingeführte LSB -Bindung (LSB) erörtert, die die Laufzeitauflösung der statischen Methode ermöglicht, um eine flexiblere Vererbung zu erfordern. Die praktischen Anwendungen und potenziellen Perfo von LSB

Rahmensicherheitsmerkmale: Schutz vor Schwachstellen. Rahmensicherheitsmerkmale: Schutz vor Schwachstellen. Mar 28, 2025 pm 05:11 PM

In Artikel werden wichtige Sicherheitsfunktionen in Frameworks erörtert, um vor Schwachstellen zu schützen, einschließlich Eingabevalidierung, Authentifizierung und regelmäßigen Aktualisierungen.

Erklären Sie JSON Web Tokens (JWT) und ihren Anwendungsfall in PHP -APIs. Erklären Sie JSON Web Tokens (JWT) und ihren Anwendungsfall in PHP -APIs. Apr 05, 2025 am 12:04 AM

JWT ist ein offener Standard, der auf JSON basiert und zur sicheren Übertragung von Informationen zwischen Parteien verwendet wird, hauptsächlich für die Identitätsauthentifizierung und den Informationsaustausch. 1. JWT besteht aus drei Teilen: Header, Nutzlast und Signatur. 2. Das Arbeitsprinzip von JWT enthält drei Schritte: Generierung von JWT, Überprüfung von JWT und Parsingnayload. 3. Bei Verwendung von JWT zur Authentifizierung in PHP kann JWT generiert und überprüft werden, und die Funktionen und Berechtigungsinformationen der Benutzer können in die erweiterte Verwendung aufgenommen werden. 4. Häufige Fehler sind Signaturüberprüfungsfehler, Token -Ablauf und übergroße Nutzlast. Zu Debugging -Fähigkeiten gehört die Verwendung von Debugging -Tools und Protokollierung. 5. Leistungsoptimierung und Best Practices umfassen die Verwendung geeigneter Signaturalgorithmen, das Einstellen von Gültigkeitsperioden angemessen.

See all articles