ThinkPHP容器源码深度解析(5)

本文涉及的产品
容器镜像服务 ACR,镜像仓库100个 不限时长
简介: ThinkPHP容器源码深度解析

实战案例


光说不干,事事落空;又说又干,马到成功。直接开干


新建文件kaka/container/countableTest.php,并且添加以下内容


image.png


接着在文件application/index/controller/Container.php中学会使用Countable。


这里注意一下用法,是直接使用count();


image.png

image.png



Countable中的count()跟平时使用count()方法有什么区别


顺便看一下PHP源码中的解释


可以看到第一个参数可以是数组也可是是countable


咔咔的理解是Countable只是重写了SPL中的count方法,为了就是方便定制自己需要的统计规则而已。


int count ( mixed $array_or_countable [, int $mode = COUNT_NORMAL ] )


count你不知道的用法


既然说到了这里,咔咔给大家在普及一个count不是很常用的一个用法。


在平时开发的过程中,这样的用法是最普遍的,也是大家最经常见到的一个使用案例。


image.png


但是如果这时给你一个多维数组,例如下图这样,让你统计这个多维数组,你该怎么统计呢!


这个时候估计大多数小伙伴的想法就是循环然后定义一个计数器累计。


其实count()函数在这一块就已经解决了这个需求。


下方打印结果就是"4----6"


直接使用count()函数一个数组得到的就是第一层数组的长度。


但是count()函数还有第二个参数,设置为1就是递归地计数数组中元素的数目(计算多维数组中的所有元素)


所以你这时在去看文档就会发现,count()函数本身就有俩个参数


第一个参数是必须饿,选择是数组


第二个参数默认是0就是不对多维数组中的所有元素进行计数


当第二个参数为1时就是递归的计算多维数组中的所有元素。


image.png


七、Container容器类剖析

上文中实现了一个自己创建的容器,接下来看看源码中的容器,经过了上文容器中出现的技术点都已经囊括完了。


在接下里阅读容器源码就不会很吃力,如果之前的文章没看,一定要大概过一遍哈!


大家无数次打开的一个文件public/index.php。


曾有多少次打开这个文件想对源码进行一探究竟,但是看着看着就放弃了。


image.png


经过之前的注册树模式之后,你肯定就会明白这行代码会返回什么Container::get('app')


这行代码返回就是app的实例,可以进行简单的断点一下。


可以看到返回就是app类里边的众多属性。


所以说注册树模式不会的在继续返回去看之前写的,要不越看越迷糊。


image.png


那么框架中的容器是怎么定义的呢!它到底是怎么实现的呢!


也就是只需要去关注这个get()方法做的事情就可以了。


image.png


代码就会追踪到文件thinkphp/library/think/Container.php中的get()方法


这里的getInstance()方法不陌生了吧!这就是上文说过的单例模式。


image.png


可以进行代码追踪getInstance()这个方法,你就会在同文件中看到这个单例模式的方法,返回Container实例。


image.png


Container实例调用make方法


代码static::getInstance()返回了Container的实例后,就会去调用本类的make方法,接下来就是对make方法进行详解了。


image.png


在开始阅读make方法里边的源码之前,我们需要先对几个属性进行简单的梳理一下。


这四个属性一定要有点印象,并且一定要区别instance和instances。


这俩个属性一个是单例模式返回当前类的实例,一个是容器中的所有的实例。


image.png


第一次执行结果

   /**
     * 创建类的实例
     * @access public
     * @param  string        $abstract       类名或者标识
     * @param  array|true    $vars           变量
     * @param  bool          $newInstance    是否每次创建新的实例
     * @return object
     */
    public function make($abstract, $vars = [], $newInstance = false)
    {
        // 判断$vars这个变量是否为true
        if (true === $vars) {
            // 总是创建新的实例化对象
            $newInstance = true;
            $vars        = [];
        }
        // app  这里就是在容器别名里获取传递过来的app    如果没有则就是app
        $abstract = isset($this->name[$abstract]) ? $this->name[$abstract] : $abstract;
        // 从容器实例中获取  如果存在则直接返回对应的实例  也就是使用注册树模式
        if (isset($this->instances[$abstract]) && !$newInstance) {
            return $this->instances[$abstract];
        }
        // think\App 从容器标识中获取
        if (isset($this->bind[$abstract])) {
            // 将think\App 复制给$concrete变量
            $concrete = $this->bind[$abstract];
            // 用于代表匿名函数的类  判断是不是闭包
            if ($concrete instanceof Closure) {
                $object = $this->invokeFunction($concrete, $vars);
            } else {
                // $this->name['app'] = think\App
                $this->name[$abstract] = $concrete;
                // 在执行一次本类的make方法,也就是本方法
                return $this->make($concrete, $vars, $newInstance);
            }
        } else {
            $object = $this->invokeClass($abstract, $vars);
        }
        if (!$newInstance) {
            $this->instances[$abstract] = $object;
        }
        return $object;
    }

