亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁(yè) > 語(yǔ)言 > PHP > 正文

PHP命名空間與自動(dòng)加載機(jī)制的基礎(chǔ)介紹

2024-05-05 00:11:04
字體:
來(lái)源:轉(zhuǎn)載
供稿:網(wǎng)友

前言

include 和 require 是PHP中引入文件的兩個(gè)基本方法。在小規(guī)模開(kāi)發(fā)中直接使用 include 和 require 沒(méi)喲什么不妥,但在大型項(xiàng)目中會(huì)造成大量的 include 和 require 堆積。這樣的代碼既不優(yōu)雅,執(zhí)行效率也很低,而且維護(hù)起來(lái)也相當(dāng)困難。

為了解決這個(gè)問(wèn)題,部分框架會(huì)給出一個(gè)引入文件的配置清單,在對(duì)象初始化的時(shí)候把需要的文件引入。但這只是讓代碼變得更簡(jiǎn)潔了一些,引入的效果仍然是差強(qiáng)人意。PHP5 之后,隨著 PHP 面向?qū)ο笾С值耐晟?,__autoload 函數(shù)才真正使得自動(dòng)加載成為可能。

* include 和 require 功能是一樣的,它們的不同在于 include 出錯(cuò)時(shí)只會(huì)產(chǎn)生警告,而 require 會(huì)拋出錯(cuò)誤終止腳本。

* include_once 和 include 唯一的區(qū)別在于 include_once 會(huì)檢查文件是否已經(jīng)引入,如果是則不會(huì)重復(fù)引入。

=================自動(dòng)加載==================

實(shí)現(xiàn)自動(dòng)加載最簡(jiǎn)單的方式就是使用 __autoload 魔術(shù)方法。當(dāng)需要使用的類沒(méi)有被引入時(shí),這個(gè)函數(shù)會(huì)在PHP報(bào)錯(cuò)前被觸發(fā),未定義的類名會(huì)被當(dāng)作參數(shù)傳入。至于函數(shù)具體的邏輯,這需要用戶自己去實(shí)現(xiàn)。

首先創(chuàng)建一個(gè) autoload.php 來(lái)做一個(gè)簡(jiǎn)單的測(cè)試:

// 類未定義時(shí),系統(tǒng)自動(dòng)調(diào)用function __autoload($class){ /* 具體處理邏輯 */ echo $class;// 簡(jiǎn)單的輸出未定義的類名}new HelloWorld();/** * 輸出 HelloWorld 與報(bào)錯(cuò)信息 * Fatal error: Class 'HelloWorld' not found */

通過(guò)這個(gè)簡(jiǎn)單的例子可以發(fā)現(xiàn),在類的實(shí)例化過(guò)程中,系統(tǒng)所做的工作大致是這樣的:

