深度入门laravel

作者: lerko_ | 来源:发表于2016-11-27 22:10 被阅读378次

    入门laravel

    初始化流程

    http 的初始化

    引入composer的autoload

    实例化Applocation(初始化ioc容器,application本身就是ioc容器。并且注册本身到自己的ioc容器中。注册基础的服务提供者【并且执行服务提供者的register和boot方法】。注册一些类别名到ioc容器)

    注册Http的Kernel到ioc容器

    从make中去除kernel(实例化了kernel,kernel调用了一些bootstrap的类来初始化一些系统功能【门面注册,内容提供者注册--app.php中配置的那些】,调用application的boot方法等)

    捕获Request

    发送response(Symfony\Component\HttpFoundation\Response)

    console 的初始化

    几个重要的系统定义初始化的值

    Illuminate\Foundation\Http\Kernel $bootstrappers 初始化一些系统的必须类 包括下面的几个类 (都有bootstrap()方法 初始化了系统绝大多数东西)

        protected $bootstrappers = [
            'Illuminate\Foundation\Bootstrap\DetectEnvironment',
            'Illuminate\Foundation\Bootstrap\LoadConfiguration',
            'Illuminate\Foundation\Bootstrap\ConfigureLogging',
            'Illuminate\Foundation\Bootstrap\HandleExceptions', 
            //初始化门面
            'Illuminate\Foundation\Bootstrap\RegisterFacades',
            //初始化内容提供者
            'Illuminate\Foundation\Bootstrap\RegisterProviders',
            //调用Applocation实例到ioc容器中
            'Illuminate\Foundation\Bootstrap\BootProviders',
        ];
    

    application 注册核心的别名到ioc容器中 契约以及契约的实现类

    public function registerCoreContainerAliases()
       {
           $aliases = [
               'app'                  => ['Illuminate\Foundation\Application', 'Illuminate\Contracts\Container\Container', 'Illuminate\Contracts\Foundation\Application'],
               'auth'                 => ['Illuminate\Auth\AuthManager', 'Illuminate\Contracts\Auth\Factory'],
               'auth.driver'          => ['Illuminate\Contracts\Auth\Guard'],
               'blade.compiler'       => ['Illuminate\View\Compilers\BladeCompiler'],
               'cache'                => ['Illuminate\Cache\CacheManager', 'Illuminate\Contracts\Cache\Factory'],
               'cache.store'          => ['Illuminate\Cache\Repository', 'Illuminate\Contracts\Cache\Repository'],
               'config'               => ['Illuminate\Config\Repository', 'Illuminate\Contracts\Config\Repository'],
               'cookie'               => ['Illuminate\Cookie\CookieJar', 'Illuminate\Contracts\Cookie\Factory', 'Illuminate\Contracts\Cookie\QueueingFactory'],
               'encrypter'            => ['Illuminate\Encryption\Encrypter', 'Illuminate\Contracts\Encryption\Encrypter'],
               'db'                   => ['Illuminate\Database\DatabaseManager'],
               'db.connection'        => ['Illuminate\Database\Connection', 'Illuminate\Database\ConnectionInterface'],
               'events'               => ['Illuminate\Events\Dispatcher', 'Illuminate\Contracts\Events\Dispatcher'],
               'files'                => ['Illuminate\Filesystem\Filesystem'],
               'filesystem'           => ['Illuminate\Filesystem\FilesystemManager', 'Illuminate\Contracts\Filesystem\Factory'],
               'filesystem.disk'      => ['Illuminate\Contracts\Filesystem\Filesystem'],
               'filesystem.cloud'     => ['Illuminate\Contracts\Filesystem\Cloud'],
               'hash'                 => ['Illuminate\Contracts\Hashing\Hasher'],
               'translator'           => ['Illuminate\Translation\Translator', 'Symfony\Component\Translation\TranslatorInterface'],
               'log'                  => ['Illuminate\Log\Writer', 'Illuminate\Contracts\Logging\Log', 'Psr\Log\LoggerInterface'],
               'mailer'               => ['Illuminate\Mail\Mailer', 'Illuminate\Contracts\Mail\Mailer', 'Illuminate\Contracts\Mail\MailQueue'],
               'auth.password'        => ['Illuminate\Auth\Passwords\PasswordBrokerManager', 'Illuminate\Contracts\Auth\PasswordBrokerFactory'],
               'auth.password.broker' => ['Illuminate\Auth\Passwords\PasswordBroker', 'Illuminate\Contracts\Auth\PasswordBroker'],
               'queue'                => ['Illuminate\Queue\QueueManager', 'Illuminate\Contracts\Queue\Factory', 'Illuminate\Contracts\Queue\Monitor'],
               'queue.connection'     => ['Illuminate\Contracts\Queue\Queue'],
               'queue.failer'         => ['Illuminate\Queue\Failed\FailedJobProviderInterface'],
               'redirect'             => ['Illuminate\Routing\Redirector'],
               'redis'                => ['Illuminate\Redis\Database', 'Illuminate\Contracts\Redis\Database'],
               'request'              => ['Illuminate\Http\Request', 'Symfony\Component\HttpFoundation\Request'],
               'router'               => ['Illuminate\Routing\Router', 'Illuminate\Contracts\Routing\Registrar'],
               'session'              => ['Illuminate\Session\SessionManager'],
               'session.store'        => ['Illuminate\Session\Store', 'Symfony\Component\HttpFoundation\Session\SessionInterface'],
               'url'                  => ['Illuminate\Routing\UrlGenerator', 'Illuminate\Contracts\Routing\UrlGenerator'],
               'validator'            => ['Illuminate\Validation\Factory', 'Illuminate\Contracts\Validation\Factory'],
               'view'                 => ['Illuminate\View\Factory', 'Illuminate\Contracts\View\Factory'],
           ];
    
           foreach ($aliases as $key => $aliases) {
               foreach ($aliases as $alias) {
                   $this->alias($key, $alias);
               }
           }
       }
    

    app.php 配置文件中的aliases 就是吧对应的类设置别名(通过AliasLoad 去找到类【prependToLoaderStack】注册了门面的autoload)

    需要知道的知识

    类:Closure
    这个是一个回调函数的具体类
    方法  bind(Closure $closure , object $newthis [, mixed $newscope = 'static' ] ) 
         bindto(object $newthis [, mixed $newscope = 'static' ])
    
    bind:将回调函数绑定到一个具体的类 $newthis(这个绑定之后才有上下文)
        $newscope 代表的是这个类的作用域(即回调函数能访问这个作用域【类】中的成员变量   也可以是一个命名空间加类名的字符串)
        即 $newthis是绑定上下文  $newscope取得上下文的变量
    bindTo 参数如上
    
    static::class --> 返回的是php这个类的域名以及类名的全部string
    
    callable 标示一个可以调用的类型
    
    接口:ArrayAccess:让类可以像数组一样的进行使用
        
    
    //laravel的基础类
    Macroable:实现了一个注册callable类型就可以添加到类中的trait
    
    

    applocatin 分析

    Applocation 类继承了接口:IApplication, HttpKernelInterface,继承了类Container

    • Application 接口
    interface Application extends Container
    {
        /**
         * 获取laravel的版本
         * @return string
         */
        public function version();
    
        /**
         * 获取laravel框架的基本路径(项目根目录)
         * @return string
         */
        public function basePath();
    
        /**
         * 取得环境
         * @return string
         */
        public function environment();
    
        /**
         * 判断框架是否是在维护
         * @return bool
         */
        public function isDownForMaintenance();
    
        /**
         * 注册所有的配置提供者
         * @return void
         */
        public function registerConfiguredProviders();
    
        /**
         * 注册服务提供者到applocation中  并且调用服务的register方法
         * @param  \Illuminate\Support\ServiceProvider|string  $provider
         * @param  array  $options
         * @param  bool   $force
         * @return \Illuminate\Support\ServiceProvider
         */
        public function register($provider, $options = [], $force = false);
    
        /**
         *注册一个延迟的服务提供者(如果已经有注册服务的话就从deferredServices中unset服务 然后在调用register方法注册服务)
         * @param  string  $provider
         * @param  string  $service
         * @return void
         */
        public function registerDeferredProvider($provider, $service = null);
    
        /**
         * 调用所有的boot的callback并且吧所有注册的服务提供者都调用boot方法
         * @return void
         */
        public function boot();
    
        /**
         * 注册一个boot(框架启动的回调函数)
         *
         * @param  mixed  $callback
         * @return void
         */
        public function booting($callback);
    
        /**
         * 注册一个框架启动完成之后的回调函数
         *
         * @param  mixed  $callback
         * @return void
         */
        public function booted($callback);
    
        /**
         * Get the path to the cached "compiled.php" file.
         *
         * @return string
         */
        public function getCachedCompilePath();
    
        /**
         * Get the path to the cached services.php file.
         *
         * @return string
         */
        public function getCachedServicesPath();
    }
    
    //http接口
    interface HttpKernelInterface
    {
        const MASTER_REQUEST = 1;
        const SUB_REQUEST = 2;
        public function handle(Request $request, $type = self::MASTER_REQUEST, $catch = true);
    }
    
    //依赖注入容器
    interface Container
    {
        /**
         * Determine if the given abstract type has been bound.
         *
         * @param  string  $abstract
         * @return bool
         */
        public function bound($abstract);
    
        /**
         * Alias a type to a different name.
         *
         * @param  string  $abstract
         * @param  string  $alias
         * @return void
         */
        public function alias($abstract, $alias);
    
        /**
         * Assign a set of tags to a given binding.
         *
         * @param  array|string  $abstracts
         * @param  array|mixed   ...$tags
         * @return void
         */
        public function tag($abstracts, $tags);
    
        /**
         * Resolve all of the bindings for a given tag.
         *
         * @param  array  $tag
         * @return array
         */
        public function tagged($tag);
    
        /**
         * Register a binding with the container.
         *
         * @param  string|array  $abstract
         * @param  \Closure|string|null  $concrete
         * @param  bool  $shared
         * @return void
         */
        public function bind($abstract, $concrete = null, $shared = false);
    
        /**
         * Register a binding if it hasn't already been registered.
         *
         * @param  string  $abstract
         * @param  \Closure|string|null  $concrete
         * @param  bool  $shared
         * @return void
         */
        public function bindIf($abstract, $concrete = null, $shared = false);
    
        /**
         * Register a shared binding in the container.
         *
         * @param  string|array  $abstract
         * @param  \Closure|string|null  $concrete
         * @return void
         */
        public function singleton($abstract, $concrete = null);
    
        /**
         * "Extend" an abstract type in the container.
         *
         * @param  string    $abstract
         * @param  \Closure  $closure
         * @return void
         *
         * @throws \InvalidArgumentException
         */
        public function extend($abstract, Closure $closure);
    
        /**
         * Register an existing instance as shared in the container.
         *
         * @param  string  $abstract
         * @param  mixed   $instance
         * @return void
         */
        public function instance($abstract, $instance);
    
        /**
         * Define a contextual binding.
         *
         * @param  string  $concrete
         * @return \Illuminate\Contracts\Container\ContextualBindingBuilder
         */
        public function when($concrete);
    
        /**
         * Resolve the given type from the container.
         *
         * @param  string  $abstract
         * @param  array   $parameters
         * @return mixed
         */
        public function make($abstract, array $parameters = []);
    
        /**
         * Call the given Closure / class@method and inject its dependencies.
         *
         * @param  callable|string  $callback
         * @param  array  $parameters
         * @param  string|null  $defaultMethod
         * @return mixed
         */
        public function call($callback, array $parameters = [], $defaultMethod = null);
    
        /**
         * Determine if the given abstract type has been resolved.
         *
         * @param  string $abstract
         * @return bool
         */
        public function resolved($abstract);
    
        /**
         * Register a new resolving callback.
         *
         * @param  string    $abstract
         * @param  \Closure|null  $callback
         * @return void
         */
        public function resolving($abstract, Closure $callback = null);
    
        /**
         * Register a new after resolving callback.
         *
         * @param  string    $abstract
         * @param  \Closure|null  $callback
         * @return void
         */
        public function afterResolving($abstract, Closure $callback = null);
    }
    
    • 通过Illuminate\Config\Repository读取配置文件的数据

    apploction 初始化注册的的基础服务和基础容器

    1. 基础服务
    //事件服务
    class EventServiceProvider extends ServiceProvider
    {
        /**
         * Register the service provider.
         *
         * @return void
         */
        public function register()
        {
            $this->app->singleton('events', function ($app) {
                return (new Dispatcher($app))->setQueueResolver(function () use ($app) {
                    return $app->make('Illuminate\Contracts\Queue\Factory');
                });
            });
        }
    }
    //路由服务
    class RoutingServiceProvider extends ServiceProvider
    {
        /**
         * Register the service provider.
         *
         * @return void
         */
        public function register()
        {
            $this->registerRouter();
    
            $this->registerUrlGenerator();
    
            $this->registerRedirector();
    
            $this->registerPsrRequest();
    
            $this->registerPsrResponse();
    
            $this->registerResponseFactory();
        }
    }
    

    主要关注 EventServiceProvider

    核心概念

    依赖注入

    依赖注入接口

    //Application 是这个接口的子类
    <?php
    namespace Illuminate\Contracts\Container;
    use Closure;
    interface Container
    {
        //判断是不是存在这个ioc
        public function bound($abstract);
        public function alias($abstract, $alias);
        public function tag($abstracts, $tags);
        public function tagged($tag);
        public function bind($abstract, $concrete = null, $shared = false);
        public function bindIf($abstract, $concrete = null, $shared = false);
        public function singleton($abstract, $concrete = null);
        public function extend($abstract, Closure $closure);
        public function instance($abstract, $instance);
        public function when($concrete);
        public function make($abstract, array $parameters = []);
        public function call($callback, array $parameters = [], $defaultMethod = null);
        public function resolved($abstract);
        public function resolving($abstract, Closure $callback = null);
        public function afterResolving($abstract, Closure $callback = null);
    }
    
    

    依赖注入的一些类型

    1. 注册单例
    2. 注册别名

    服务提供者

    服务提供者是一个在框架运行流程中可以注册到框架的applocation中 然后有两个主要的方法
    1. register
    2. boot
    
    执行顺序 register->boot
    
    app.php配置文件中有许多初始化的服务提供者
    系统的启动也是这些提供者提供的
    

    门面

    门面抽象类中的getFacadeAccessor()方法(重载) 就是用来获取ioc容器中alias别名容器中对应的【契约=》类】;

    //门面就是一个注册函数或者对象到类中 之后可以通过__callStatic 来访问这些方法
    // 使得很多注册的门面可以直接通过static的方式进行访问
    //门面类
    namespace Illuminate\Support\Facades;
    abstract class Facade
    
    //门面注册类  (在application中进行了执行)
    namespace Illuminate\Foundation\Bootstrap;
    class RegisterFacades
    

    主要使用(http)

    路由

    控制器

    abstract class Controller
    {
        /**
         * The middleware registered on the controller.
         *
         * @var array
         */
        protected $middleware = [];
    
        /**
         * Register middleware on the controller. 注册一个中间件
         *
         * @param  array|string|\Closure  $middleware
         * @param  array   $options
         * @return \Illuminate\Routing\ControllerMiddlewareOptions
         */
        public function middleware($middleware, array $options = [])
        {
            foreach ((array) $middleware as $m) {
                $this->middleware[] = [
                    'middleware' => $m,
                    'options' => &$options,
                ];
            }
    
            return new ControllerMiddlewareOptions($options);
        }
    
        /**
         * Get the middleware assigned to the controller.  获取注册到控制器的中间件
         *
         * @return array
         */
        public function getMiddleware()
        {
            return $this->middleware;
        }
    
        /**
         * Execute an action on the controller.  调用控制器的方法
         *
         * @param  string  $method
         * @param  array   $parameters
         * @return \Symfony\Component\HttpFoundation\Response
         */
        public function callAction($method, $parameters)
        {
            return call_user_func_array([$this, $method], $parameters);
        }
    
        /**
         * Handle calls to missing methods on the controller.  调用控制器没有的方法抛出异常
         *
         * @param  array   $parameters
         * @return mixed
         *
         * @throws \Symfony\Component\HttpKernel\Exception\NotFoundHttpException
         */
        public function missingMethod($parameters = [])
        {
            throw new NotFoundHttpException('Controller method not found.');
        }
    
        /**
         * Handle calls to missing methods on the controller.  调用未知方法直接抛出异常
         *
         * @param  string  $method
         * @param  array   $parameters
         * @return mixed
         *
         * @throws \BadMethodCallException
         */
        public function __call($method, $parameters)
        {
            throw new BadMethodCallException("Method [{$method}] does not exist.");
        }
    }
    

    中间件

    用来过滤请求 然后可以对请求进行处理
    比如 验证 跳转 过滤
    

    php artisan make:middleware CheckAge 可以直接生成中间件

    1. 普通中间件(中间件前)
    class CheckAge
    {
        public function handle($request, Closure $next)
        {
            return $next($request);
        }
    }
    
    1. 后置中间件
    class AfterMiddleware
    {
        public function handle($request, Closure $next)
        {
            $response = $next($request);
            // 执行动作
            return $response;
        }
    }
    

    相关文章

      网友评论

      • 3f182002e79a:请教一下,我使用路由中间件的时候提示 ReflectionException in Container.php line 738:
        Class old does not exist 怎么解决啊?初识laravel
        lerko_:不到类,估计是你没有注册到ioc容器中

      本文标题:深度入门laravel

      本文链接:https://www.haomeiwen.com/subject/ojbnpttx.html