这是第二次执行流程

    public function make($abstract, $vars = [], $newInstance = false)
    {
        // 判断$vars这个变量是否为true
        if (true === $vars) {
            // 总是创建新的实例化对象
            $newInstance = true;
            $vars        = [];
        }
        // app  这里就是在容器别名里获取传递过来的app    如果没有则就是app
        // 第二次执行时 $abstract = think\App
        $abstract = isset($this->name[$abstract]) ? $this->name[$abstract] : $abstract;
        // 从容器实例中获取  如果存在则直接返回对应的实例  也就是使用注册树模式
        if (isset($this->instances[$abstract]) && !$newInstance) {
            return $this->instances[$abstract];
        }
        // think\App 从容器标识中获取
        // 第二次执行$this->bind['think\App']不存在走else
        if (isset($this->bind[$abstract])) {
            // 将think\App 复制给$concrete变量
            $concrete = $this->bind[$abstract];
            // 用于代表匿名函数的类  判断是不是闭包
            if ($concrete instanceof Closure) {
                $object = $this->invokeFunction($concrete, $vars);
            } else {
                // $this->name['app'] = think\App
                $this->name[$abstract] = $concrete;
                // 在执行一次本类的make方法,也就是本方法
                // think\App
                return $this->make($concrete, $vars, $newInstance);
            }
        } else {
            // think\App
            $object = $this->invokeClass($abstract, $vars);
        }
        if (!$newInstance) {
            // 把创建的容器存起来
            //$this->instances['think\App'] = $object;
            $this->instances[$abstract] = $object;
        }
        return $object;
    }
public function invokeClass($class, $vars = [])
    {
        try {
            /**
             * ReflectionClass Object
                (
                [name] => think\App
                )
             */
            // 这里就是之前文章提到的反射
            $reflect = new ReflectionClass($class);
            if ($reflect->hasMethod('__make')) {
                $method = new ReflectionMethod($class, '__make');
                if ($method->isPublic() && $method->isStatic()) {
                    $args = $this->bindParams($method, $vars);
                    return $method->invokeArgs(null, $args);
                }
            }
            // 通过反射获取think\App的构造函数
            $constructor = $reflect->getConstructor();
            $args = $constructor ? $this->bindParams($constructor, $vars) : [];
            // 从给出的参数创建一个新的类实例
            return $reflect->newInstanceArgs($args);
        } catch (ReflectionException $e) {
            throw new ClassNotFoundException('class not exists: ' . $class, $class);
        }
    }
