PHP微信支付开发实例详细教程​(附代码)

HP微信支付开发过程,分享给大家,供大家参考,具体内容如下

1.开发环境
Thinkphp 3.2.3
微信:服务号,已认证
开发域名:http://test.paywechat.com (自定义的域名,外网不可访问)

2.需要相关文件和权限
微信支付需申请开通
微信公众平台开发者文档:http://mp.weixin.qq.com/wiki/home/index.html
微信支付开发者文档:https://pay.weixin.qq.com/wiki/doc/api/index.html
微信支付SDK下载地址:https://pay.weixin.qq.com/wiki/doc/api/jsapi.php?chapter=11_1

3.开发
下载好微信支付PHP版本的SDK,文件目录为下图:

PHP微信支付开发实例详细教程​(附代码)

PHP微信支付开发实例详细教程​(附代码)

把微信支付SDK的Cert和Lib目录放入Thinkphp,目录为

PHP微信支付开发实例详细教程​(附代码)

现在介绍微信支付授权目录问题,首先是微信支付开发配置里面的支付授权目录填写,

然后填写JS接口安全域。

PHP微信支付开发实例详细教程​(附代码)

最后设置网页授权

PHP微信支付开发实例详细教程​(附代码)

PHP微信支付开发实例详细教程​(附代码)

这些设置完,基本完成一半,注意设置的目录和我thinkphp里面的目录。

PHP微信支付开发实例详细教程​(附代码)

4.微信支付配置

PHP微信支付开发实例详细教程​(附代码)

把相关配置填写正确。

 


