关于php中的spl_autoload_register,splautoloadregister
一、自动加载定义
很多开发者写面向对象的应用程序时对每个类的定义建立一个 PHP 源文件。一个很大的烦恼是不得不在每个脚本开头写一个长长的包含文件列表(每个类一个文件)。
在 PHP 5 中,不再需要这样了。可以定义一个 __autoload() 函数,它会在试图使用尚未被定义的类时自动调用。通过调用此函数,脚本引擎在 PHP 出错失败前有了最后一个机会加载所需的类
__autoload 是一个魔术方法, 尝试加载未定义的类,spl_autoload_register - 注册给定的函数作为 __autoload 的实现
<p class="para"><span class="function"><span class="tip"><span class="tip">Tip</span></span></span></p> <p class="para"><span class="function">spl_autoload_register() 提供了一种更加灵活的方式来实现类的自动加载。因此,不再建议使用 <span class="function">__autoload()函数,在以后的版本中它可能被弃用。</span></span></p>
Note:
在 5.3.0 版之前,__autoload 函数抛出的异常不能被 catch 语句块捕获并会导致一个致命错误。从 5.3.0+ 之后,__autoload 函数抛出的异常可以被 catch 语句块捕获,但需要遵循一个条件。如果抛出的是一个自定义异常,那么必须存在相应的自定义异常类。__autoload 函数可以递归的自动加载自定义异常类。
Note:
自动加载不可用于 PHP 的 CLI 交互模式。
Note:
如果类名比如被用于 call_user_func(),则它可能包含一些危险的字符,比如 ../。 建议您在这样的函数中不要使用用户的输入,起码需要在 __autoload() 时验证下输入
二、spl_autoload_register
目录
classes --+ + mail.class.php + norman.class.php + db.class.php1、Single Autoloads
<?php /*** nullify any existing autoloads ***/ spl_autoload_register(null, false); /*** specify extensions that may be loaded ***/ spl_autoload_extensions('.php, .class.php'); /*** class Loader ***/ function classLoader($class) { $filename = strtolower($class) . '.class.php'; $file ='classes/' . $filename; if (!file_exists($file)) { return false; } include $file; } /*** register the loader functions ***/ spl_autoload_register('classLoader'); /*** a new instance if norman ***/ $norman = new norman; /*** make norman do something ***/ $norman->do_something(); ?>2、Mulitple Autoloads
<?php /*** nullify any existing autoloads ***/ spl_autoload_register(null, false); /*** specify extensions that may be loaded ***/ spl_autoload_extensions('.php, .class.php, .lib.php'); /*** class Loader ***/ function classLoader($class) { $filename = strtolower($class) . '.class.php'; $file ='classes/' . $filename; if (!file_exists($file)) { return false; } include $file; } function libLoader($class) { $filename = strtolower($class) . '.lib.php'; $file ='libs/' . $filename; if (!file_exists($file)) { return false; } include $file; } /*** register the loader functions ***/ spl_autoload_register('classLoader'); spl_autoload_register('libLoader'); /*** a new instance of norman ***/ $norman = new norman; /*** make norman do some thing ***/ $norman->do_something();3、Interfaces
接口文件
<?php /* * icontroller.class.php * interface to ensure all classes have an index method * */ interface iController { public function index(); } ?>autoload文件
<?php /*** nullify any existing autoloads ***/ spl_autoload_register(null, false); /*** specify extensions that may be loaded ***/ spl_autoload_extensions('.php, .class.php'); /*** class Loader ***/ function classLoader($class) { $filename = strtolower($class) . '.class.php'; $file ='classes/' . $filename; if (!file_exists($file)) { return false; } include $file; } /*** register the loader functions ***/ spl_autoload_register('classLoader'); class blog implements iController { public function index() { echo 'hello from the index'; } } /*** a new blog instance ***/ $blog = new blog; /*** run the index method ***/ $blog->index(); ?>4、一个标准的示例
spl_autoload_register( 'autoload' ); /** * autoload * * @author testdd * @param string $class * @param string $dir * @return bool */ function autoload( $class, $dir = null ) { if ( is_null( $dir ) ) $dir = '/path/to/project'; foreach ( scandir( $dir ) as $file ) { // directory? if ( is_dir( $dir.$file ) && substr( $file, 0, 1 ) !== '.' ) autoload( $class, $dir.$file.'/' ); // php file? if ( substr( $file, 0, 2 ) !== '._' && preg_match( "/.php$/i" , $file ) ) { // filename matches class? if ( str_replace( '.php', '', $file ) == $class || str_replace( '.class.php', '', $file ) == $class ) { include $dir . $file; } } } }5、框架中的写法
<?php /** * Autoloader * @author Jianxiang Qin <TalkativeDoggy@gmail.com> * @license http://opensource.org/licenses/BSD-3-Clause New BSD License * @version svn:$Id$ * @package Lotusphp\Autoloader */ /** * 自动加载类和函数 * * 按需加载类,每次只加载用到的类。 * * 函数库文件不是按需加载!若支持加载函数,则所有定义函数的文件都会加载。 * * 代码中用到一个类或者函数的时候,不需要使用include/require来包含类库文件或者函数库文件。 * * 基于Autoloader组件的代码中将不用使用include/require。 * * Autoloader缓存的是绝对路径,能让Opcode Cache有效缓存文件。 * * Autoloader要求类的名字唯一,不在意类文件的路径和文件名。目前不支持命名空间(PHP5.3) * * 传统的include/require通常存在以下问题。 * <ul> * <li>目录名和文件名变化引起程序代码变化。</li> * <li>Windows和Linux对文件路径的大小写和目录分隔符号的处理不同导致代码在不同平台迁移时出现问题。</li> * <li>include_path相对路径的性能低(显著地低)。</li> * <li>为了保证不重复包含,使用include_once和require_once导致效率低(不是显著的低)。</li> * </ul> * @author Jianxiang Qin <TalkativeDoggy@gmail.com> Yi Zhao <zhao5908@gmail.com> * @category runtime * @package Lotusphp\Autoloader * @todo 所有class-file mapping当成一个数据写入storeHandle */ class LtAutoloader { /** * @var bool true|false 是否自动加载定义了函数的文件。 * false 只自动加载定义了class或者interface的文件。 * true (默认) 自动加载定义了函数的文件。 */ public $isLoadFunction = true; /** * @var array 要扫描的文件类型 * 若该属性设置为array("php","inc","php3"), * 则扩展名为"php","inc","php3"的文件会被扫描, * 其它扩展名的文件会被忽略 */ public $allowFileExtension = array('php', 'inc'); /** * @var array 不扫描的目录 * 若该属性设置为array(".svn", ".setting"), * 则所有名为".setting"的目录也会被忽略 */ public $skipDirNames = array('.svn', '.git'); /** @var LtStoreFile 存储句柄默认使用 @link LtStoreFile */ public $storeHandle; /** @var array 指定需要自动加载的目录列表 */ public $autoloadPath; /** @var bool * true 开发模式下 每次都会扫描目录列表 * false 生产环境下 只扫描一次 */ public $devMode = true; /** @var array 函数名 -> 文件路径 映射 */ private $functionFileMapping = array(); /** @var array 类名 -> 文件路径 映射 */ private $classFileMapping = array(); /** @var array 定义了函数的文件列表 */ private $functionFiles = array(); /** @var LtStoreFile 持久存储句柄,存储文件的get_token_all分析结果/filesize/filehash @link LtStoreFile */ private $persistentStoreHandle; /** @var int store name space id */ private $storeNameSpaceId; /** @var int number of parse error */ private $parseErrorAmount = 0; /** @var int number of library files successfully parsed */ private $libFileAmount = 0; /** * 递归扫描指定的目录列表,根据@see LtAutoloader::$isLoadFunction是否加载全部的函数定义文件。 * 注册自动加载函数,按需加载类文件。 * @return void */ public function init() { $this->storeNameSpaceId = sprintf("%u", crc32(serialize($this->autoloadPath))); if (true != $this->devMode) { if ($this->storeHandle instanceof LtStore) { $this->storeHandle->prefix = 'Lt-Autoloader-' . $this->storeNameSpaceId; } else { if (null == $this->storeHandle) { $this->storeHandle = new LtStoreFile; $this->storeHandle->prefix = 'Lt-Autoloader-' . $this->storeNameSpaceId; $this->storeHandle->useSerialize = true; $this->storeHandle->init(); } else { trigger_error("You passed a value to autoloader::storeHandle, but it is NOT an instance of LtStore"); } } } else { $this->storeHandle = new LtStoreMemory; } // Whether scanning directory if ($storedMap = $this->storeHandle->get("map")) { $this->classFileMapping = $storedMap["classes"]; $this->functionFiles = $storedMap["functions"]; } else { $this->setPersistentStoreHandle(); $autoloadPath = $this->preparePath($this->autoloadPath); foreach($autoloadPath as $path) { if (is_file($path)) { $this->addFileMap($path); } } $this->scanDirs($autoloadPath); unset($autoloadPath); } // Whether loading function files $this->loadFunctionFiles(); spl_autoload_register(array($this, "loadClass")); } protected function setPersistentStoreHandle() { $this->persistentStoreHandle = new LtStoreFile; $this->persistentStoreHandle->prefix = 'Lt-parsed-token-' . $this->storeNameSpaceId; $this->persistentStoreHandle->useSerialize = true; } /** * Autoloader扫描项目,若某个php文件中定义了函数,则此文件的绝对路径被缓存, * 每次执行LtAutoloader->init()方法时,自动include所有定义了函数的php文件。 * 因为PHP的Autoload机制是针对Class的.function文件没有办法按需加载 * @return void */ protected function loadFunctionFiles() { if ($this->isLoadFunction && count($this->functionFiles)) { foreach ($this->functionFiles as $functionFile) { include_once($functionFile); } } } /** * 被注册的自动加载函数 * @param string $className * @return void */ protected function loadClass($className) { if ($filePath = $this->getFilePathByClassName($className)) { include($filePath); } } /** * 将目录分隔符号统一成linux目录分隔符号/ * @param string $path * @return boolean */ protected function convertPath($path) { $path = str_replace("\\", "/", $path); if (!is_readable($path)) { trigger_error("Directory is not exists/readable: {$path}"); return false; } $path = rtrim(realpath($path), '\\/'); if (preg_match("/\s/i", $path)) { trigger_error("Directory contains space/tab/newline is not supported: {$path}"); return false; } return $path; } /** * The string or an Multidimensional array into a one-dimensional array * 将字符串和多维数组转换成一维数组 * @param mixed $paths * @return array one-dimensional array */ protected function preparePath($paths) { $oneDPathArray = array(); if (!is_array($paths)) { $paths = array($paths); } $i = 0; while (isset($paths[$i])) { if (!is_array($paths[$i]) && $path = $this->convertPath($paths[$i])) { $oneDPathArray[] = $path; } else { foreach($paths[$i] as $v) { $paths[] = $v; } } $i ++; } unset($paths); return $oneDPathArray; } /** * Using iterative algorithm scanning subdirectories * save autoloader filemap * 递归扫描目录包含子目录,保存自动加载的文件地图。 * @param array $dirs one-dimensional * @return void * @todo in_array换成array_key_exists以提升性能 */ protected function scanDirs($dirs) { $i = 0; while (isset($dirs[$i])) { $dir = $dirs[$i]; $files = scandir($dir); foreach ($files as $file) { $currentFile = $dir . DIRECTORY_SEPARATOR . $file; if (is_file($currentFile)) { $this->addFileMap($currentFile); } else if (is_dir($currentFile)) { if (in_array($file, array(".", "..")) || in_array($file, $this->skipDirNames)) { continue; } else { // if $currentFile is a directory, pass through the next loop. $dirs[] = $currentFile; } } else { trigger_error("$currentFile is not a file or a directory."); } } //end foreach $i ++; } //end while if(0 == $this->parseErrorAmount) { $this->functionFiles = array_unique(array_values($this->functionFileMapping)); $map = array("classes" => $this->classFileMapping, "functions" => $this->functionFiles); if ($this->storeHandle->get("map")) { $this->storeHandle->update("map", $map); } else { $this->storeHandle->add("map", $map); } } else { trigger_error($this->parseErrorAmount . " error(s) occoured when scanning and parsing your lib files"); } } /** * 分析出字符串中的类,接口,函数。 * @param string $src * @return array * @todo 若当前文件包含了直接执行的php语句,或者html,输出警告 * @todo 若当前文件有语法错误,抛出异常 */ protected function parseLibNames($src) { $libNames = array(); $tokens = token_get_all($src); $level = 0; $found = false; $name = ''; foreach ($tokens as $token) { if (is_string($token)) { if ('{' == $token) { $level ++; } else if ('}' == $token) { $level --; } } else { list($id, $text) = $token; if (T_CURLY_OPEN == $id || T_DOLLAR_OPEN_CURLY_BRACES == $id) { $level ++; } if (0 < $level) { continue; } switch ($id) { case T_STRING: if ($found) { $libNames[strtolower($name)][] = $text; $found = false; } break; case T_CLASS: case T_INTERFACE: case T_FUNCTION: $found = true; $name = $text; break; } } } return $libNames; } /** * 保存类名、接口名和对应的文件绝对路径。 * @param string $className * @param string $file * @return boolean */ protected function addClass($className, $file) { $key = strtolower($className); if (isset($this->classFileMapping[$key])) { $existedClassFile = $this->classFileMapping[$key]; trigger_error("duplicate class [$className] found in:\n$existedClassFile\n$file\n, or please clear the cache"); return false; } else { $this->classFileMapping[$key] = $file; return true; } } /** * 保存函数名和对应的文件绝对路径 * @param string $functionName * @param string $file * @return boolean */ protected function addFunction($functionName, $file) { $functionName = strtolower($functionName); if (isset($this->functionFileMapping[$functionName])) { $existedFunctionFile = $this->functionFileMapping[$functionName]; trigger_error("duplicate function [$functionName] found in:\n$existedFunctionFile\n$file\n"); return false; } else { $this->functionFileMapping[$functionName] = $file; return true; } } /** * 将文件添加到自动加载的FileMap, * 添加之前会判断自从上次扫描后有没有修改,若没有修改则无需重复添加, * 若修改过,则分析文件内容,根据内容中包含的类、接口,函数添加到FileMap * @param string $filePath * @return boolean */ protected function addFileMap($filePath) { if (!in_array(pathinfo($filePath, PATHINFO_EXTENSION), $this->allowFileExtension)) {//init()会调用这个方法, 不要将这个判断移动到scanDir()中 return false; } $fileSize = filesize($filePath); $fileHash = md5_file($filePath); $savedFileInfo = $this->persistentStoreHandle->get($filePath); if (!isset($savedFileInfo['file_size']) || $savedFileInfo['file_size'] != $fileSize || $savedFileInfo['file_hash'] != $fileHash) { if($libNames = $this->parseLibNames(trim(file_get_contents($filePath)))) { $newFileInfo = array('file_size' => $fileSize, 'file_hash' => $fileHash, 'lib_names' => $libNames); if (isset($savedFileInfo['file_size'])) { $this->persistentStoreHandle->update($filePath, $newFileInfo); } else { $this->persistentStoreHandle->add($filePath, $newFileInfo); } } else { $this->parseErrorAmount ++; } } else { $libNames = $savedFileInfo['lib_names']; } foreach ($libNames as $libType => $libArray) { $method = "function" == $libType ? "addFunction" : "addClass"; foreach ($libArray as $libName) { if (!$this->$method($libName, $filePath)) { $this->parseErrorAmount ++; } } } return true; } protected function getFilePathByClassName($className) { $key = strtolower($className); if (isset($this->classFileMapping[$key])) { return $this->classFileMapping[$key]; } else { return false; } } }6、set_include_path 方式
set_include_path(implode(PATH_SEPARATOR, array(get_include_path(), './services', './printers'))); spl_autoload_register();7、PSR-4: Autoloader
http://www.php-fig.org/psr/psr-4/
参考文章
http://www.phpro.org/tutorials/SPL-Autoload.html
https://github.com/qinjx/adv_php_book/blob/master/class_autoload.md
http://php.net/manual/zh/language.oop5.autoload.php

PHP在現代Web開發中仍然重要,尤其在內容管理和電子商務平台。 1)PHP擁有豐富的生態系統和強大框架支持,如Laravel和Symfony。 2)性能優化可通過OPcache和Nginx實現。 3)PHP8.0引入JIT編譯器,提升性能。 4)雲原生應用通過Docker和Kubernetes部署,提高靈活性和可擴展性。

PHP適合web開發,特別是在快速開發和處理動態內容方面表現出色,但不擅長數據科學和企業級應用。與Python相比,PHP在web開發中更具優勢,但在數據科學領域不如Python;與Java相比,PHP在企業級應用中表現較差,但在web開發中更靈活;與JavaScript相比,PHP在後端開發中更簡潔,但在前端開發中不如JavaScript。

PHP和Python各有優勢,適合不同場景。 1.PHP適用於web開發,提供內置web服務器和豐富函數庫。 2.Python適合數據科學和機器學習,語法簡潔且有強大標準庫。選擇時應根據項目需求決定。

PHP是一種廣泛應用於服務器端的腳本語言,特別適合web開發。 1.PHP可以嵌入HTML,處理HTTP請求和響應,支持多種數據庫。 2.PHP用於生成動態網頁內容,處理表單數據,訪問數據庫等,具有強大的社區支持和開源資源。 3.PHP是解釋型語言,執行過程包括詞法分析、語法分析、編譯和執行。 4.PHP可以與MySQL結合用於用戶註冊系統等高級應用。 5.調試PHP時,可使用error_reporting()和var_dump()等函數。 6.優化PHP代碼可通過緩存機制、優化數據庫查詢和使用內置函數。 7

PHP成為許多網站首選技術棧的原因包括其易用性、強大社區支持和廣泛應用。 1)易於學習和使用,適合初學者。 2)擁有龐大的開發者社區,資源豐富。 3)廣泛應用於WordPress、Drupal等平台。 4)與Web服務器緊密集成,簡化開發部署。