相关文章
|
12天前
|
算法 测试技术 C语言
深入理解HTTP/2:nghttp2库源码解析及客户端实现示例
通过解析nghttp2库的源码和实现一个简单的HTTP/2客户端示例,本文详细介绍了HTTP/2的关键特性和nghttp2的核心实现。了解这些内容可以帮助开发者更好地理解HTTP/2协议,提高Web应用的性能和用户体验。对于实际开发中的应用,可以根据需要进一步优化和扩展代码,以满足具体需求。
62 29
|
9天前
|
前端开发 数据安全/隐私保护 CDN
二次元聚合短视频解析去水印系统源码
二次元聚合短视频解析去水印系统源码
27 3
|
11天前
|
JavaScript 算法 前端开发
JS数组操作方法全景图,全网最全构建完整知识网络!js数组操作方法全集(实现筛选转换、随机排序洗牌算法、复杂数据处理统计等情景详解,附大量源码和易错点解析)
这些方法提供了对数组的全面操作,包括搜索、遍历、转换和聚合等。通过分为原地操作方法、非原地操作方法和其他方法便于您理解和记忆,并熟悉他们各自的使用方法与使用范围。详细的案例与进阶使用,方便您理解数组操作的底层原理。链式调用的几个案例,让您玩转数组操作。 只有锻炼思维才能可持续地解决问题,只有思维才是真正值得学习和分享的核心要素。如果这篇博客能给您带来一点帮助,麻烦您点个赞支持一下,还可以收藏起来以备不时之需,有疑问和错误欢迎在评论区指出~
|
11天前
|
负载均衡 JavaScript 前端开发
分片上传技术全解析:原理、优势与应用(含简单实现源码)
分片上传通过将大文件分割成多个小的片段或块,然后并行或顺序地上传这些片段,从而提高上传效率和可靠性,特别适用于大文件的上传场景,尤其是在网络环境不佳时,分片上传能有效提高上传体验。 博客不应该只有代码和解决方案,重点应该在于给出解决方案的同时分享思维模式,只有思维才能可持续地解决问题,只有思维才是真正值得学习和分享的核心要素。如果这篇博客能给您带来一点帮助,麻烦您点个赞支持一下,还可以收藏起来以备不时之需,有疑问和错误欢迎在评论区指出~
|
4月前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
152 2
|
21天前
|
移动开发 前端开发 JavaScript
从入门到精通:H5游戏源码开发技术全解析与未来趋势洞察
H5游戏凭借其跨平台、易传播和开发成本低的优势,近年来发展迅猛。接下来,让我们深入了解 H5 游戏源码开发的技术教程以及未来的发展趋势。
|
19天前
|
存储 前端开发 JavaScript
在线教育网课系统源码开发指南:功能设计与技术实现深度解析
在线教育网课系统是近年来发展迅猛的教育形式的核心载体,具备用户管理、课程管理、教学互动、学习评估等功能。本文从功能和技术两方面解析其源码开发,涵盖前端(HTML5、CSS3、JavaScript等)、后端(Java、Python等)、流媒体及云计算技术,并强调安全性、稳定性和用户体验的重要性。
|
27天前
|
机器学习/深度学习 自然语言处理 算法
生成式 AI 大语言模型(LLMs)核心算法及源码解析:预训练篇
生成式 AI 大语言模型(LLMs)核心算法及源码解析:预训练篇
141 0
|
3月前
|
设计模式 存储 安全
【23种设计模式·全精解析 | 创建型模式篇】5种创建型模式的结构概述、实现、优缺点、扩展、使用场景、源码解析
创建型模式的主要关注点是“怎样创建对象?”,它的主要特点是"将对象的创建与使用分离”。这样可以降低系统的耦合度,使用者不需要关注对象的创建细节。创建型模式分为5种:单例模式、工厂方法模式抽象工厂式、原型模式、建造者模式。
【23种设计模式·全精解析 | 创建型模式篇】5种创建型模式的结构概述、实现、优缺点、扩展、使用场景、源码解析
|
3月前
|
存储 设计模式 算法
【23种设计模式·全精解析 | 行为型模式篇】11种行为型模式的结构概述、案例实现、优缺点、扩展对比、使用场景、源码解析
行为型模式用于描述程序在运行时复杂的流程控制,即描述多个类或对象之间怎样相互协作共同完成单个对象都无法单独完成的任务,它涉及算法与对象间职责的分配。行为型模式分为类行为模式和对象行为模式,前者采用继承机制来在类间分派行为,后者采用组合或聚合在对象间分配行为。由于组合关系或聚合关系比继承关系耦合度低,满足“合成复用原则”,所以对象行为模式比类行为模式具有更大的灵活性。 行为型模式分为: • 模板方法模式 • 策略模式 • 命令模式 • 职责链模式 • 状态模式 • 观察者模式 • 中介者模式 • 迭代器模式 • 访问者模式 • 备忘录模式 • 解释器模式
【23种设计模式·全精解析 | 行为型模式篇】11种行为型模式的结构概述、案例实现、优缺点、扩展对比、使用场景、源码解析

推荐镜像

更多