zoukankan      html  css  js  c++  java
  • laravel的模块化是如何实现的

    laravel的模块化是如何实现的

    在laravel提供的官方文档上,有一个这样的名词 服务提供者,文档中介绍了它在laravel框架中的角色,以及如何使用它,但却没有讲明服务提供者的本质--它是为了解决什么问题而存在的? 不解决这一点,对于它的理解,则只会停留在表面.服务提供者是laravel实现模块化设计的手法.

    为什么要进行模块化设计这里就不说的,可以参考下这些:模块化设计 , 模块化的意义何在?

    为了实现模块化,必然要将一段程序组合起来,完成特定的事,从而形成模块.在laravel中, 一个模块都表现为一个Service.应用程序主体与组件(模块)之间必然要通过某种方式连接起来,才能使组件被主体所调用.在laravel中一个Service会被它的 Service Prorider 注入到ioc中,这样组件就与主体联系了起来, 具体的表现形式则是组件被主体所用,主体使用组件完成组件所擅长的事. 源代码是最好的文档,接下来,我们就看看"文档"是怎么说的.

    laravel的启动过程

    我们从启动开始,详细地分析在整个应用程序的生命周期中,Service Provider到底是什么?通过阅读源码方式来了解Serivce Provider在laravel中到底做了什么.

    laravle的启动过程做了很多事,这里就不一一叙述了,主要说明有关Sevice Provider的部分.laravel有两个入口分别处理不同的请求1:public/index.php(http) 2.php arisan(cli).虽然入口有两种但它们的启动过程却相差无几(具体可以再详细了解这两种方式).

    基础启动

    bootstrap/app.php则是它的主要启动文件, 在IlluminateFoundationApplication::__construct中就可以看到应用主体的启动做了什么:

    • 绑定Appliation和Container
    • 注册最基础的Service Provider
    • 注册别名
    • 注册该应用程序的路径

    上面启动过程的代码较简单,代码不分析了.就说一说为什么要做这几步,以及这几步对lavarel有什么作用.

    laravel中以ioc为基础来构建应用程序的,所以,把这个最基础的Applicatoin放入容器中,供需要时可以随时从容器中提取使用(container 与 Application的关系看看代码就清晰了).

    一个应用的合理运行必然离不开一些最基本的功能点,就像人类一样,虽然最重要的是大脑,但同样也不能没有血管,心脏.所以在laravel中也同样存在一些不可或缺的service provider, 比如事件和路由器.IlluminateEventsEventServiceProvider则把构建观察者的类注入到了ioc中,同时,在laravel源代码中大量使用了观察者模式来处理问题,IlluminateRoutingRoutingServiceProvider则把路由相关的服务注入到了主体中,因为一个http请求最不可或缺的当然是路由;

    注册别名,则是为了调用ioc中的类时更方便,还有一点要注意是的,可以为一个类注册多个别名,为什么呢,其实这多个别名是存在关系的,它们大多是父类与子类的关系,就像Appliation和Container;

    注册应用的路径则把一样常用的,跟应用有关的路径放到ioc中方便取用.启动的第一步到这就结束了.

    http应用请求的启动

    紧接着,又注册了三个类到ioc中,分别是http处理核心类,cli处理核心类,和异常处理.在这里注册了两种不同请亲的处理类,所以两个请求入口所做的事情都差不多了.接下来以http请求为例来,继续看下去.public/index.php:50则初始化了AppHttpKernel::__construct,也就初始化了中间件.接着public/index.php:52则调用了IlluminateFoundationHttpKernel::handle来处理请求,进一步看下去,也就是IlluminateFoundationHttpKernel::bootstrap中通过Application::bootstrapWith完成http请求环境的初始化,具体则是:

    • 标记Application为已经引导启动状态
    • 依次启动IlluminateFoundationHttpKernel::$bootstrappers中的项目
      1.IlluminateFoundationBootstrapDetectEnvironment:设置应用程序的环境
      2.IlluminateFoundationBootstrapLoadConfiguration:载入应用程序的配置文件
      3.IlluminateFoundationBootstrapConfigureLogging:绑定日志处理类
      4.IlluminateFoundationBootstrapHandleExceptions:异常处理
      5.IlluminateFoundationBootstrapRegisterFacades:Facade模式的应用
      6.IlluminateFoundationBootstrapRegisterProviders:注册Service Provider
      7.IlluminateFoundationBootstrapBootProviders:标记启动完成 , 执行Srvice Providerregister方法,完成注入

    然后,在vendor/laravel/framework/src/Illuminate/Foundation/Http/Kernel.php:129就进入到了路由器,进而进入对应Controller进行处理,最后完成处理,请求结束.这一部分就不说了.

    cli请求的启动

    这一部分略,步骤与http类似

    从它的请求生命周期可以看出以下几点
    1:Service Provider在启动阶段就已经与注入了IOC
    2:分为了两种,一种是由系统控制,而另一种则是可由用户控制.第一种是构成应用程序的基础,不能缺少;第二种则将控制权交给用户,由用户掌控应用的运行,比如加载第三方composer包,或自定义的Service等等.

    此时,模块化已经初现雏形.在Controller中,可以调用或组合不同的Service, 来完成特定的逻辑,因为它已经在IOC中了;我们可以控制不同的Service或增或减,自定义这个应用程序的功能,成为新的系统.接下来,再看看Service Provider做了什么,使模块化在laravel中更完善.

    注册Service Provider

    在注册基础的Service Provider中,不难发现,完成注册过程的是IlluminateFoundationApplication::register,这个方法比较简单,一是执行了ServierProceder::register方法,通过这个接口将Service注册动IOC中;二是在系统已经启动的情况下,执行ServierProceder::boot方法,作用在这里讲的很清楚.

    在注册用户自定义(config/app.php:124)时,IlluminateFoundationApplication::registerConfiguredProviders则完成了注册过程,其处理核心IlluminateFoundationProviderRepository::load主要做了以下几步
    1:解析所有的Service Provider, 通过它的defer属性来决定是否延迟加载
    2:将解析的结果,也就是一个Service Provider的数组,缓存为文件,下次直接载入解析后的缓存文件
    3:只加载defer属性不为ture的Service Provider, 并注册
    4:将延迟加载的放支容器中(APP::$deferredServices),在需要的时候再加载

    从注册过程可以看出,我们可以定义Service Provider加载的时间,并不仅仅在启动这过程中加载,而是在需要的时候再加载.一种是通过IlluminateFoundationApplication::make;还有一种方式延迟加载则是在when方法中通过事件注册到某项事件上去,这一点则要好好看看IlluminateFoundationProviderRepository类了.

    理清它的注册过程后,我们再仔细看看Service Provider的抽象类,它作为基类,提供了Servicer Provider能做的所有事情.了解它能更好的理解Service Provider.这个类并不复杂,唯一一个不是很清晰的只有commands方法:它将命令注册到了IlluminateConsoleEventsArtisanStarting事件中,为什么这样做? 在cli时,启动时,会有该事件的执行,从而通过Service Provider把命令注入到了IOC中,这样执行命令就行云流水了.

    到这里,用Service Provider来实现模块化的功能得到了强化,既可以延迟加载,还可以注册命令.接下来,通过一些例子,看看它具体做了什么.

    Service Provider的使用

    Service Provider有很多,我们随便挑两个来看看

    QueueServiceProvider

    IlluminateQueueQueueServiceProvider并非系统控制级别,看源代码,可以发现,里面有很多注册方法,注册了很多有关队列的类到IOC中,同时也注册了很多命令.在provides方法中,它返回的就是一个所有在这个Service Provider出现的的各个类,它是做什么的?其实我们可以发现,这个Service Provider是延迟加载的,所以这个Service所提供的各种服务并不会出现在服务中,上面我们说过,在需要的时候再加载.这一功能就是由provides中的返回值来决定的,当调用IOC中,那些延迟加载服务中出现的那些类时,再加载这些延迟的服务,从而可以获得延迟服务所提供的服务.

    EventServiceProvider

    IlluminateEventsEventServiceProvider是应用程序的核心,它又做了什么呢?
    它也是向IOC中注册了事件处理的类,同时为这个类设置了队列解析的类.

    HashServiceProvider

    IlluminateHashingHashServiceProvider也向IOC注册一hash处理的类,并且是延迟加载的.

    小结

    查看Service Provicer的使用后,我们可以发现,它们都做了同们的事,就是把相应的服务注入到ICO中,以供使用,从而构成完成的系统.所以,现在我们就可以明白Service Provider是lararel完成模块化设计的方法,只不过融入了一些laravel自己的特点.

  • 相关阅读:
    JavaScript循环 — for、for/in、while、do/while
    Git
    js根据日期获取所在周
    nodejs安装 Later version of Node.js is already installed. Setup will now exit 及 node与npm版本不符
    sqlserver 2014 json
    根据官方数据制作中国省市区数据库
    kubernetes系列③:集群升级-实践(参照官方文档)
    kubernetes系列:服务外部访问集中管理组件-ingress-nginx
    kubernetes系列-部署篇:Kubernetes的包管理工具-helm
    kubernetes系列-部署篇:使用kubeadm初始化一个高可用的Kubernetes集群
  • 原文地址:https://www.cnblogs.com/z1298703836/p/5300333.html
Copyright © 2011-2022 走看看