PHP在現代編程中仍然是一個強大且廣泛使用的工具,尤其在web開發領域。 1)PHP易用且與數據庫集成無縫,是許多開發者的首選。 2)它支持動態內容生成和麵向對象編程,適合快速創建和維護網站。 3)PHP的性能可以通過緩存和優化數據庫查詢來提升,其廣泛的社區和豐富生態系統使其在當今技術棧中仍具重要地位。

在PHP中,弱引用是通過WeakReference類實現的,不會阻止垃圾回收器回收對象。弱引用適用於緩存系統和事件監聽器等場景,需注意其不能保證對象存活,且垃圾回收可能延遲。

\_\_invoke方法允許對象像函數一樣被調用。 1.定義\_\_invoke方法使對象可被調用。 2.使用$obj(...)語法時,PHP會執行\_\_invoke方法。 3.適用於日誌記錄和計算器等場景,提高代碼靈活性和可讀性。


熱AI工具

Undresser.AI Undress
人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover
用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool
免費脫衣圖片

Clothoff.io
AI脫衣器

AI Hentai Generator
免費產生 AI 無盡。

熱門文章

熱工具

mPDF
mPDF是一個PHP庫,可以從UTF-8編碼的HTML產生PDF檔案。原作者Ian Back編寫mPDF以從他的網站上「即時」輸出PDF文件,並處理不同的語言。與原始腳本如HTML2FPDF相比,它的速度較慢,並且在使用Unicode字體時產生的檔案較大,但支援CSS樣式等,並進行了大量增強。支援幾乎所有語言,包括RTL(阿拉伯語和希伯來語)和CJK(中日韓)。支援嵌套的區塊級元素(如P、DIV),

SecLists
SecLists是最終安全測試人員的伙伴。它是一個包含各種類型清單的集合,這些清單在安全評估過程中經常使用,而且都在一個地方。 SecLists透過方便地提供安全測試人員可能需要的所有列表,幫助提高安全測試的效率和生產力。清單類型包括使用者名稱、密碼、URL、模糊測試有效載荷、敏感資料模式、Web shell等等。測試人員只需將此儲存庫拉到新的測試機上,他就可以存取所需的每種類型的清單。

EditPlus 中文破解版
體積小,語法高亮,不支援程式碼提示功能

SublimeText3 Linux新版
SublimeText3 Linux最新版

Dreamweaver Mac版
視覺化網頁開發工具