本文目錄一覽:
- 1、php魔術方法。。
- 2、怎麼樣在php配置中載入魔術常量
- 3、PHP魔術變數是什麼求詳解,不要去複製然後來粘貼
- 4、請詳細說明一下php5中的魔術方法
- 5、PHP中魔術變數__METHOD__與__FUNCTION__的區別
php魔術方法。。
在 PHP 里,將所有以 __(兩個下劃線)開頭的類方法保留為魔術方法,所以在定義類方法時,除了上述魔術方法,建議不要以 __ 為前綴。
魔術方法的引入是對面向對象編程思想的進一步實現,重載就是通過魔術方法來實現的。
PHP 魔術方法主要有:
__construct(), __destruct(), __call(), __callStatic(), __get(), __set(), __isset(), __unset(),__sleep(), __wakeup(), __toString(), __invoke(), __set_state(), __clone() 和 __debugInfo() 等。
最常見的是__construct(), __destruct(),__get(), __set()等:
__construct和__destruct是類的構造函數和析構函數。
怎麼樣在php配置中載入魔術常量
PHP中,有不少很好用的魔術變數,用好之,能事半功倍,下面小結之:
1 _LINE_
顯示當前代碼的行數:
echo “This is line number : “. __LINE__;
2 _FILE_
顯示當前文件的路徑
3 _METHOD_
顯示當前方法的名,比如
class Magicconstant
{
function __construct()
{
echo “This is function a”;
}
function b()
{
echo “br/”;
echo “This is function b”;
echo “br/”;
echo __METHOD__;
}
}
$cm = new Magicconstant();
$cm-b();
顯示
This is function a
This is function b
Magicconstant::b
4 _FUNCTION_
顯示當前所在函數的名
function b()
{
echo “br/”;
echo “This is function b”;
echo “br/”;
echo __FUNCTION__;
}
輸出:
This is function b
Magicconstant::b
5 _DIR_
顯示當前目錄名,如
echo “The directory name is : “. __DIR__;
輸出:
The directory name is : D:\wamp\www
6 _CLASS_
顯示當前的類
class Magicconstant
{
function __construct()
{
echo “The class name is : “.__CLASS__;
}
}
$cm = new Magicconstant();
顯示:
The class name is : Magicconstant
7 _NAMESPACE_
顯示當前命名空間
namespace MagicConstant
{
echo “The namespace is : “.__NAMESPACE__;
}
輸出:
The namespace is : MagicConstant
8 _sleep_
_sleep_用在將類序列化之前,
?php
class User
{
public $userName = ”;
public $userAddress = ”;
public $userPhone = ”;
public function setName($name=”)
{
$this-userName = $name;
}
function setAddress($address=”)
{
$this-userAddress = $address;
}
function setPhone($phone=”)
{
$this-userPhone = $phone;
}
function __sleep()
{
return array(‘userAddress’,’userPhone’);
// This will serialize only address and Phone number but not Name
}
}
?
?php
$User = new User();
$User-setName(‘Avinash’);
$User-setAddress(‘Address Here’);
$User-setPhone(‘1234567890’);
$serialData = serialize($User);
echo $serialData;
?
OutPut: O:4:”User”:2:{s:11:”userAddress”;s:12:”Address Here”;s:9:”userPhone”;s:10:”1234567890″;}
程序運行時, serialize() 檢查類中是否有 __sleep() ,如果有,則該函數將在任何序列化之前運行. 該函數必須返回一個需要進行序列化保存的成員屬性數組,並且只序列化該函數返回的這些成員屬性. 該函數有兩個作用: 第一. 在序列化之前,關閉對象可能具有的任何資料庫連接等. 第二. 指定對象中需要被序列化的成員屬性,如果某個屬性比較大而不需要儲存下來,可以不把它寫進__sleep()要返回的數組中,這樣該屬性就不會被序列化
又如:
class Test {
public $mySecret; //我的秘密不想讓人知道
public function __construct($secret) {
$this-mySecret = $secret;
}
public function __sleep() {
$this-mySecret = “你休想知道我的秘密!”;
return array(‘mySecret’); //一定要返回變數,不然返回的是個空,所以序列化也就沒有東西了。
}
}
$test = new Test(“我的心裡話 我愛某某某”);
echo serialize($test); //輸出 O:4:”Test”:1:{s:8:”mySecret”;s:28:”你休想知道我的秘密!”;}
_wakeup_
用在反序列化時:
?php
class User
{
public $userName = ”;
public $userAddress = ”;
public $userPhone = ”;
public function setName($name=”)
{
$this-userName = $name;
}
function setAddress($address=”)
{
$this-userAddress = $address;
}
function setPhone($phone=”)
{
$this-userPhone = $phone;
}
function __sleep()
{
return array(‘userAddress’,’userPhone’);
// This will serialize only address and Phone number but not Name
}
function __wakeup()
{
echo “In Wakeup function”.”br/”;
}
}
?
?php
$User = new User();
$User-setName(‘Avinash’);
$User-setAddress(‘Address Here’);
$User-setPhone(‘1234567890’);
$serialData = serialize($User);
echo $serialData;
echo “br/”;
var_dump(unserialize($serialData));
?
Output:
In Wakeup function
object(User)#2 (3) {
[“userName”]=
string(0) “”
[“userAddress”]=
string(12) “Address Here”
[“userPhone”]=
string(10) “1234567890”
}
//以上信息轉自 不代表個人觀點 僅供參考 若滿意 請採納本答案
PHP魔術變數是什麼求詳解,不要去複製然後來粘貼
魔術變數就是指那些根據不同環境有不同的取值
比如
__FILE__
就是一個典型的魔術變數,當它出現在a.php文件里時它的值就是a.php的具體路徑,出現在b.php文件里就是b.php的具體路徑
其他的一些魔術變數還有__FUNCTION__, __CLASS__
總之這些變數的值往往不是固定的,變來變去的,很神奇,像魔術一樣,故叫做魔術變數
請詳細說明一下php5中的魔術方法
PHP處理對象部分的內核完全重新開發過,提供更多功能的同時也提高了性能。在以前版本的php中,處理對象和處理基本類型(數字,字元串)的方式是一樣的。這種方式的缺陷是:當將對象賦值給一個變數時,或者通過參數傳遞對象時,對象將被完全拷貝一份。在新的版本里,上述操作將傳遞引用(可以把引用理解成對象的標識符),而非值。
很多PHP程序員可能甚至沒有察覺到老的對象處理方式。事實上,大多數的php應用都可以很好地運行。或者僅僅需要很少的改動。
私有和受保護成員
PHP5引入了私有和受保護成員變數的概念。我們可以用它來定義類成員的可見性。
例子
受保護成員可以被子類訪問, 而私有成員只能被類本身訪問。
代碼:——————————————————————————–
?php
class MyClass {
private $Hello = “Hello, World!\n”;
protected $Bar = “Hello, Foo!\n”;
protected $Foo = “Hello, Bar!\n”;
function printHello() {
print “MyClass::printHello() ” . $this-Hello;
print “MyClass::printHello() ” . $this-Bar;
print “MyClass::printHello() ” . $this-Foo;
}
}
class MyClass2 extends MyClass {
protected $Foo;
function printHello() {
MyClass::printHello(); /* Should print */
print “MyClass2::printHello() ” . $this-Hello; /* Shouldn’t print out anything */
print “MyClass2::printHello() ” . $this-Bar; /* Shouldn’t print (not declared)*/
print “MyClass2::printHello() ” . $this-Foo; /* Should print */
}
}
$obj = new MyClass();
print $obj-Hello; /* Shouldn’t print out anything */
print $obj-Bar; /* Shouldn’t print out anything */
print $obj-Foo; /* Shouldn’t print out anything */
$obj-printHello(); /* Should print */
$obj = new MyClass2();
print $obj-Hello; /* Shouldn’t print out anything */
print $obj-Bar; /* Shouldn’t print out anything */
print $obj-Foo; /* Shouldn’t print out anything */
$obj-printHello();
?
——————————————————————————–
私有方法和受保護方法
PHP5也引入了私有方法和受保護方法的概念。
例子:
代碼:——————————————————————————–
?php
class Foo {
private function aPrivateMethod() {
echo “Foo::aPrivateMethod() called.\n”;
}
protected function aProtectedMethod() {
echo “Foo::aProtectedMethod() called.\n”;
$this-aPrivateMethod();
}
}
class Bar extends Foo {
public function aPublicMethod() {
echo “Bar::aPublicMethod() called.\n”;
$this-aProtectedMethod();
}
}
$o = new Bar;
$o-aPublicMethod();
?
——————————————————————————–
以前的不使用類的老代碼,沒有訪問修飾符(public, protected, private)的代碼可以不經改動運行。
抽象類和抽象方法
Php5也引入了抽象類和抽象方法的概念。抽象方法只是聲明了方法的簽名並不提供它的實現。包含抽象方法的類必須被聲明成抽象類。
例子:
代碼:——————————————————————————–
?php
abstract class AbstractClass {
abstract public function test();
}
class ImplementedClass extends AbstractClass {
public function test() {
echo “ImplementedClass::test() called.\n”;
}
}
$o = new ImplementedClass;
$o-test();
?
——————————————————————————–
抽象類不能被實例化。以前的不使用抽象類的老代碼可以不經改動運行。
介面
Php5引入了介面。一個類可以實現多個介面。
例子:
代碼:——————————————————————————–
?php
interface Throwable {
public function getMessage();
}
class MyException implements Throwable {
public function getMessage() {
// …
}
}
?
——————————————————————————–
以前的不使用介面的老代碼可以不經改動運行
類的型別提示
PHP5依然是弱類型的,不過在定義函數參數時,可以使用類的型別提示來聲明期望傳入的對象類型
Example
代碼:——————————————————————————–
?php
interface Foo {
function a(Foo $foo);
}
interface Bar {
function b(Bar $bar);
}
class FooBar implements Foo, Bar {
function a(Foo $foo) {
// …
}
function b(Bar $bar) {
// …
}
}
$a = new FooBar;
$b = new FooBar;
$a-a($b);
$a-b($b);
?
——————————————————————————–
和其他強類型語言一樣,php5類的型別提示在運行期間檢查而非編譯期間檢查。即:
代碼:——————————————————————————–
?php
function foo(ClassName $object) {
// …
}
?
和下面的代碼是一樣的:
?php
function foo($object) {
if (!($object instanceof ClassName)) {
die(“Argument 1 must be an instance of ClassName”);
}
}
?
——————————————————————————–
這個語法只適用於類,不適用於內建類型。
Final
PHP 5 引入了final關鍵字來聲明final成員和final方法。final成員和final方法不能被子類覆蓋。
Example
代碼:——————————————————————————–
?php
class Foo {
final function bar() {
// …
}
}
?
——————————————————————————–
更進一步,可以把類聲明成final。將類聲明成final可以阻止這個類被繼承。final類裡面的方法預設地都是final的,無需再聲明一次。
Example
代碼:——————————————————————————–
?php
final class Foo {
// class definition
}
// the next line is impossible
// class Bork extends Foo {}
?
——————————————————————————–
屬性不能定義成為final.
以前的不使用final的老代碼可以不經改動運行.
對象克隆
Php4沒有提供一種機制來讓用戶自己定義複製構造子(copy constructor)控制對象的複製過程。Php4做二進位的拷貝,因而很精確地複製了對象的所有屬性。
精確地複製對象的所有屬性可能並不是我們一直想要的。有個例子可以很好地說明我們確實需要複製構造子:比如一個GTK Window的對象 a。 a持有它所需要的全部資源。當複製的這個GTK Window到對象b時候,我們更希望b持有新的資源對象。再舉個例子:對象a包含了一個對象c, 當你把對象a 複製到對象c的時候。我們可能更希望對象b包含一個新的對象c的copy, 而不是一個對象c的引用。(譯者註:這裡所說的就是淺克隆和深克隆。)
對象的複製是通過clone這個關鍵字達到的(Clone調用被克隆對象的__clone()方法)。對象的__clone方法不能夠直接被調用。
代碼:——————————————————————————–
?php
$copy_of_object = clone $object;
?
——————————————————————————–
當developer創建對象的一份拷貝的時候,php5將會檢查 __clone()方法是否存在。如果不存在,那麼它就會呼叫預設的__clone()方法,複製對象的所有屬性。如果__clone()方法已經定義過,那麼_clone()方法就會負責設置新對象的屬性。為了方便起見,Engine會預設地複製所有的屬性。所以在__clone()方法中,只需要覆蓋那些需要更改的屬性就可以了。如下:
Example
代碼:——————————————————————————–
?php
class MyCloneable {
static $id = 0;
function MyCloneable() {
$this-id = self::$id++;
}
function __clone() {
$this-address = “New York”;
$this-id = self::$id++;
}
}
$obj = new MyCloneable();
$obj-name = “Hello”;
$obj-address = “Tel-Aviv”;
print $obj-id . “\n”;
$obj_cloned = clone $obj;
print $obj_cloned-id . “\n”;
print $obj_cloned-name . “\n”;
print $obj_cloned-address . “\n”;
?
——————————————————————————–
統一構造函數
Php5允許開發者聲明一個類的構造方法。擁有構造方法的類在每次創建新的對象的時候都會呼叫這個方法,因此構造方法適合對象在被使用之前的初始化工作
Php4中,構造方法的名稱和類的名稱一樣。考慮到從子類構造方法呼叫父類構造方法的情況是非常普遍的,而將類從一個繼承體系中搬遷引起的父類變更就常常導致需要更改類的構造方法,php4的做法顯然是不太合理的。
Php5引入了一個聲明構建函數的標準方法: __construct().如下:
Example
代碼:——————————————————————————–
?php
class BaseClass {
function __construct() {
print “In BaseClass constructor\n”;
}
}
class SubClass extends BaseClass {
function __construct() {
parent::__construct();
print “In SubClass constructor\n”;
}
}
$obj = new BaseClass();
$obj = new SubClass();
?
——————————————————————————–
為保持向後的兼容性,如果php5不能夠找到 __construct(),它會尋找老式的構造方法,即與類同名的方法。簡單的說,只有當老代碼里包含了一個__construct()方法的時候,才存在一個兼容性的問題。
析構方法
對於面向對象的編程來說,可以定義析構方法是非常有用的一個功能。析構方法可以用來記錄調試信息,關閉資料庫連接等等一些清除收尾的工作。Php4中沒有析構方法,儘管php4已經支持可以註冊一個函數以便請求結束的時候被調用。
Php5引進的析構方法的概念和其他面向對象的語言(比如java)是一致的。當指向這個對象的最後一個引用被銷毀的時候,析構方法被調用,調用完成後釋放內存。注意:析構方法不接受任何參數。
Example
代碼:——————————————————————————–
?php
class MyDestructableClass {
function __construct() {
print “In constructor\n”;
$this-name = “MyDestructableClass”;
}
function __destruct() {
print “Destroying ” . $this-name . “\n”;
}
}
$obj = new MyDestructableClass();
?
——————————————————————————–
和構建方法一樣,父類的析構方法也不會被隱含調用。子類可以在自己的析構方法通過調用parent::__destruct()來顯式地調用它。
Constants
Php5引入了class級別的常量。
代碼:——————————————————————————–
?php
class Foo {
const constant = “constant”;
}
echo “Foo::constant = ” . Foo::constant . “\n”;
?
——————————————————————————–
老的沒有使用const的代碼仍然正常運行。
Exceptions
Php4沒有異常控制。Php5引入了和其它語言(java)相似的異常控制模式。應該注意的是php5裡面支持捕捉全部異常,但是不支持finally子句。
在catch語句塊裡面,可以重新拋出異常。也可以有多個catch語句,在這種情況下,被捕捉到的異常從上往下依次比較和catch語句比較異常,第一個類型匹配的catch語句將會被執行。如果一直搜索到底還沒有發現匹配的catch子句,則尋找下一個try/catch語句。最後不能捕捉的異常將被顯示出來。如果異常被捕捉,那麼程序會接著catch語句塊的下面開始執行。
Example
代碼:——————————————————————————–
?php
class MyException {
function __construct($exception) {
$this-exception = $exception;
}
function Display() {
print “MyException: $this-exception\n”;
}
}
class MyExceptionFoo extends MyException {
function __construct($exception) {
$this-exception = $exception;
}
function Display() {
print “MyException: $this-exception\n”;
}
}
try {
throw new MyExceptionFoo(‘Hello’);
}
catch (MyException $exception) {
$exception-Display();
}
catch (Exception $exception) {
echo $exception;
}
?
——————————————————————————–
上面的例子表明可以定義一個並不繼承自 Exception的異常類,但是,最好還是從Exception繼承並定義自己的異常。這是因為系統內建的Exception類能夠收集到很多有用的信息, 而不繼承它的異常類是得不到這些信息的。下面的php代碼模仿了系統內建Exception類。每個屬性後面都加了注釋。每個屬性都有一個getter,由於這些getter方法經常被系統內部處理調用,所以這些方法被標明了final。
Example
代碼:——————————————————————————–
?php
class Exception {
function __construct(string $message=NULL, int code=0) {
if (func_num_args()) {
$this-message = $message;
}
$this-code = $code;
$this-file = __FILE__; // of throw clause
$this-line = __LINE__; // of throw clause
$this-trace = debug_backtrace();
$this-string = StringFormat($this);
}
protected $message = ‘Unknown exception’; // exception message
protected $code = 0; // user defined exception code
protected $file; // source filename of exception
protected $line; // source line of exception
private $trace; // backtrace of exception
private $string; // internal only!!
final function getMessage() {
return $this-message;
}
final function getCode() {
return $this-code;
}
final function getFile() {
return $this-file;
}
final function getTrace() {
return $this-trace;
}
final function getTraceAsString() {
return self::TraceFormat($this);
}
function _toString() {
return $this-string;
}
static private function StringFormat(Exception $exception) {
// … a function not available in PHP scripts
// that returns all relevant information as a string
}
static private function TraceFormat(Exception $exception) {
// … a function not available in PHP scripts
// that returns the backtrace as a string
}
}
?
——————————————————————————–
如果我們定義的一異常類都是繼承自Exception基類
無兼容性問題。老的代碼不會受到這一特性的影響。
Dereferencing objects returned from functions
Php4中不能再次引用函數返回的對象以進一步呼叫返回對象的方法,而php5是可以的。
代碼:——————————————————————————–
?php
class Circle {
function draw() {
print “Circle\n”;
}
}
class Square {
function draw() {
print “Square\n”;
}
}
function ShapeFactoryMethod($shape) {
switch ($shape) {
case “Circle”:
return new Circle();
case “Square”:
return new Square();
}
}
ShapeFactoryMethod(“Circle”)-draw();
ShapeFactoryMethod(“Square”)-draw();
?
——————————————————————————–
靜態成員變數能夠被初始化。
Example
代碼:——————————————————————————–
?php
class foo {
static $my_static = 5;
public $my_prop = ‘bla’;
}
print foo::$my_static;
$obj = new foo;
print $obj-my_prop;
?
——————————————————————————–
靜態方法
PHP 5 引入了靜態方法,可以在不實例化類的情況下呼叫靜態方法。
Example
代碼:——————————————————————————–
?php
class Foo {
public static function aStaticMethod() {
// …
}
}
Foo::aStaticMethod();
?
——————————————————————————–
偽變數$this不能夠在靜態方法方法中使用。
instanceof
Php5引入了instanceof關鍵字,允許用它來測試一個對象是一個類的實例,或者是一個派生類的實例,或者實現了某個介面
Example
代碼:——————————————————————————–
?php
class baseClass { }
$a = new baseClass;
if ($a instanceof baseClass) {
echo “Hello World”;
}
?
——————————————————————————–
Static function variables
現在,靜態變數在編譯階段處理。因此程序員可以通過引用為靜態變數賦值。這可以改善性能,不過,不能夠使用對靜態變數的間接引用了。
按引用傳遞的函數參數現在也可以設置預設值了。
Example
代碼:——————————————————————————–
?php
function my_function($var = null) {
if ($var === null) {
die(“$var needs to have a value”);
}
}
?
——————————————————————————–
__autoload()
__autoload() 攔截函數在一個未聲明的類被初始化的時候自動調用。該類的名字會被自動傳遞給__autoload()函數。而__autoload()也只有這麼唯一的一個參數。
Example
代碼:——————————————————————————–
?php
function __autoload($className) {
include_once $className . “.php”;
}
$object = new ClassName;
?
——————————————————————————–
可重載的方法呼叫和屬性訪問
方法呼叫和屬性訪問都能夠通過__call, __get() and __set()方法重載。
Example: __get() and __set()
代碼:——————————————————————————–
?php
class Setter {
public $n;
public $x = array(“a” = 1, “b” = 2, “c” = 3);
function __get($nm) {
print “Getting [$nm]\n”;
if (isset($this-x[$nm])) {
$r = $this-x[$nm];
print “Returning: $r\n”;
return $r;
} else {
print “Nothing!\n”;
}
}
function __set($nm, $val) {
print “Setting [$nm] to $val\n”;
if (isset($this-x[$nm])) {
$this-x[$nm] = $val;
print “OK!\n”;
} else {
print “Not OK!\n”;
}
}
}
$foo = new Setter();
$foo-n = 1;
$foo-a = 100;
$foo-a++;
$foo-z++;
var_dump($foo);
?
——————————————————————————–
Example: __call()
代碼:——————————————————————————–
?php
class Caller {
private $x = array(1, 2, 3);
function __call($m, $a) {
print “Method $m called:\n”;
var_dump($a);
return $this-x;
}
}
$foo = new Caller();
$a = $foo-test(1, “2”, 3.4, true);
var_dump($a);
?
——————————————————————————–
迭代
當和foreach一起使用對象的時候,迭代的方式被重載過了。預設的行為是迭代類的所有屬性。
Example
代碼:——————————————————————————–
?php
class Foo {
public $x = 1;
public $y = 2;
}
$obj = new Foo;
foreach ($obj as $prp_name = $prop_value) {
// using the property
}
?
——————————————————————————–
一個類的所有對象都能夠被迭代瀏覽到, 如果這個類實現了一個空的介面:Traversable. 換句話說,實現了Traversable介面的類可以和foreach一起使用。
介面 IteratorAggregate 和Iterator允許指定類的對象在代碼中如何迭代。IteratorAggregate介面有一個方法:getIterator() 必須返回一個數組
Example
代碼:——————————————————————————–
?php
class ObjectIterator implements Iterator {
private $obj;
private $num;
function __construct($obj) {
$this-obj = $obj;
}
function rewind() {
$this-num = 0;
}
function valid() {
return $this-num $this-obj-max;
}
function key() {
return $this-num;
}
function current() {
switch($this-num) {
case 0: return “1st”;
case 1: return “2nd”;
case 2: return “3rd”;
default: return $this-num.”th”;
}
}
function next() {
$this-num++;
}
}
class Object implements IteratorAggregate {
public $max = 3;
function getIterator() {
return new ObjectIterator($this);
}
}
$obj = new Object;
// this foreach …
foreach($obj as $key = $val) {
echo “$key = $val\n”;
}
// matches the following 7 lines with the for directive.
$it = $obj-getIterator();
for($it-rewind(); $it-hasMore(); $it-next) {
$key = $it-current();
$val = $it-key();
echo “$key = $val\n”;
}
unset($it);
?
——————————————————————————–
新的__toString方法
可以通過覆蓋__toString方法來控制對象到字元串的轉換。
Example
代碼:—————————————
PHP中魔術變數__METHOD__與__FUNCTION__的區別
本文實例講述了PHP中魔術變數__METHOD__與__FUNCTION__的區別,分享給大家供大家參考。具體分析如下:
__METHOD__類的方法名(PHP
5.0.0
新加)。返回該方法被定義時的名字(區分大小寫)。
__FUNCTION__函數名稱(PHP
4.3.0
新加)。自
PHP
5
起本常量返回該函數被定義時的名字(區分大小寫)。在
PHP
4
中該值總是小寫字母的。
文檔解釋都是說返回函數(方法)的名字。區別在於:
__FUNCTION__只是返回方法的名字;
__METHOD__返回類的名字和方法的名字。
?php
class
Test{
public
function
doit(){
echo
__FUNCTION__;
}
public
function
doitAgain(){
echo
__METHOD__;
}
}
$obj
=
new
Test();
$obj-doit();
echo
‘br’;
$obj-doitAgain();
?
輸出結果為:
doit
Test::doitAgain
希望本文所述對大家的PHP程序設計有所幫助。
原創文章,作者:小藍,如若轉載,請註明出處:https://www.506064.com/zh-tw/n/153524.html