/* 模擬系統(tǒng)實(shí)例化過(guò)程 */function instance($class){ // 如果類存在則返回其實(shí)例 if (class_exists($class, false)) {  return new $class(); } // 查看 autoload 函數(shù)是否被用戶定義 if (function_exists('__autoload')) {  __autoload($class); // 最后一次引入的機(jī)會(huì) } // 再次檢查類是否存在 if (class_exists($class, false)) {  return new $class(); } else { // 系統(tǒng):我實(shí)在沒(méi)轍了  throw new Exception('Class Not Found'); }}

明白了 __autoload 函數(shù)的工作原理之后,那就讓我們來(lái)用它去實(shí)現(xiàn)自動(dòng)加載。

首先創(chuàng)建一個(gè)類文件(建議文件名與類名一致),代碼如下:

class [ClassName] { // 對(duì)象實(shí)例化時(shí)輸出當(dāng)前類名 function __construct() {  echo '<h1>' . __CLASS__ . '</h1>'; }}

(我這里創(chuàng)建了一個(gè) HelloWorld 類用作演示)接下來(lái)我們就要定義 __autoload 的具體邏輯,使它能夠?qū)崿F(xiàn)自動(dòng)加載:

function __autoload($class){ // 根據(jù)類名確定文件名 $file = $class . '.php'; if (file_exists($file)) {  include $file; // 引入PHP文件 }}new HelloWorld();/** * 輸出 <h1>HelloWorld</h1> */

=================命名空間==================

其實(shí)命名空間并不是什么新生事物,很多語(yǔ)言(例如C++)早都支持這個(gè)特性了。只不過(guò) PHP 起步比較晚,直到 PHP 5.3 之后才支持。

命名空間簡(jiǎn)而言之就是一種標(biāo)識(shí),它的主要目的是解決命名沖突的問(wèn)題。

就像在日常生活中,有很多姓名相同的人,如何區(qū)分這些人呢?那就需要加上一些額外的標(biāo)識(shí)。

把工作單位當(dāng)成標(biāo)識(shí)似乎不錯(cuò),這樣就不用擔(dān)心 “撞名” 的尷尬了。

這里我們來(lái)做一個(gè)小任務(wù),去介紹百度的CEO李彥宏:

namespace 百度;class 李彥宏{ function __construct() {  echo '百度創(chuàng)始人'; }}

↑ 這就是李彥宏的基本資料了,namespace 是他的單位標(biāo)識(shí),class 是他的姓名。

命名空間通過(guò)關(guān)鍵字 namespace 來(lái)聲明。如果一個(gè)文件中包含命名空間,它必須在其它所有代碼之前聲明命名空間。

new 百度/李彥宏(); // 限定類名new /百度/李彥宏(); // 完全限定類名

↑ 在一般情況下,無(wú)論是向別人介紹 "百度 李彥宏" 還是 "百度公司 李彥宏",他們都能夠明白。

在當(dāng)前命名空間沒(méi)有聲明的情況下,限定類名和完全限定類名是等價(jià)的。因?yàn)槿绻恢付臻g,則默認(rèn)為全局(/)。

namespace 谷歌;new 百度/李彥宏(); // 谷歌/百度/李彥宏(實(shí)際結(jié)果)new /百度/李彥宏(); // 百度/李彥宏(實(shí)際結(jié)果)

↑ 如果你在谷歌公司向他們的員工介紹李彥宏,一定要指明是 "百度公司的李彥宏"。否則他會(huì)認(rèn)為百度是谷歌的一個(gè)部門(mén),而李彥宏只是其中的一位員工而已。

這個(gè)例子展示了在命名空間下,使用限定類名和完全限定類名的區(qū)別。(完全限定類名 = 當(dāng)前命名空間 + 限定類名)

/* 導(dǎo)入命名空間 */use 百度/李彥宏;new 李彥宏(); // 百度/李彥宏(實(shí)際結(jié)果)/* 設(shè)置別名 */use 百度/李彥宏 AS CEO;new CEO(); // 百度/李彥宏(實(shí)際結(jié)果)/* 任何情況 */new /百度/李彥宏();// 百度/李彥宏(實(shí)際結(jié)果)

↑ 第一種情況是別人已經(jīng)認(rèn)識(shí)李彥宏了,你只需要直接說(shuō)名字,他就能知道你指的是誰(shuí)。第二種情況是李彥宏就是他們的CEO,你直接說(shuō)CEO,他可以立刻反應(yīng)過(guò)來(lái)。

使用命名空間只是讓類名有了前綴,不容易發(fā)生沖突,系統(tǒng)仍然不會(huì)進(jìn)行自動(dòng)導(dǎo)入。

如果不引入文件,系統(tǒng)會(huì)在拋出 "Class Not Found" 錯(cuò)誤之前觸發(fā) __autoload 函數(shù),并將限定類名傳入作為參數(shù)。

所以上面的例子都是基于你已經(jīng)將相關(guān)文件手動(dòng)引入的情況下實(shí)現(xiàn)的,否則系統(tǒng)會(huì)拋出 " Class '百度/李彥宏' not found"。

=================spl_autoload==================

接下來(lái)讓我們要在含有命名空間的情況下去實(shí)現(xiàn)自動(dòng)加載。這里我們使用 spl_autoload_register() 函數(shù)來(lái)實(shí)現(xiàn),這需要你的 PHP 版本號(hào)大于 5.12。

spl_autoload_register 函數(shù)的功能就是把傳入的函數(shù)(參數(shù)可以為回調(diào)函數(shù)或函數(shù)名稱形式)注冊(cè)到 SPL __autoload 函數(shù)隊(duì)列中,并移除系統(tǒng)默認(rèn)的 __autoload() 函數(shù)。

一旦調(diào)用 spl_autoload_register() 函數(shù),當(dāng)調(diào)用未定義類時(shí),系統(tǒng)就會(huì)按順序調(diào)用注冊(cè)到 spl_autoload_register() 函數(shù)的所有函數(shù),而不是自動(dòng)調(diào)用 __autoload() 函數(shù)。

現(xiàn)在,我們來(lái)創(chuàng)建一個(gè) Linux 類,它使用 os 作為它的命名空間(建議文件名與類名保持一致):

namespace os; // 命名空間class Linux // 類名{ function __construct() {  echo '<h1>' . __CLASS__ . '</h1>'; }}

接著,在同一個(gè)目錄下新建一個(gè) PHP 文件,使用 spl_autoload_register 以函數(shù)回調(diào)的方式實(shí)現(xiàn)自動(dòng)加載:

spl_autoload_register(function ($class) { // class = os/Linux /* 限定類名路徑映射 */ $class_map = array(  // 限定類名 => 文件路徑  'os//Linux' => './Linux.php', ); /* 根據(jù)類名確定文件名 */ $file = $class_map[$class]; /* 引入相關(guān)文件 */ if (file_exists($file)) {  include $file; }});new /os/Linux();

這里我們使用了一個(gè)數(shù)組去保存類名與文件路徑的關(guān)系,這樣當(dāng)類名傳入時(shí),自動(dòng)加載器就知道該引入哪個(gè)文件去加載這個(gè)類了。

但是一旦文件多起來(lái)的話,映射數(shù)組會(huì)變得很長(zhǎng),這樣的話維護(hù)起來(lái)會(huì)相當(dāng)麻煩。如果命名能遵守統(tǒng)一的約定,就可以讓自動(dòng)加載器自動(dòng)解析判斷類文件所在的路徑。接下來(lái)要介紹的PSR-4 就是一種被廣泛采用的約定方式。

=================PSR-4規(guī)范==================

PSR-4 是關(guān)于由文件路徑自動(dòng)載入對(duì)應(yīng)類的相關(guān)規(guī)范,規(guī)范規(guī)定了一個(gè)完全限定類名需要具有以下結(jié)構(gòu):

/<頂級(jí)命名空間>(/<子命名空間>)*/<類名>

如果繼續(xù)拿上面的例子打比方的話,頂級(jí)命名空間相當(dāng)于公司,子命名空間相當(dāng)于職位,類名相當(dāng)于人名。那么李彥宏標(biāo)準(zhǔn)的稱呼為 "百度公司 CEO 李彥宏"。

PSR-4 規(guī)范中必須要有一個(gè)頂級(jí)命名空間,它的意義在于表示某一個(gè)特殊的目錄(文件基目錄)。子命名空間代表的是類文件相對(duì)于文件基目錄的這一段路徑(相對(duì)路徑),類名則與文件名保持一致(注意大小寫(xiě)的區(qū)別)。

舉個(gè)例子:在全限定類名 /app/view/news/Index 中,如果 app 代表 C:/Baidu,那么這個(gè)類的路徑則是 C:/Baidu/view/news/Index.php

我們就以解析 /app/view/news/Index 為例,編寫(xiě)一個(gè)簡(jiǎn)單的 Demo:

$class = 'app/view/news/Index';/* 頂級(jí)命名空間路徑映射 */$vendor_map = array(  'app' => 'C:/Baidu',);/* 解析類名為文件路徑 */$vendor = substr($class, 0, strpos($class, '//')); // 取出頂級(jí)命名空間[app]$vendor_dir = $vendor_map[$vendor]; // 文件基目錄[C:/Baidu]$rel_path = dirname(substr($class, strlen($vendor))); // 相對(duì)路徑[/view/news]$file_name = basename($class) . '.php'; // 文件名[Index.php]/* 輸出文件所在路徑 */echo $vendor_dir . $rel_path . DIRECTORY_SEPARATOR . $file_name;

通過(guò)這個(gè) Demo 可以看出限定類名轉(zhuǎn)換為路徑的過(guò)程。那么現(xiàn)在就讓我們用規(guī)范的面向?qū)ο蠓绞饺?shí)現(xiàn)自動(dòng)加載器吧。

首先我們創(chuàng)建一個(gè)文件 Index.php,它處于 /app/mvc/view/home 目錄中:

namespace app/mvc/view/home;class Index{  function __construct()  {    echo '<h1> Welcome To Home </h1>';  }}

接著我們?cè)趧?chuàng)建一個(gè)加載類(不需要命名空間),它處于 / 目錄中:

