PHP如何更科学地接入第三方渠道(从支付开始)

相当多的时候,我们写业务代码时,为了尽快实现产品功能,并不会采用较为复杂的设计模式。这也是为什么很多老项目往往有一堆if-else,别怪前任,有时候他也是不得已而为之,业务一直变化+膨胀,Boss催得又紧,通常都会选择用最快速的方法来实现逻辑再说。以最常见的接入第三方支付为例,接入微信支付、支付宝支付、苹果支付等等。

最简单粗暴的写法是,每次接入新的支付渠道都在PayController中引入新渠道的支付处理类,在switch中增加case来处理对应的渠道逻辑,代码如下:

<?php

namespace app\controller;

use app\model\Goods;
use app\model\Payment;
use payment\wxpay\WxPay;
use payment\alipay\AliPay;

class PayController extends Base;
{
    function __construct()
    {
        // code...
    }

    // 以下均为模拟代码,很多逻辑没写出来,上面的类也是我虚构的,大体是Thinkphp为例
    public function doPay()
    {
        $pay_way  = $_POST['pay_way'];
        $goods_id = $_POST['goods_id'];

        $goods  = Goods::get($goods_id);
        $config = Payment::where(['pay_way' => $pay_way])->find();

        switch ($pay_way) {
            case 'wxpay':
                $payment = new WxPay($config);
                break;
            
            case 'alipay':
                $payment = new AliPay($config);
                break;
            // 更多渠道...
            default:
                return self::error('暂不支持该渠道');
                break;
        }

        // 实际情况以你自己的逻辑参数为准
        $result = $payment->doPay([
            'goods_id'      => $goods_id,
            'goods_name'    => $goods->title,
            'amount'        => $goods->price,
            'order_no'      => makeOrderNo(),
        ]);

        return self::success($result);
    }
}

随着对接的支付渠道越来越多,我们发现use的类越来越多,case的内容也越来越多,产生了大量重复的代码,这本能地让我感到恶心,于是我决定在config中配置好由那个支付类处理,然后通过反射类来实现:

<?php

namespace app\controller;

use app\model\Goods;
use app\model\Payment;

class PayController extends Base;
{
    function __construct()
    {
        // code...
    }

    // 以下均为模拟代码,很多逻辑没写出来,上面的类也是我虚构的,大体是Thinkphp为例
    public function doPay()
    {
        $pay_way  = $_POST['pay_way'];
        $goods_id = $_POST['goods_id'];

        $goods  = Goods::get($goods_id);
        $config = Payment::where(['pay_way' => $pay_way])->find();

        // 如payment\wxpay\WxPay
        if ($config && class_exists($config->class_name)) {
            $reflection = new \ReflectionClass($config->class_name);
            $payment = $reflection->newInstanceArgs($config->class_name);
        }else{
            return self::error('暂不支持该渠道');
        }

        // 实际情况以你自己的逻辑参数为准
        $result = $payment->doPay([
            'goods_id'      => $goods_id,
            'goods_name'    => $goods->title,
            'amount'        => $goods->price,
            'order_no'      => makeOrderNo(),
        ]);

        return self::success($result);
    }
}

此时大部分情况其实已经满足了......

直到我开始接入各种短信渠道,我又用了一次反射,从某种意义上来说,我又重复了一堆代码,于是我终于开始考虑依赖注入的方式,首先实现一个用来自动实例化以及存放实例的DI容器类:

<?php

namespace app\common;

class DI
{
    private $container = array();
    
    // 向容器内注入实例
    public function set($key, $class, $arg = [])
    {
        if (count($arg) == 1 && is_array($arg[0])) {
            $arg = $arg[0];
        }

        /*
         * 注入的时候不做任何的类型检测与转换
         * 由于编程人员人为问题,该注入资源并不一定会被用到
         */
        $this->container[$key] = array(
            "class" => $class,
            "params" => $arg,
        );
    }

    // 移除容器内的实例
    public function delete($key)
    {
        unset($this->container[$key]);
    }

    // 清空容器
    public function clear()
    {
        $this->container = array();
    }

    // 获取容器指定实例
    public function get($key)
    {
        if (isset($this->container[$key])) {
            $result = $this->container[$key];
            if (is_object($result['class'])) {
                return $result['class'];
            } else if (is_callable($result['class'])) {
                return $this->container[$key]['class'];
            } else if (is_string($result['class']) && class_exists($result['class'])) {
                $reflection = new \ReflectionClass($result['class']);
                $ins = $reflection->newInstanceArgs($result['params']);
                $this->container[$key]['class'] = $ins;
                return $this->container[$key]['class'];
            } else {
                return $result['class'];
            }
        } else {
            return null;
        }
    }
}

控制器就变成如下代码:

<?php

namespace app\controller;

use app\common\DI;
use app\model\Goods;
use app\model\Payment;
use app\services\PaymentService

class PayController extends Base;
{
    function __construct()
    {
        // code...
    }

    // 以下均为模拟代码,很多逻辑没写出来,上面的类也是我虚构的,大体是Thinkphp为例
    public function doPay()
    {
        $pay_way  = $_POST['pay_way'];
        $goods_id = $_POST['goods_id'];

        $goods  = Goods::get($goods_id);
        $config = Payment::where(['pay_way' => $pay_way])->find();

        // 如payment\wxpay\WxPay
        $di = new DI();
        $di->set($pay_type, $config->class_name);
        $payment = new PaymentService($di->get($pay_type));
        $payment->config($config);

        // 实际情况以你自己的逻辑参数为准
        $result = $payment->doPay([
            'goods_id'      => $goods_id,
            'goods_name'    => $goods->title,
            'amount'        => $goods->price,
            'order_no'      => makeOrderNo(),
        ]);

        return self::success($result);
    }
}

PaymentService代码如下:

<?php
namespace app\services;

use app\model\Payment;

class PaymentService
{
    private $pay;
    
    public function __construct($pay)
    {
        $this->pay = $pay;
    }
    
    public function config(Payment $config)
    {
        $this->pay->config($config);
    }
    
    public function doPay($data)
    {
        return $this->pay->doPay($data);
    }
    
    public function notify($data)
    {
        return $this->pay->notify($data);
    }
}

以上就是多种类型的不同渠道(支付、短信、第三方登录等)对接的大致流程,具体的支付类没有放出代码,实现起来比较简单,就不贴代码了。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,384评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,845评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,148评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,640评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,731评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,712评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,703评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,473评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,915评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,227评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,384评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,063评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,706评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,302评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,531评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,321评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,248评论 2 352

推荐阅读更多精彩内容

  • 表情是什么,我认为表情就是表现出来的情绪。表情可以传达很多信息。高兴了当然就笑了,难过就哭了。两者是相互影响密不可...
    Persistenc_6aea阅读 124,793评论 2 7
  • 16宿命:用概率思维提高你的胜算 以前的我是风险厌恶者,不喜欢去冒险,但是人生放弃了冒险,也就放弃了无数的可能。 ...
    yichen大刀阅读 6,044评论 0 4