* @date 2015年8月4日
* @todu
*/
class ParentController extends Controller { 
protected $options = array (
'token' => '', // 填写你设定的key
'encodingaeskey' => '', // 填写加密用的EncodingAESKey
'appid' => '', // 填写高级调用功能的app id
'appsecret' => '', // 填写高级调用功能的密钥
'debug' => false,
'logcallback' => ''
); 
public $errCode = 40001; 
public $errMsg = "no access"; 
/**
* 获取access_token
* @return mixed|boolean|unknown
*/
public function getToken(){
$cache_token = S('exp_wechat_pay_token');
if(!empty($cache_token)){
return $cache_token;
}
$url = 'https://api.weixin.qq.com/cgi-bin/token?grant_type=client_credential&appid=%s&secret=%s';
$url = sprintf($url,$this->options['appid'],$this->options['appsecret']); 
$result = $this->http_get($url);
$result = json_decode($result,true); 
if(empty($result)){
return false;
} 
S('exp_wechat_pay_token',$result['access_token'],array('type'=>'file','expire'=>3600));
return $result['access_token'];
}
/**
* 发送客服消息
* @param array $data 消息结构{"touser":"OPENID","msgtype":"news","news":{...}}
*/
public function sendCustomMessage($data){
$token = $this->getToken();
if (empty($token)) return false; 
$url = 'https://api.weixin.qq.com/cgi-bin/message/custom/send?access_token=%s';
$url = sprintf($url,$token);
$result = $this->http_post($url,self::json_encode($data));
if ($result)
{
$json = json_decode($result,true);
if (!$json || !empty($json['errcode'])) {
$this->errCode = $json['errcode'];
$this->errMsg = $json['errmsg'];
return false;
}
return $json;
}
return false;
}
/**
* 发送模板消息
* @param unknown $data
* @return boolean|unknown
*/
public function sendTemplateMessage($data){
$token = $this->getToken();
if (empty($token)) return false;
$url = "https://api.weixin.qq.com/cgi-bin/message/template/send?access_token=%s";
$url = sprintf($url,$token);
$result = $this->http_post($url,self::json_encode($data));
if ($result)
{
$json = json_decode($result,true);
if (!$json || !empty($json['errcode'])) {
$this->errCode = $json['errcode'];
$this->errMsg = $json['errmsg'];
return false;
}
return $json;
}
return false;
}
public function getFileCache($name){
return S($name);
}
/**
* 微信api不支持中文转义的json结构
* @param array $arr
*/
static function json_encode($arr) {
$parts = array ();
$is_list = false;
//Find out if the given array is a numerical array
$keys = array_keys ( $arr );
$max_length = count ( $arr ) - 1;
if (($keys [0] === 0) && ($keys [$max_length] === $max_length )) { //See if the first key is 0 and last key is length - 1
$is_list = true;
for($i = 0; $i < count ( $keys ); $i ++) { //See if each key correspondes to its position
if ($i != $keys [$i]) { //A key fails at position check.
$is_list = false; //It is an associative array.
break;
}
}
}
foreach ( $arr as $key => $value ) {
if (is_array ( $value )) { //Custom handling for arrays
if ($is_list)
$parts [] = self::json_encode ( $value ); /* :RECURSION: */
else
$parts [] = '"' . $key . '":' . self::json_encode ( $value ); /* :RECURSION: */
} else {
$str = '';
if (! $is_list)
$str = '"' . $key . '":';
//Custom handling for multiple data types
if (!is_string ( $value ) && is_numeric ( $value ) && $value<2000000000)
$str .= $value; //Numbers
elseif ($value === false)
$str .= 'false'; //The booleans
elseif ($value === true)
$str .= 'true';
else
$str .= '"' . addslashes ( $value ) . '"'; //All other things
// :TODO: Is there any more datatype we should be in the lookout for? (Object?)
$parts [] = $str;
}
}
$json = implode ( ',', $parts );
if ($is_list)
return '[' . $json . ']'; //Return numerical JSON
return '{' . $json . '}'; //Return associative JSON
}
/**
+----------------------------------------------------------
* 生成随机字符串
+----------------------------------------------------------
* @param int $length 要生成的随机字符串长度
* @param string $type 随机码类型:0,数字+大小写字母;1,数字;2,小写字母;3,大写字母;4,特殊字符;-1,数字+大小写字母+特殊字符
+----------------------------------------------------------
* @return string
+----------------------------------------------------------
*/
static public function randCode($length = 5, $type = 2){
$arr = array(1 => "0123456789", 2 => "abcdefghijklmnopqrstuvwxyz", 3 => "ABCDEFGHIJKLMNOPQRSTUVWXYZ", 4 => "~@#$%^&*(){}[]|");
if ($type == 0) {
array_pop($arr);
$string = implode("", $arr);
} elseif ($type == "-1") {
$string = implode("", $arr);
} else {
$string = $arr[$type];
}
$count = strlen($string) - 1;
$code = '';
for ($i = 0; $i < $length; $i++) {
$code .= $string[rand(0, $count)];
}
return $code;
} 
/**
* GET 请求
* @param string $url
*/
private function http_get($url){
$oCurl = curl_init();
if(stripos($url,"https://")!==FALSE){
curl_setopt($oCurl, CURLOPT_SSL_VERIFYPEER, FALSE);
curl_setopt($oCurl, CURLOPT_SSL_VERIFYHOST, FALSE);
curl_setopt($oCurl, CURLOPT_SSLVERSION, 1); //CURL_SSLVERSION_TLSv1
}
curl_setopt($oCurl, CURLOPT_URL, $url);
curl_setopt($oCurl, CURLOPT_RETURNTRANSFER, 1 );
$sContent = curl_exec($oCurl);
$aStatus = curl_getinfo($oCurl);
curl_close($oCurl);
if(intval($aStatus["http_code"])==200){
return $sContent;
}else{
return false;
}
}
/**
* POST 请求
* @param string $url
* @param array $param
* @param boolean $post_file 是否文件上传
* @return string content
*/
private function http_post($url,$param,$post_file=false){
$oCurl = curl_init();
if(stripos($url,"https://")!==FALSE){
curl_setopt($oCurl, CURLOPT_SSL_VERIFYPEER, FALSE);
curl_setopt($oCurl, CURLOPT_SSL_VERIFYHOST, false);
curl_setopt($oCurl, CURLOPT_SSLVERSION, 1); //CURL_SSLVERSION_TLSv1
}
if (is_string($param) || $post_file) {
$strPOST = $param;
} else {
$aPOST = array();
foreach($param as $key=>$val){
$aPOST[] = $key."=".urlencode($val);
}
$strPOST = join("&", $aPOST);
}
curl_setopt($oCurl, CURLOPT_URL, $url);
curl_setopt($oCurl, CURLOPT_RETURNTRANSFER, 1 );
curl_setopt($oCurl, CURLOPT_POST,true);
curl_setopt($oCurl, CURLOPT_POSTFIELDS,$strPOST);
$sContent = curl_exec($oCurl);
$aStatus = curl_getinfo($oCurl);
curl_close($oCurl);
if(intval($aStatus["http_code"])==200){
return $sContent;
}else{
return false;
}
}
}