class Loader{  /* 路徑映射 */  public static $vendorMap = array(    'app' => __DIR__ . DIRECTORY_SEPARATOR . 'app',  );  /**   * 自動(dòng)加載器   */  public static function autoload($class)  {    $file = self::findFile($class);    if (file_exists($file)) {      self::includeFile($file);    }  }  /**   * 解析文件路徑   */  private static function findFile($class)  {    $vendor = substr($class, 0, strpos($class, '//')); // 頂級(jí)命名空間    $vendorDir = self::$vendorMap[$vendor]; // 文件基目錄    $filePath = substr($class, strlen($vendor)) . '.php'; // 文件相對(duì)路徑    return strtr($vendorDir . $filePath, '//', DIRECTORY_SEPARATOR); // 文件標(biāo)準(zhǔn)路徑  }  /**   * 引入文件   */  private static function includeFile($file)  {    if (is_file($file)) {      include $file;    }  }}

最后,將 Loader 類中的 autoload 注冊(cè)到 spl_autoload_register 函數(shù)中:

include 'Loader.php'; // 引入加載器spl_autoload_register('Loader::autoload'); // 注冊(cè)自動(dòng)加載new /app/mvc/view/home/Index(); // 實(shí)例化未引用的類/** * 輸出: <h1> Welcome To Home </h1> */

示例中的代碼其實(shí)就是 ThinkPHP 自動(dòng)加載器源碼的精簡(jiǎn)版,它是 ThinkPHP 5 能實(shí)現(xiàn)惰性加載的關(guān)鍵。

至此,自動(dòng)加載的原理已經(jīng)全部講完了,如果有興趣深入了解的話,可以參考下面的 ThinkPHP 源碼。

class Loader{  protected static $instance = [];  // 類名映射  protected static $map = [];  // 命名空間別名  protected static $namespaceAlias = [];  // PSR-4  private static $prefixLengthsPsr4 = [];  private static $prefixDirsPsr4  = [];  private static $fallbackDirsPsr4 = [];  // PSR-0  private static $prefixesPsr0   = [];  private static $fallbackDirsPsr0 = [];  // 自動(dòng)加載的文件  private static $autoloadFiles = [];  // 自動(dòng)加載  public static function autoload($class)  {    // 檢測(cè)命名空間別名    if (!empty(self::$namespaceAlias)) {      $namespace = dirname($class);      if (isset(self::$namespaceAlias[$namespace])) {        $original = self::$namespaceAlias[$namespace] . '//' . basename($class);        if (class_exists($original)) {          return class_alias($original, $class, false);        }      }    }    if ($file = self::findFile($class)) {      // Win環(huán)境嚴(yán)格區(qū)分大小寫(xiě)      if (IS_WIN && pathinfo($file, PATHINFO_FILENAME) != pathinfo(realpath($file), PATHINFO_FILENAME)) {        return false;      }      __include_file($file);      return true;    }  }  /**   * 查找文件   * @param $class   * @return bool   */  private static function findFile($class)  {    if (!empty(self::$map[$class])) {      // 類庫(kù)映射      return self::$map[$class];    }    // 查找 PSR-4    $logicalPathPsr4 = strtr($class, '//', DS) . EXT;    $first = $class[0];    if (isset(self::$prefixLengthsPsr4[$first])) {      foreach (self::$prefixLengthsPsr4[$first] as $prefix => $length) {        if (0 === strpos($class, $prefix)) {          foreach (self::$prefixDirsPsr4[$prefix] as $dir) {            if (is_file($file = $dir . DS . substr($logicalPathPsr4, $length))) {              return $file;            }          }        }      }    }    // 查找 PSR-4 fallback dirs    foreach (self::$fallbackDirsPsr4 as $dir) {      if (is_file($file = $dir . DS . $logicalPathPsr4)) {        return $file;      }    }    // 查找 PSR-0    if (false !== $pos = strrpos($class, '//')) {      // namespaced class name      $logicalPathPsr0 = substr($logicalPathPsr4, 0, $pos + 1)      . strtr(substr($logicalPathPsr4, $pos + 1), '_', DS);    } else {      // PEAR-like class name      $logicalPathPsr0 = strtr($class, '_', DS) . EXT;    }    if (isset(self::$prefixesPsr0[$first])) {      foreach (self::$prefixesPsr0[$first] as $prefix => $dirs) {        if (0 === strpos($class, $prefix)) {          foreach ($dirs as $dir) {            if (is_file($file = $dir . DS . $logicalPathPsr0)) {              return $file;            }          }        }      }    }    // 查找 PSR-0 fallback dirs    foreach (self::$fallbackDirsPsr0 as $dir) {      if (is_file($file = $dir . DS . $logicalPathPsr0)) {        return $file;      }    }    return self::$map[$class] = false;  }  // 注冊(cè)classmap  public static function addClassMap($class, $map = '')  {    if (is_array($class)) {      self::$map = array_merge(self::$map, $class);    } else {      self::$map[$class] = $map;    }  }  // 注冊(cè)命名空間  public static function addNamespace($namespace, $path = '')  {    if (is_array($namespace)) {      foreach ($namespace as $prefix => $paths) {        self::addPsr4($prefix . '//', rtrim($paths, DS), true);      }    } else {      self::addPsr4($namespace . '//', rtrim($path, DS), true);    }  }  // 添加Ps0空間  private static function addPsr0($prefix, $paths, $prepend = false)  {    if (!$prefix) {      if ($prepend) {        self::$fallbackDirsPsr0 = array_merge(          (array) $paths,          self::$fallbackDirsPsr0        );      } else {        self::$fallbackDirsPsr0 = array_merge(          self::$fallbackDirsPsr0,          (array) $paths        );      }      return;    }    $first = $prefix[0];    if (!isset(self::$prefixesPsr0[$first][$prefix])) {      self::$prefixesPsr0[$first][$prefix] = (array) $paths;      return;    }    if ($prepend) {      self::$prefixesPsr0[$first][$prefix] = array_merge(        (array) $paths,        self::$prefixesPsr0[$first][$prefix]      );    } else {      self::$prefixesPsr0[$first][$prefix] = array_merge(        self::$prefixesPsr0[$first][$prefix],        (array) $paths      );    }  }  // 添加Psr4空間  private static function addPsr4($prefix, $paths, $prepend = false)  {    if (!$prefix) {      // Register directories for the root namespace.      if ($prepend) {        self::$fallbackDirsPsr4 = array_merge(          (array) $paths,          self::$fallbackDirsPsr4        );      } else {        self::$fallbackDirsPsr4 = array_merge(          self::$fallbackDirsPsr4,          (array) $paths        );      }    } elseif (!isset(self::$prefixDirsPsr4[$prefix])) {      // Register directories for a new namespace.      $length = strlen($prefix);      if ('//' !== $prefix[$length - 1]) {        throw new /InvalidArgumentException("A non-empty PSR-4 prefix must end with a namespace separator.");      }      self::$prefixLengthsPsr4[$prefix[0]][$prefix] = $length;      self::$prefixDirsPsr4[$prefix]        = (array) $paths;    } elseif ($prepend) {      // Prepend directories for an already registered namespace.      self::$prefixDirsPsr4[$prefix] = array_merge(        (array) $paths,        self::$prefixDirsPsr4[$prefix]      );    } else {      // Append directories for an already registered namespace.      self::$prefixDirsPsr4[$prefix] = array_merge(        self::$prefixDirsPsr4[$prefix],        (array) $paths      );    }  }  // 注冊(cè)命名空間別名  public static function addNamespaceAlias($namespace, $original = '')  {    if (is_array($namespace)) {      self::$namespaceAlias = array_merge(self::$namespaceAlias, $namespace);    } else {      self::$namespaceAlias[$namespace] = $original;    }  }  // 注冊(cè)自動(dòng)加載機(jī)制  public static function register($autoload = '')  {    // 注冊(cè)系統(tǒng)自動(dòng)加載    spl_autoload_register($autoload ?: 'think//Loader::autoload', true, true);    // 注冊(cè)命名空間定義    self::addNamespace([      'think'  => LIB_PATH . 'think' . DS,      'behavior' => LIB_PATH . 'behavior' . DS,      'traits'  => LIB_PATH . 'traits' . DS,    ]);    // 加載類庫(kù)映射文件    if (is_file(RUNTIME_PATH . 'classmap' . EXT)) {      self::addClassMap(__include_file(RUNTIME_PATH . 'classmap' . EXT));    }    // Composer自動(dòng)加載支持    if (is_dir(VENDOR_PATH . 'composer')) {      self::registerComposerLoader();    }    // 自動(dòng)加載extend目錄    self::$fallbackDirsPsr4[] = rtrim(EXTEND_PATH, DS);  }  // 注冊(cè)composer自動(dòng)加載  private static function registerComposerLoader()  {    if (is_file(VENDOR_PATH . 'composer/autoload_namespaces.php')) {      $map = require VENDOR_PATH . 'composer/autoload_namespaces.php';      foreach ($map as $namespace => $path) {        self::addPsr0($namespace, $path);      }    }    if (is_file(VENDOR_PATH . 'composer/autoload_psr4.php')) {      $map = require VENDOR_PATH . 'composer/autoload_psr4.php';      foreach ($map as $namespace => $path) {        self::addPsr4($namespace, $path);      }    }    if (is_file(VENDOR_PATH . 'composer/autoload_classmap.php')) {      $classMap = require VENDOR_PATH . 'composer/autoload_classmap.php';      if ($classMap) {        self::addClassMap($classMap);      }    }    if (is_file(VENDOR_PATH . 'composer/autoload_files.php')) {      $includeFiles = require VENDOR_PATH . 'composer/autoload_files.php';      foreach ($includeFiles as $fileIdentifier => $file) {        if (empty(self::$autoloadFiles[$fileIdentifier])) {          __require_file($file);          self::$autoloadFiles[$fileIdentifier] = true;        }      }    }  }  /**   * 導(dǎo)入所需的類庫(kù) 同java的Import 本函數(shù)有緩存功能   * @param string $class  類庫(kù)命名空間字符串   * @param string $baseUrl 起始路徑   * @param string $ext   導(dǎo)入的文件擴(kuò)展名   * @return boolean   */  public static function import($class, $baseUrl = '', $ext = EXT)  {    static $_file = [];    $key     = $class . $baseUrl;    $class    = str_replace(['.', '#'], [DS, '.'], $class);    if (isset($_file[$key])) {      return true;    }    if (empty($baseUrl)) {      list($name, $class) = explode(DS, $class, 2);      if (isset(self::$prefixDirsPsr4[$name . '//'])) {        // 注冊(cè)的命名空間        $baseUrl = self::$prefixDirsPsr4[$name . '//'];      } elseif ('@' == $name) {        //加載當(dāng)前模塊應(yīng)用類庫(kù)        $baseUrl = App::$modulePath;      } elseif (is_dir(EXTEND_PATH . $name)) {        $baseUrl = EXTEND_PATH;      } else {        // 加載其它模塊的類庫(kù)        $baseUrl = APP_PATH . $name . DS;      }    } elseif (substr($baseUrl, -1) != DS) {      $baseUrl .= DS;    }    // 如果類存在 則導(dǎo)入類庫(kù)文件    if (is_array($baseUrl)) {      foreach ($baseUrl as $path) {        $filename = $path . DS . $class . $ext;        if (is_file($filename)) {          break;        }      }    } else {      $filename = $baseUrl . $class . $ext;    }    if (!empty($filename) && is_file($filename)) {      // 開(kāi)啟調(diào)試模式Win環(huán)境嚴(yán)格區(qū)分大小寫(xiě)      if (IS_WIN && pathinfo($filename, PATHINFO_FILENAME) != pathinfo(realpath($filename), PATHINFO_FILENAME)) {        return false;      }      __include_file($filename);      $_file[$key] = true;      return true;    }    return false;  }  /**   * 實(shí)例化(分層)模型   * @param string $name     Model名稱   * @param string $layer    業(yè)務(wù)層名稱   * @param bool  $appendSuffix 是否添加類名后綴   * @param string $common    公共模塊名   * @return Object   * @throws ClassNotFoundException   */  public static function model($name = '', $layer = 'model', $appendSuffix = false, $common = 'common')  {    if (isset(self::$instance[$name . $layer])) {      return self::$instance[$name . $layer];    }    if (strpos($name, '/')) {      list($module, $name) = explode('/', $name, 2);    } else {      $module = Request::instance()->module();    }    $class = self::parseClass($module, $layer, $name, $appendSuffix);    if (class_exists($class)) {      $model = new $class();    } else {      $class = str_replace('//' . $module . '//', '//' . $common . '//', $class);      if (class_exists($class)) {        $model = new $class();      } else {        throw new ClassNotFoundException('class not exists:' . $class, $class);      }    }    self::$instance[$name . $layer] = $model;    return $model;  }  /**   * 實(shí)例化(分層)控制器 格式:[模塊名/]控制器名   * @param string $name     資源地址   * @param string $layer    控制層名稱   * @param bool  $appendSuffix 是否添加類名后綴   * @param string $empty    空控制器名稱   * @return Object|false   * @throws ClassNotFoundException   */  public static function controller($name, $layer = 'controller', $appendSuffix = false, $empty = '')  {    if (strpos($name, '/')) {      list($module, $name) = explode('/', $name);    } else {      $module = Request::instance()->module();    }    $class = self::parseClass($module, $layer, $name, $appendSuffix);    if (class_exists($class)) {      return new $class(Request::instance());    } elseif ($empty && class_exists($emptyClass = self::parseClass($module, $layer, $empty, $appendSuffix))) {      return new $emptyClass(Request::instance());    }  }  /**   * 實(shí)例化驗(yàn)證類 格式:[模塊名/]驗(yàn)證器名   * @param string $name     資源地址   * @param string $layer    驗(yàn)證層名稱   * @param bool  $appendSuffix 是否添加類名后綴   * @param string $common    公共模塊名   * @return Object|false   * @throws ClassNotFoundException   */  public static function validate($name = '', $layer = 'validate', $appendSuffix = false, $common = 'common')  {    $name = $name ?: Config::get('default_validate');    if (empty($name)) {      return new Validate;    }    if (isset(self::$instance[$name . $layer])) {      return self::$instance[$name . $layer];    }    if (strpos($name, '/')) {      list($module, $name) = explode('/', $name);    } else {      $module = Request::instance()->module();    }    $class = self::parseClass($module, $layer, $name, $appendSuffix);    if (class_exists($class)) {      $validate = new $class;    } else {      $class = str_replace('//' . $module . '//', '//' . $common . '//', $class);      if (class_exists($class)) {        $validate = new $class;      } else {        throw new ClassNotFoundException('class not exists:' . $class, $class);      }    }    self::$instance[$name . $layer] = $validate;    return $validate;  }  /**   * 數(shù)據(jù)庫(kù)初始化 并取得數(shù)據(jù)庫(kù)類實(shí)例   * @param mixed     $config 數(shù)據(jù)庫(kù)配置   * @param bool|string  $name 連接標(biāo)識(shí) true 強(qiáng)制重新連接   * @return /think/db/Connection   */  public static function db($config = [], $name = false)  {    return Db::connect($config, $name);  }  /**   * 遠(yuǎn)程調(diào)用模塊的操作方法 參數(shù)格式 [模塊/控制器/]操作   * @param string    $url     調(diào)用地址   * @param string|array $vars     調(diào)用參數(shù) 支持字符串和數(shù)組   * @param string    $layer    要調(diào)用的控制層名稱   * @param bool     $appendSuffix 是否添加類名后綴   * @return mixed   */  public static function action($url, $vars = [], $layer = 'controller', $appendSuffix = false)  {    $info  = pathinfo($url);    $action = $info['basename'];    $module = '.' != $info['dirname'] ? $info['dirname'] : Request::instance()->controller();    $class = self::controller($module, $layer, $appendSuffix);    if ($class) {      if (is_scalar($vars)) {        if (strpos($vars, '=')) {          parse_str($vars, $vars);        } else {          $vars = [$vars];        }      }      return App::invokeMethod([$class, $action . Config::get('action_suffix')], $vars);    }  }  /**   * 字符串命名風(fēng)格轉(zhuǎn)換   * type 0 將Java風(fēng)格轉(zhuǎn)換為C的風(fēng)格 1 將C風(fēng)格轉(zhuǎn)換為Java的風(fēng)格   * @param string $name 字符串   * @param integer $type 轉(zhuǎn)換類型   * @return string   */  public static function parseName($name, $type = 0)  {    if ($type) {      return ucfirst(preg_replace_callback('/_([a-zA-Z])/', function ($match) {        return strtoupper($match[1]);      }, $name));    } else {      return strtolower(trim(preg_replace("/[A-Z]/", "_//0", $name), "_"));    }  }  /**   * 解析應(yīng)用類的類名   * @param string $module 模塊名   * @param string $layer 層名 controller model ...   * @param string $name  類名   * @param bool  $appendSuffix   * @return string   */  public static function parseClass($module, $layer, $name, $appendSuffix = false)  {    $name = str_replace(['/', '.'], '//', $name);    $array = explode('//', $name);    $class = self::parseName(array_pop($array), 1) . (App::$suffix || $appendSuffix ? ucfirst($layer) : '');    $path = $array ? implode('//', $array) . '//' : '';    return App::$namespace . '//' . ($module ? $module . '//' : '') . $layer . '//' . $path . $class;  }  /**   * 初始化類的實(shí)例   * @return void   */  public static function clearInstance()  {    self::$instance = [];  }}/** * 作用范圍隔離 * * @param $file * @return mixed */function __include_file($file){  return include $file;}function __require_file($file){  return require $file;}

總結(jié)

以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,謝謝大家對(duì)VeVb武林網(wǎng)的支持。


注:相關(guān)教程知識(shí)閱讀請(qǐng)移步到PHP教程頻道。
發(fā)表評(píng)論 共有條評(píng)論
用戶名: 密碼:
驗(yàn)證碼: 匿名發(fā)表

圖片精選

国产成人毛片| 亚洲精品极品| 男人的天堂免费| 超级砰砰砰97免费观看最新一期| 欧美色欧美亚洲另类二区精品| 成人看片在线观看| 国产精品麻豆一区| 日本24小时在线高清视频| 91精品欧美一区二区三区综合在| 在线视频超级| av中文资源在线| 69堂成人精品视频免费| 妺妺窝人体色www婷婷| 亚洲男人的天堂在线视频| 国产夫妻在线播放| 欧美精品久久久久久久小说| 永久免费av网站| 国产精品白丝jk白祙喷水网站| 精品盗摄女厕tp美女嘘嘘| www.国产在线| 欧美国产日韩中文字幕在线| 亚洲精品一区二区三区在线观看| 国产综合久久久久久久久久久久| 一本大道久久a久久综合婷婷| 91精品丝袜国产高跟在线| 久久久久亚洲av无码网站| 亚洲欧美清纯在线制服| 欧美另类亚洲| 精品精品视频| 欧美bbbbb性bbbbb视频| 懂色一区二区三区av片| 免费日本视频一区| 和岳每晚弄的高潮嗷嗷叫视频| 18岁以下禁止观看的美女视频| 秋霞午夜一区二区三区视频| 一区二区三区日| 亚洲aa在线观看| 在线观看a级片| 宅男在线一区| 欧美 亚洲 另类 激情 另类| 日韩小视频在线观看专区| 韩国一区二区三区美女美女秀| 黄色毛片在线观看| 亚洲综合丁香婷婷六月香| 噜噜噜久久,亚洲精品国产品| 国产精品免费av一区二区| wwww在线观看免费视频| 久久密一区二区三区| jizz欧美性20| 无限资源日本好片| 五月综合激情日本mⅴ| 免费观看一级欧美片| 国产精品免费久久| 岛国av免费在线| 欧美日韩在线播放视频| 日韩伦理视频| 青青草国产精品亚洲专区无| 亚洲成人黄色网| sese视频| 亚洲www永久成人夜色| 欧美在线不卡视频| 性欧美成人播放77777| 精品福利电影| 亚洲第一网站免费视频| 久久99国产综合精品免费| 国产精品成人一区二区三区电影毛片| 拔插拔插华人永久免费| www.亚洲人.com| a4yy在线播放免费观看视频| 在线资源av| 国内精品视频久久| 在线一级观看| 色妞在线综合亚洲欧美| 亚洲一区bb| 男人网站在线观看| heyzo高清中文字幕在线| 日韩国产精品一区二区三区| 一本一道久久a久久精品| 男人日女人下面视频| 欧美男男freegayvideosroom| 免费成人网www| 国产乱码精品一区二区三区日韩精品| 久久久噜噜噜久久中文字幕色伊伊| 国产精品区一区二区三| 欧美激情在线观看视频免费| 日本天码aⅴ片在线电影网站| 在线观看日韩片| 樱桃视频成人在线观看| 亚洲高清视频免费| 99草草国产熟女视频在线| 欧美日韩一区视频| 99久久精品国产一区二区成人| 欧美另类中文字幕| 亚洲五月激情网| 亚洲国产精品久久久久秋霞影院| 这里只有精品66| 制服.丝袜.亚洲.另类.中文| 91在线视频免费看| 在线视频中文字幕一区二区| 久久精品免视看国产成人| www91在线观看| 日本网站免费在线观看| 国产suv一区二区三区88区| 国产酒店精品激情| 欧美黄色成人网| 日韩二区在线观看| 一级黄色录像在线观看| 中文字幕一区二区三中文字幕| 天堂资源在线| 日本19禁啪啪吃奶大尺度| www.youjizz.com亚洲| 精品毛片三在线观看| 国产福利在线| 国产馆av播放| 91在线一区二区三区| 日产中文字幕在线精品一区| 日韩伦理一区二区三区av在线| 国产精品丝袜久久久久久高清| 久久成人亚洲精品| 久久人人爽人人爽人人片av高清| 国产午夜精品视频免费不卡69堂| 国产成人综合网| 高清日韩av电影| 中文在线最新版地址| 成人看片黄a免费看在线| xxxxx99| 欧美黄色性视频| 久久综合伊人77777| 日韩精品电影在线| 99er在线视频| 在线中文字幕av| 亚洲男人的天堂成人| 波多野结衣在线一区二区| 国产欧美自拍视频| jjzz黄色片| 四虎国产精品免费| 欧美日本二区| 欧美一区二区三区精品电影| 中文字幕在线永久在线视频| 久草在线免费资源| 狠狠色丁香久久综合频道| 国产精品无码人妻一区二区在线| 91一区二区三区在线| 国产97色在线| 国产精品玖玖玖在线资源| 国产亚洲一区二区三区在线观看| 999久久久久| 日韩夫妻性生活xx| 午夜影院免费播放| 日本xxx免费| 黄色成人av网| 婷婷av在线| 日本黄色不卡视频| 日韩国产在线播放| 成人国产1314www色视频| 亚洲免费综合| 综合电影一区二区三区| 黄色99视频| 911久久香蕉国产线看观看| 91视视频在线观看入口直接观看www| 成人高清伦理免费影院在线观看| 成人免费毛片在线观看| 国产欧美一区二区精品仙草咪| av一区二区在线播放| 日韩视频123| 日韩三区四区| 欧美丝袜一区| 国产原创视频在线观看| 老司机aⅴ在线精品导航| 国产精品免费一区二区三区四区| 亚洲福利一区二区| 精品国产va久久久久久久| 国产精品a久久久久久| 亚洲成人自拍视频| 欧美色网一区二区| 国产伦精品一区二区三区照片| 欧美电影免费播放| 国产欧美一区二区三区网站| 日韩精品视频中文字幕| 亚洲国产日韩在线观看| 精品成人一区二区三区| 国产91精品一区二区| 青娱乐国产在线| 激情av中文字幕| 国产免费一区视频观看免费| 黄色欧美在线| 一个人看的视频www| 久久亚洲国产成人精品性色| 欧美深夜福利| 亚洲最大的网站| 国产精品探花在线播放| 久蕉在线视频| 日韩精品视频免费在线观看| 日本黄色免费网址| 狠狠色一日本高清视频| 美乳视频一区二区| 大桥未久女教师av一区二区| 国产精品视频最多的网站| 亚洲免费黄色录像| 日韩欧美在线观看免费| 国产精品videossex国产高清| 自拍偷拍欧美激情| 亚洲欧美日韩视频二区| 最新热久久免费视频| 亚洲成人精品在线| 一二三四在线观看视频韩国| 亚洲国产一二三区| 亚洲va久久久噜噜噜久久狠狠| 26uuu精品一区二区| 欧美日韩尤物久久| 美女把尿口扒开让男人桶在线观看| ass精品国模裸体欣赏pics| 一区二区日本视频| 亚洲国产精品久久久久秋霞影院| av在线女优影院| 91亚洲精品在看在线观看高清| 久久久久久久国产精品| 国产成人免费9x9x人网站视频| 黄色av网站在线播放| 亚洲一区三区视频在线观看| 成人av免费在线看| 日本肉体xxxx裸体xxx免费| 国产日韩欧美在线观看视频| 亚州欧美日韩中文视频| 亚洲综合av一区二区三区| 一本色道久久综合亚洲aⅴ蜜桃| 日韩a在线播放| 精品一区毛片| 亚洲18色成人| 九一九一国产精品| 岛国片在线免费观看| 精品剧情v国产在线观看| 在线观影网站| 乡村艳史在线观看| 天天干天天操天天干天天操| 日韩欧美大尺度| 日韩欧美国产成人| 国产福利一区二区精品秒拍| 亚洲一卡久久| 三上悠亚ssⅰn939无码播放| 在线视频在线视频7m国产| 在线观看视频你懂得| 成黄免费在线| 色午夜这里只有精品| 秋霞午夜剧场| 亚洲不卡在线视频| 久久久久久久久久久久久久一区| 精品国产成人在线影院| 欧美视频日韩视频在线观看| 黄色录像免费观看| 亚洲国产欧美另类丝袜| 亚洲一区三区在线观看| 18加网站在线| 在线观看亚洲网站| 久久久精品国产免大香伊| 欧美日韩视频专区在线播放| 日韩久久精品网| 三级网在线观看| 日韩精品视频在线观看视频| 久久精品aaaaaa毛片| 窝窝九色成人影院| 欧美一级片免费| 国产一区视频观看| 国产精品久久久久久9999| 久久人体做爰大胆| 91精品国产综合久久久久久久久| 人妻 日韩精品 中文字幕| 香港三日本三级少妇66| 国产v片免费观看| 涩涩涩视频在线观看| 亚洲欧洲色图综合| 成人h猎奇视频网站| 亚洲男人av| 国产又粗又猛又爽视频| 午夜成人影视| 欧美美女在线直播| 亚洲精品在线二区| 免费偷拍视频网站| 日本a级片视频| 欧美在线激情网| 天堂影院一区二区| 国精产品一区一区三区mba视频| 美女在线观看www| 久久精品视频在线观看榴莲视频| 真实国产乱子伦对白在线| 国产欧美日韩卡一| 日韩视频在线观看一区二区三区| 91精品免费视频| 在线观看中文av| 五月天色一区| 91精品国产aⅴ一区二区| 国产suv精品一区二区33| 欧美熟妇另类久久久久久多毛| 日韩av网址在线观看| 亚洲男人天堂手机在线| 欧美黑人经典片免费观看| 久久99国产综合精品免费| 午夜在线视频免费| 人人妻人人澡人人爽人人精品| 亚洲精品tv久久久久久久久久| 久久精品亚洲一区二区| 蜜臀精品一区二区三区| 国产精品色综合| 欧美黑人xxxxx性受| 日韩电影中文 亚洲精品乱码| 亚洲第一页在线| 污污动漫在线观看| 国产三级视频在线| 亚洲一区二区三区精品视频| 久久密一区二区三区| zzzwww在线看片免费| 麻豆视频免费在线播放| www免费视频观看在线| 国产精品免费久久久久久| 亚洲精品a区| 欧美老女人bb| 欧美男gay| 亚洲一区二区精品在线| 污污视频在线看| 亚洲一级特黄毛片| 成人免费高清在线播放| 日韩欧美在线视频一区二区| 久久久久久有精品国产| 一本色道无码道dvd在线观看| 黄色三级视频在线| 天堂在线中文网| 天堂av网在线| 成人性生交大片免费看在线播放|