PHP的依赖注入(DI) 和 控制反转(IoC)

简介

  • IoC - Inversion of Control 控制反转
  • DI - Dependency Injection 依赖注入
  • 依赖注入和控制反转说的实际上是同一个东西,它们是一种设计模式,这种设计模式用来减少程序间的耦合

优势(为什么使用)

  • 使用依赖注入,最重要的一点好处就是有效的分离了对象和它所需要的外部资源,使得它们松散耦合,有利于功能复用,更重要的是使得程序的整个体系结构变得非常灵活。

概念

  • 依赖注入和控制反转是对同一件事情的不同描述,从某个方面讲,就是它们描述的角度不同。

  • 依赖注入是从应用程序的角度在描述,可以把依赖注入,即:应用程序依赖容器创建并注入它所需要的外部资源;

  • 而控制反转是从容器的角度在描述,即:容器控制应用程序,由容器反向的向应用程序注入应用程序所需要的外部资源。

问题

  1. 整个过程中参与者都有谁?

  2. 依赖:谁依赖于谁?为什么需要依赖?

  3. 注入:谁注入于谁?到底注入了什么?

  4. 控制反转:谁控制谁?控制什么?为何叫反转(有反转就应该有正转了,正转是什么呢?)

  5. 依赖注入和控制反转是同一概念吗?

回答

  1. 整个过程中参与者都有谁?

    一般有三方参与者,一个是某个对象;一个是IoC/DI的容器;另一个是某个对象的外部资源。某个对象指的就是任意的、普通的PHP对象; IoC/DI的容器简单点说就是指用来实现IoC/DI功能的一个框架程序;对象的外部资源指的就是对象需要的,但是是从对象外部获取的,都统称资源,比如:对象需要的其它对象、或者是对象需要的文件资源等等。

  2. 谁依赖于谁:

    当然是某个对象依赖于IoC/DI的容器

  3. 为什么需要依赖:

    对象需要IoC/DI的容器来提供对象需要的外部资源

  4. 谁注入于谁:

    是IoC/DI的容器 注入 某个对象

  5. 到底注入什么:

    就是注入某个对象所需要的外部资源

  6. 谁控制谁:

    当然是IoC/DI的容器来控制对象了

  7. 控制什么:

    主要是控制对象实例的创建

  8. 为何叫反转:

    反转是相对于正向而言的,那么什么算是正向的呢?
    考虑一下常规情况下的应用程序,如果要在A里面使用C,你会怎么做呢?当然是直接去创建C的对象,也就是说,是在A类中主动去获取所需要的外部资源C($c = new C();),这种情况被称为正向的。那么什么是反向呢?就是A类不再主动去获取C,而是被动等待,等待IoC/DI的容器获取一个C的实例,然后反向的注入到A类中。
    用图例来说明一下,先看没有IoC/DI的时候,常规的A类使用C类的示意图,如下图所示:

php调试支付宝时所需的秘钥获取

获取支付宝公钥及设置应用秘钥

  • 登录支付宝开放平台https://openhome.alipay.com/

5A6B083E-CB2E-448F-815D-E75730EC0C92.png

  • 生成应用公钥及及商户私钥
    1529656833057.jpg
  • 应用公钥设置
    WX20180622-164851@2x.png
  • php代码里用到的支付宝公钥和商户私钥
    WX20180622-165225@2x.png

获取合作伙伴id及秘钥(安全检验码key)
WX20180622-164633@2x.png

  • 配置合作伙伴id及秘钥(安全检验码key)

     // 配置文件路径:config/latrell-alipay.php
     return [
     //合作身份者id,以2088开头的16位纯数字。
     'partner_id' => '20888xxxxxxxx', // 
     //卖家支付宝帐户。
     'seller_id' => 'xxxxx@163.com' // 
     ];
    

laravel 支付宝支付接口调试

Alipay

  • 支付宝SDK在Laravel5/Lumen封装包。
  • 该拓展包想要达到在Laravel5/Lumen框架下,便捷使用支付宝的目的。

安装支付宝SDK扩展包

composer require latrell/alipay dev-master

配置及使用

  • 找到 config/app.php 配置文件中,key为 providers 的数组,在数组中添加服务提供者

     // 支付宝支付
     Latrell\Alipay\AlipayServiceProvider::class,
    

说明

  • 配置文件 config/latrell-alipay.php 为公共配置信息文件,
    config/latrell-alipay-web.php 为Web版支付宝SDK配置,
    config/latrell-alipay-mobile.php 为手机端支付宝SDK配置。
  • 运行 php artisan vendor:publish 命令,发布配置文件到你的项目中。

在bootstrap/app.php里注册服务。

$app->register(
    Latrell\Alipay\AlipayServiceProvider::class
);

代码例子

  • 网页

     // 创建支付单。
     $alipay = app('alipay.web');
     $alipay->setOutTradeNo('订单号');
     $alipay->setTotalFee('订单价格(元)');
     $alipay->setSubject('商品名称');
     $alipay->setBody('商品描述');
     $alipay->setQrPayMode('4'); //该设置为可选,添加该参数设置,支持二维码支付。
     // 跳转到支付页面。
     return redirect()->to($alipay->getPayLink());
    
  • 手机端

     // 创建支付单。
     $alipay = app('alipay.mobile');
     $alipay->setOutTradeNo('订单号');
     $alipay->setTotalFee('商品价格(元)');
     $alipay->setSubject('商品名称');
     $alipay->setBody('商品描述');
     // 返回签名后的支付参数给支付宝移动端的SDK。
     return $alipay->getPayPara();
    