* @date 2015年8月4日
* @todu
*/
class TestController extends ParentController {
private $_order_body = 'xxx';
private $_order_goods_tag = 'xxx';
public function __construct(){
parent::__construct();
require_once ROOT_PATH."Api/lib/WxPay.Api.php";
require_once ROOT_PATH."Api/lib/WxPay.JsApiPay.php";
}
public function index(){
//①、获取用户openid
$tools = new JsApiPay();
$openId = $tools->GetOpenid(); 
//②、统一下单
$input = new WxPayUnifiedOrder(); 
//商品描述
$input->SetBody($this->_order_body);
//附加数据,可以添加自己需要的数据,微信回异步回调时会附加这个数据
$input->SetAttach('xxx');
//商户订单号
$out_trade_no = WxPayConfig::MCHID.date("YmdHis");
$input->SetOut_trade_no($out_trade_no);
//总金额,订单总金额,只能为整数,单位为分 
$input->SetTotal_fee(1);
//交易起始时间
$input->SetTime_start(date("YmdHis"));
//交易结束时间
$input->SetTime_expire(date("YmdHis", time() + 600));
//商品标记
$input->SetGoods_tag($this->_order_goods_tag);
//通知地址,接收微信支付异步通知回调地址 SITE_URL=http://test.paywechat.com/Charge
$notify_url = SITE_URL.'/index.php/Test/notify.html';
$input->SetNotify_url($notify_url);
//交易类型
$input->SetTrade_type("JSAPI");
$input->SetOpenid($openId);
$order = WxPayApi::unifiedOrder($input);
$jsApiParameters = $tools->GetJsApiParameters($order);
//获取共享收货地址js函数参数
$editAddress = $tools->GetEditAddressParameters();
$this->assign('openId',$openId);
$this->assign('jsApiParameters',$jsApiParameters);
$this->assign('editAddress',$editAddress);
$this->display(); 
}
/**
* 异步通知回调方法
*/
public function notify(){
require_once ROOT_PATH."Api/lib/notify.php";
$notify = new PayNotifyCallBack();
$notify->Handle(false);
//这里的IsSuccess是我自定义的一个方法,后面我会贴出这个文件的代码,供参考。
$is_success = $notify->IsSuccess(); 
$bdata = $is_success['data']; 
//支付成功
if($is_success['code'] == 1){ 
$news = array(
'touser' => $bdata['openid'],
'msgtype' => 'news',
'news' => array (
'articles'=> array (
array(
'title' => '订单支付成功',
'description' => "支付金额:{$bdata['total_fee']}n".
"微信订单号:{$bdata['transaction_id']}n"
'picurl' => '',
'url' => ''
)
)
)
);
//发送微信支付通知
$this->sendCustomMessage($news); 
}else{//支付失败
}
}
/**
* 支付成功页面
* 不可靠的回调
*/
public function ajax_PaySuccess(){
//订单号
$out_trade_no = I('post.out_trade_no');
//支付金额
$total_fee = I('post.total_fee');
/*相关逻辑处理*/
}

接着是html模板的源码




 
微信支付样例-支付




该笔订单支付金额为1分

notify.php文件代码,这里有在官方文件里新添加的一个自定义方法。

0,'data'=>'');
//查询订单
public function Queryorder($transaction_id)
{
$input = new WxPayOrderQuery();
$input->SetTransaction_id($transaction_id);
$result = WxPayApi::orderQuery($input);
Log::DEBUG("query:" . json_encode($result));
if(array_key_exists("return_code", $result)
&& array_key_exists("result_code", $result)
&& $result["return_code"] == "SUCCESS"
&& $result["result_code"] == "SUCCESS")
{
return true;
}
$this->para['code'] = 0;
$this->para['data'] = '';
return false;
}
//重写回调处理函数
public function NotifyProcess($data, &$msg)
{
Log::DEBUG("call back:" . json_encode($data));
$notfiyOutput = array();
if(!array_key_exists("transaction_id", $data)){
$msg = "输入参数不正确";
$this->para['code'] = 0;
$this->para['data'] = '';
return false;
}
//查询订单,判断订单真实性
if(!$this->Queryorder($data["transaction_id"])){
$msg = "订单查询失败";
$this->para['code'] = 0;
$this->para['data'] = '';
return false;
}
$this->para['code'] = 1;
$this->para['data'] = $data;
return true;
}
/**
* 自定义方法 检测微信端是否回调成功方法
* @return multitype:number string
*/
public function IsSuccess(){
return $this->para;
}
}

