欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

Laravel框架中composer自动加载的实现分析

程序员文章站 2022-04-29 12:08:00
基础 自动加载允许你通过即用即加载的方式来加载需要的类文件,而不用每次都写繁琐的require 和include语句。因此,每一次请求的执行过程都只加载必须的类,也不不要...

基础

自动加载允许你通过即用即加载的方式来加载需要的类文件,而不用每次都写繁琐的require 和include语句。因此,每一次请求的执行过程都只加载必须的类,也不不要关心类的加载问题,只要需要的时候直接使用即可。

laravel 框架是通过composer 实现的自动加载。

是通过  下面的代码实现的。

require_once __dir__ . '/composer' . '/autoload_real.php';
return composerautoloaderinit7b20e4d61e2f88170fbbc44c70d38a1f::getloader();

首先我们对spl_autoload_register和spl_autoload_unregister 这两个函数进行解释一下。

spl_autoload_register 自动注册 一个或多个 自动加载函数,这些函数一般在 实例化类的时候,自动运行。

spl_autoload_unregister 恰恰相反。

贴上我实验的代码:

这是autoload.php

<?php
/**
 * created by phpstorm.
 * user: administrator
 * date: 2017/12/7
 * time: 14:10
 */
namespace app;
class autoload {

 public function __construct()
 {
  $this->autoload();
 }
 public function autoload(){
  // spl_autoload_register(array('autoload','ss'),true); 会触发致命错误,必须带上命名空间
  spl_autoload_register(array('app\autoload','ss'),true);
 }
 public function ss(){
  echo 666;
  exit;
 }
}

这是index.php

<?php
/**
 * created by phpstorm.
 * user: administrator
 * date: 2017/12/7
 * time: 14:10
 */
require 'autoload.php';
$autoload=new \app\autoload();
$b=new b();// 此时自动运行自动加载函数
echo 77;
exit;

找到getloader 这个函数,并对其进行分析:

 public static function getloader()
 {
  if (null !== self::$loader) {
   return self::$loader;
  }
  //注册自动加载函数,在加载或实例化类,运行loadclassloader函数
  spl_autoload_register(array('composerautoloaderinit7b20e4d61e2f88170fbbc44c70d38a1f', 'loadclassloader'), true, true);
  self::$loader = $loader = new \composer\autoload\classloader();
  spl_autoload_unregister(array('composerautoloaderinit7b20e4d61e2f88170fbbc44c70d38a1f', 'loadclassloader'));
/********************1********************************************************
  $map = require __dir__ . '/autoload_namespaces.php';
  foreach ($map as $namespace => $path) {
   $loader->set($namespace, $path);
  }
  $map = require __dir__ . '/autoload_psr4.php';
  foreach ($map as $namespace => $path) {
   $loader->setpsr4($namespace, $path);
  }
  $classmap = require __dir__ . '/autoload_classmap.php';
  if ($classmap) {
   $loader->addclassmap($classmap);
  }
/********************1********************************************************
  $loader->register(true);  $includefiles = require __dir__ . '/autoload_files.php';  foreach ($includefiles as $fileidentifier => $file) {   composerrequire7b20e4d61e2f88170fbbc44c70d38a1f($fileidentifier, $file);  }  return $loader; }}

/***** 包围的部分,主要对classloader 中的

$prefixespsr0   、$prefixdirspsr4  、$classmap 等属性进行赋值。即加载一些配置好的文件,在后面进行加载或寻找文件时候,就是从加载的配置文件中寻找。寻找要加载的类主要通过register 函数来实现。然后分析register函数。

public function register($prepend = false)
{
 spl_autoload_register(array($this, 'loadclass'), true, $prepend);
}

发现实际将该类中loadclass 函数注册为自动加载函数。于是开始分析loadclass函数,最终是通过findfile进行类的寻找。

public function findfile($class)
{
/// 特别注意 参数$class 是根据命名空间生成的class名称,具体请参考命名空间特性。
 // work around for php 5.3.0 - 5.3.2 https://bugs.php.net/50731
 if ('\\' == $class[0]) {
  $class = substr($class, 1);
 }
 // class map lookup 首先从加载的classmap 中寻找
 if (isset($this->classmap[$class])) {
  return $this->classmap[$class];
 }
 if ($this->classmapauthoritative) {
  return false;
 }
// 从刚才加载的配置文件中寻找文件。先按照 psr4 规则寻找,再按照psr0 寻找
// 两种规则的不同主要是对下划线的处理方式。
 $file = $this->findfilewithextension($class, '.php');
 // search for hack files if we are running on hhvm
 if ($file === null && defined('hhvm_version')) {
  $file = $this->findfilewithextension($class, '.hh');
 }
 if ($file === null) {
  // remember that this class does not exist.
  return $this->classmap[$class] = false;
 }
 return $file;
}

至此register函数分析完。我们接着分析getloader函数剩余代码。

$includefiles = require __dir__ . '/autoload_files.php';
foreach ($includefiles as $fileidentifier => $file) {
 composerrequire7b20e4d61e2f88170fbbc44c70d38a1f($fileidentifier, $file);
}

这段代码其实就是加载autoload_file.php 文件。

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对的支持。