结果通知

  • 网页

     /**
     * 异步通知
     */
     public function webNotify(Request $request)
     {
         // 验证请求。
         if (! app('alipay.web')->verify()) {
             // // 日志
             // Log::notice('Alipay notify post data verification fail.', [
             //  'data' => $request->getContent()
             // ]);
             return 'fail';
         }
    
         // 判断通知类型。
         switch ($request->trade_status) {
             case 'TRADE_SUCCESS':
             case 'TRADE_FINISHED':
             // TODO: 支付成功,取得订单号进行其它相关操作。
             // // 日志
             // Log::debug('Alipay notify post data verification success.', [
             //     'out_trade_no' => $request->out_trade_no,
             //     'trade_no' => $request->trade_no
             // ]);
             break;
         }
    
         return 'success';
     }
     
     /**
     * 同步通知
     */
     public function webReturn(Request $request)
     {
         // 验证请求。
         if (! app('alipay.web')->verify()) {
             // // 日志
             // Log::notice('Alipay return query data verification fail.', [
             //  'data' => $request->getContent()
             // ]);
             return ['alipay.fail'];
         }
    
         // 判断通知类型。
         switch ($request->trade_status) {
             case 'TRADE_SUCCESS':
             case 'TRADE_FINISHED':
             // TODO: 支付成功,取得订单号进行其它相关操作。
             // // 日志
             // Log::debug('Alipay notify get data verification success.', [
             //     'out_trade_no' => $request->out_trade_no,
             //     'trade_no' => $request->trade_no
             // ]);
             break;
         }
         return ['alipay.success'];
     }
    
  • 手机端

     /**
      * 支付宝异步通知
      */
     public function alipayNotify(Request $request)
     {
         // 验证请求。
         if (! app('alipay.mobile')->verify()) {
             // // 日志
             // Log::notice('Alipay notify post data verification fail.', [
             //  'data' => $request->getContent()
             // ]);
             return 'fail';
         }
    
         // 判断通知类型。
         switch ($request->trade_status) {
             case 'TRADE_SUCCESS':
             case 'TRADE_FINISHED':
                 // TODO: 支付成功,取得订单号进行其它相关操作。
                 // // 日志
                 // Log::debug('Alipay notify get data verification success.', [
                 //  'out_trade_no' => $request->out_trade_no,
                 //  'trade_no' => $request->trade_no
                 // ]);
             break;
         }
    
         return 'success';
     }
    

搭建视频流服务器

为了实现视频点播和直播,整了一天,看了好多资料,终于有了不小进展,在这里分享给大家。
什么是视频点播?可能你对这个名词不熟悉,但是爱奇艺,腾讯视频等这些视频播放软件想必你肯定知道。没错,视频点播就是在线观看某个视频。额,视频直播?这个就不解释了吧,美女直播,游戏直播等等,哈哈。

首先我们看如何实现视频点播,视频点播支持flv文件及H264编码视频,ACC编码音频的mp4文件:
第一步,创建单独的目录(因为软件较多,容易混乱),下载需要的软件:
我们需要下载nginx,pcre,zlib,openssl以及nginx-rtmp-module

mkdir ser_work

cd ser_work

wget http://nginx.org/download/nginx-1.10.3.tar.gz

wget http://zlib.net/zlib-1.2.11.tar.gz

wget https://ftp.pcre.org/pub/pcre/pcre-8.40.tar.gz

wget https://www.openssl.org/source/openssl-1.0.2k.tar.gz

wget https://github.com/arut/nginx-rtmp-module/archive/master.zip

第二步解压前四个文件

tar -zxvf 文件名

第三步 编译安装nginx:

./configure --prefix=/usr/local/nginx --with-debug --with-pcre=../pcre-8.40 --with-zlib=../zlib-1.2.11 --with-openssl=../openssl-1.0.2k --add-module=../nginx-rtmp-module-master
make
sudo make install

第四步,进入 /usr/local/nginx/conf,配置nginx.conf文件,在文件末尾添加如下内容:

#user  nobody;
worker_processes  1;

#error_log  logs/error.log;
#error_log  logs/error.log  notice;
#error_log  logs/error.log  info;

#pid        logs/nginx.pid;


events {
    worker_connections  1024;
}

rtmp {
    server {
        listen 1935;
        chunk_size 4096;
        application hls {
            live on;
            hls on;
            hls_path /usr/local/nginx/html/hls;
            hls_fragment 3s;
            hls_playlist_length 6s;
        }
        #application rtmplive {
        #    live on;
        #    max_connections 1024;
        #    record off;
        #}
    }
}

http {
    include       mime.types;
    default_type  application/octet-stream;
    sendfile        on;
    #tcp_nopush     on;

    #keepalive_timeout  0;
    keepalive_timeout  65;

    #gzip  on;

    server {
        listen       8080;
        server_name  localhost;

        #charset koi8-r;

        #access_log  logs/host.access.log  main;

        location / {
            root   html;
            index  index.html index.htm;
        }
        # 网页直播地址设置
        location /hls {
            types {
                application/vnd.apple.mpegurl m3u8;
                video/mp2t ts;
            }
            root html;
            add_header Cache-Control no-cache;
        }
        #error_page  404              /404.html;

        # redirect server error pages to the static page /50x.html
        #
        error_page   500 502 503 504  /50x.html;
        location = /50x.html {
            root   html;
        }
    }
}   

视频推流

  • 使用推流工具OBS进行推流
  • 推流地址及秘钥
    rtmp://127.0.0.1:1935/hls
    7s3iodndk
  • 网页观看地址
    http://127.0.0.1:8080/hls/7s3iodndk.m3u8
    VLC观看地址:rtmp://127.0.0.1:1935/hls/7s3iodndk

WX20180611-112600@2x.png

WX20180611-112908@2x.png