<p>到这里基本上完成,可以在微信端打开http://域名/Charge/index.php/Test/index/<br ?-->
我的环境,HTTP服务器没有重写url,微信支付继续探索中,有些地方可能写的有问题或不足,望大家谅解,互相学习。

本文来自投稿,不代表重蔚自留地立场,如若转载,请注明出处https://www.cwhello.com/4523.html

如有侵犯您的合法权益请发邮件951076433@qq.com联系删除

(0)
重蔚的头像重蔚管理团队
上一篇 2018年3月7日
下一篇 2018年3月7日

相关推荐

  • PHP实现数据库集群的方法。

    随着Web应用程序和电子商务网站的不断发展,数据存储和处理的需求也日益增加。为了应对这一挑战,数据库集群成为了一种越来越受欢迎的解决方案。然而,由于各种原因,许多企业选择使用PHP作为开发语言来实现其数据…

    2023年5月21日
    01
  • php使用ftp实现文件上传与下载功能(附代码)

    ftp文件上传 php自带有ftp操作的函数包,一个比较简单实现的ftp文件上传操作可以通过以下几个步骤来完成: 1、确认ftp server的ip地址与port端口信息(如果使用的是默认端口则可以不关心); 2、进行ftp_connect操…

    2018年3月22日
    0306
  • PHP中的水平扩展。

    PHP是一种广泛应用于Web开发的脚本语言,它的流行度和应用范围都在不断扩大。但在PHP的应用中,我们有时候会遇到一些性能问题,比如PHP编写的应用程序在高并发下响应速度缓慢,甚至会出现崩溃的情况。为了解决这些…

    2023年5月28日
    02
  • PHP中的SSL/TLS。

    作为一种面向网络的脚本语言,PHP在Web应用程序中发挥着重要的作用。特别是在涉及到用户隐私和安全性的场景中,SSL/TLS技术的支持成为了PHP开发者极为重要的一个方面。SSL(Secure Sockets Layer)和TLS(Transport…

    2023年5月28日
    03
  • 如何利用PHP实现商城的礼品卡系统。

    随着互联网的不断发展,越来越多的消费者开始通过电子商城进行购物,而在电子商城中,礼品卡成为了一种非常受欢迎的赠送方式。那么,如何利用PHP实现商城的礼品卡系统呢?本文将为大家介绍相关实现方法。一、礼品卡…

    2023年5月30日
    01
  • 如何在PHP中实现音视频会议。

    在当今繁忙的社交和商业环境中,音视频会议已成为人们与客户或同事交流的必要方式。PHP是一种广泛使用的编程语言,其中实现音视频会议是可能的。在本文中,我们将讨论如何在PHP中实现音视频会议。选择一个合适的Web…

    2023年5月23日
    06
  • 如何使用PHP进行智能问答和知识图谱。

    随着人工智能技术的发展,智能问答系统和知识图谱已经成为了人们使用互联网获取信息的重要方式之一。PHP是一种非常流行的编程语言,非常适合用于开发智能问答系统和知识图谱。本文将介绍如何使用PHP进行智能问答和…

    2023年5月28日
    04
  • PHP入门指南:PHP和Spark。

    PHP是一种非常流行的服务器端编程语言,因为它简单易学、开放源代码和跨平台。目前,很多大企业都采用PHP语言来构建应用程序,例如Facebook和WordPress等。Spark是一种快速且轻量级的开发框架,可用于构建Web应用程…

    2023年5月22日
    04

联系我们

QQ:951076433

在线咨询:点击这里给我发消息邮件:951076433@qq.com工作时间:周一至周五,9:30-18:30,节假日休息