程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> 網頁編程 >> PHP編程 >> 關於PHP編程 >> PHP魔術函數__autoload的用法與一些問題

PHP魔術函數__autoload的用法與一些問題

編輯:關於PHP編程

本文章來講述php5的一個新的功能,我們來介紹PHP魔術函數__autoload的用法與一些問題,下面給大家總結了用法過程中出現的一些問題與注意事項。

__autoload()用法

php手冊中的一些講講法

自動加載對象
很多開發者寫面向對象的應用程序時對每個類的定義建立一個 PHP 源文件。一個很大的煩惱是不得不在每個腳本(每個類一個文件)開頭寫一個長長的包含文件列表。

在 PHP 5 中,不再需要這樣了。可以定義一個 __autoload 函數,它會在試圖使用尚未被定義的類時自動調用。通過調用此函數,腳本引擎在 PHP 出錯失敗前有了最後一個機會加載所需的類。

Note:

在 __autoload 函數中拋出的異常不能被 catch 語句塊捕獲並導致致命錯誤。


Note:

如果使用 PHP 的 CLI 交互模式 時,Autoloading 不存在。

Example #1 Autoload 例子

本例嘗試分別從 MyClass1.php 和 MyClass2.php 文件中加載 MyClass1 和 MyClass2 類。

 代碼如下 復制代碼

<?php
function __autoload($class_name) {
    require_once $class_name . '.php';
}

$obj  = new MyClass1();
$obj2 = new MyClass2();
?>

PHP在魔術函數__autoload()方法出現以前,如果你要在一個程序文件中實例化100個對象,那麼你必須用include或者require包含進來100個類文件,或者你把這100個類定義在同一個類文件中——相信這個文件一定會非常大。
但是__autoload()方法出來了,以後就不必為此大傷腦筋了,這個類會在你實例化對象之前自動加載制定的文件。

下邊我們通過一個例子來看一下,具體的使用方法,並在稍後說明使用PHP魔術函數__autoload應該注意些什麼。

 代碼如下 復制代碼


 //定義一個類ClassA,文件名為ClassA.php
class ClassA{
 public  function __construct(){
  echo "ClassA load success!";
 }
}

 //定義一個類ClassB,文件名為ClassB.php,ClassB繼承ClassA
class ClassB extends ClassA {
 public function __construct(){
  //parent::__construct();
  echo "ClassB load success!";
 }
}

定義兩個測試用的類之後,我們來編寫一個含有__autoload()方法的PHP運行程序文件如下:

 代碼如下 復制代碼

 function __autoload($classname){
 $classpath="./".$classname.'.php';
 if(file_exists($classpath)){
  require_once($classpath);
 }
 else{
  echo 'class file'.$classpath.'not found!';
 }
}
 
$newobj = new ClassA();
$newobj = new ClassB();

這個文件的運行是一點問題都沒有的,可見autoload是多麼的好用啊,呵呵……
但是不得不提醒你一下幾個方面是必須要注意的。

1、如果類存在繼承關系(例如:ClassB extends ClassA),並且ClassA不在ClassB所在目錄
利用__autoload魔術函數實例化ClassB的時候就會受到一個致命錯誤:
Fatal error: Class ‘Classd’ not found in ……ClassB.php on line 2,

解決方法:把所有存在extends關系的類放在同一個文件目錄下,或者在實例化一個繼承類的時候在文件中手工包含被繼承的類;

2、另外一個需要注意的是,類名和類的文件名必須一致,才能更方便的使用魔術函數__autoload;

其他需要注意的事情:
3、在CLI模式下運行PHP腳本的話這個方法無效;

4、如果你的類名稱和用戶的輸入有關——或者依賴於用戶的輸入,一定要注意檢查輸入的文件名,例如:.././這樣的文件名是非常危險的。


__autoload的問題

__autoload魔術方法或者你願意稱他為魔術函數,太專一了。當他加載需要包含的類文件時,甚至不管類文件中定義類之外的其它聲明。

開始回放這個機制。

首先我們在創建一個Test.class.php文件,鍵入如下內容

 $publicPara='中共十七大啥時候召開的?';

 代碼如下 復制代碼 class Test{
 public function  __construct(){
  global $publicPara;
  if(isset($publicPara)){
   echo $publicPara;
  }
  else{
   echo "管我啥事兒了?";
  }
 }
}

記得保存這個文件哦!

然後重新建立一個文件命名為do.php鍵入如下內容

 代碼如下 復制代碼  <?php
require_once('Test.class.php');
new Test();

這樣的話輸出的正如我們期望的那樣是:中共十七大啥時候召開的?

但是當你用到__autoload這個魔術方法的時候,問題就出來了

 代碼如下 復制代碼  <?php
function __autoload($classname){
require_once($classname.".class.php");
}
new Test();

這次的輸入居然是:管我啥事兒了?

很顯然他忽略了我們在class外定義的變量,也就是他只把我們需要序列化的類加載進來,而不管其他的聲明了,也僅僅是其他的聲明而已,諸如echo了啥的都還是執行的

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved