- 論壇徽章:
- 0
|
原創(chuàng),轉(zhuǎn)載請聲明出處 http://blog.chinaunix.net/u/27731/showart.php?id=1328279
作者:流水孟春 121169238(at)qq.com
1、動態(tài)語言
動態(tài)語言就是能夠在運行時改變程序結(jié)構(gòu)和變量類型的語言。例如:新的類和對象可以被加載和創(chuàng)建,新的函數(shù)或方法可以加入和去除等等。
動態(tài)語言都被設(shè)計成弱類型,也就是說變量被賦值以后才能確定它的數(shù)據(jù)類型,當(dāng)代碼在實際執(zhí)行時,才會檢測變量是否被非法使用。
2、PHP動態(tài)特性:
特性1、弱類型變量
PHP變量是弱類型變量就意味著,我們不需要聲明變量的類型,在運行時自動檢測變量的類型,并且可以認(rèn)為改動變量的類型。
?php
// PHP弱類型變量例子
$test = 1;
print gettype($test) . ": $test\n";
$test = 1.23456789;
print gettype($test) . ": $test\n";
$test = 'test'; // 變量自動改變類型
print gettype($test) . ": $test\n";
$test = array('t'=>'HI,I m from an array!');
settype($test, 'object'); // 改變變量的類型
print gettype($test) . ": {$test->t}\n";
// 測試變量類型
if(is_object($test)) {
print "Test Type: My Type Is object.\n";
} else {
print gettype($test);
}
?>
弱類型并不意味著代碼不安全,或者不健壯。極限編程已經(jīng)給成為一種軟件開發(fā)方法。這個方法側(cè)重測試,使用全面的單元測試方案來驅(qū)動開發(fā)過程。通過不同環(huán)境
下執(zhí)行所編寫的代碼,就可以保證代碼的安全性和健壯性。經(jīng)驗證明,在弱類型語言中,綜合運用弱類型和單元測試通常比傳統(tǒng)系統(tǒng)編程語言的類型檢查更好(請參
考《Thinking in Java》作者Bruce Eckel的博客http://mindview.net/WebLog/log-0025)。
特性2、可變變量
一個變量的變量名可以動態(tài)的設(shè)置和使用
?php
// 可變變量例子
$var = 'hi';
$$var = 'hello';
print $var;
print $$var;
print $hi; // 等價于上一行
?>
特性3、變量函數(shù)
這意味著如果一個變量名后有圓括號,PHP 將尋找與變量的值同名的函數(shù),并且將嘗試執(zhí)行它。除了別的事情以外,這個可以被用于實現(xiàn)回調(diào)函數(shù),函數(shù)表等等。
?php
// 變量函數(shù)例子
// 無參數(shù)函數(shù)
function a(){
print "i'm a\n";
}
// 有參數(shù)函數(shù)
function b($param) {
print "i'm b, param: $param\n";
}
$x = 'a';
$x();
$x = 'b';
$x('xxxxxxxxxxx');
?>
特性4、可變長度參數(shù)列表
調(diào)用函數(shù)的時候,函數(shù)的參數(shù)個數(shù)可以不一樣
?php
function foo() {
$numargs = func_num_args();
print "Number of arguments: $numargs\n";
print "the third argument: " . func_get_arg(2) . "\n";
}
foo(1, 2, '...@...');
foo(1, 2, '############', 4);
?>
特性5,從數(shù)組中導(dǎo)出變量
把數(shù)組中的數(shù)據(jù)復(fù)制出變量,將鍵名當(dāng)作變量名,值作為變量的值。
?php
// 例1
$a = array('x' => 100, 'y'=> 200, 'z' => 300);
// 從數(shù)組中將變量導(dǎo)入到當(dāng)前的符號表
extract($a); // 和extract相反的函數(shù)是compact()
print "$x $y $z";
?>
?php
// 例2
$a = array(100, 200, 300);
list($x, , $z) = $a;
print "$x $z";
?>
特性6,用一個數(shù)組的值作為其鍵名,另一個數(shù)組的值作為其值,創(chuàng)建一個數(shù)組
?php
$a = array('green', 'red', 'yellow');
$b = array('avocado', 'apple', 'banana');
$c = array_combine($a, $b);
print_r($c);
?>
特性7,動態(tài)創(chuàng)建函數(shù)
?php
// lambda函數(shù)
$newfunc = create_function('$a,$b', 'return "$a + $b = " . $a + $b;'); // 創(chuàng)建了一個匿名函數(shù)
print "\nNew anonymous function: $newfunc\n";
print $newfunc(2, 4);
print "\n";
// 效果類似于
function test($a,$b){
return "$a + $b = " . $a + $b;
}
print test(2, 4);
?>
特性8、自動加載對象
可以定義一個 __autoload 函數(shù),它會在試圖使用尚未被定義的類時自動調(diào)用。
?php
// ClassA.php, 這段代碼寫在ClassA.php
class A {
function __construct() {
print 'yeah!';
}
}
?>
?php
function __autoload($className) {
require_once $className . '.php';
}
// 這段代碼寫在b.php
new A(); // 程序運行到這里的時候,A類未定義,將自動調(diào)用__autoload()函數(shù)
?>
特性9、__get和__set代替所有對屬性變量數(shù)組的訪問
?php
class Setter{
public $n;
private $x = array("a" => 1, "b" => 2, "c" => 3);
private function __get($nm) {
echo "Getting [$nm]\n";
if (isset($this->x[$nm])) {
$r = $this->x[$nm];
print "Returning: $r\n";
return $r;
} else {
echo "Nothing!\n";
}
}
private function __set($nm, $val) {
echo "Setting [$nm] to $val\n";
if (isset($this->x[$nm])) {
$this->x[$nm] = $val;
echo "OK!\n";
} else {
echo "Not OK!\n";
}
}
private function __isset($nm) {
echo "Checking if $nm is set\n";
return isset($this->x[$nm]);
}
private function __unset($nm) {
echo "Unsetting $nm\n";
unset($this->x[$nm]);
}
}
$foo = new Setter();
$foo->n = 1;
$foo->a = 100;
$foo->a++;
$foo->z++;
var_dump(isset($foo->a)); //true
unset($foo->a);
var_dump(isset($foo->a)); //false
// this doesn't pass through the __isset() method
// because 'n' is a public property
var_dump(isset($foo->n));
var_dump($foo);
?>
特性10、自定義未定義的方法
你調(diào)用未定義方法時,方法名和方法接收的參數(shù)將會傳給__call方法
?php
class Caller {
private $x = array(1, 2, 3);
private function __call($m, $a) {
print "Method $m called:\n";
print_r($a);
return $this->x;
}
}
$foo = new Caller();
$a = $foo->test(1, "2", 3.4, true);
print_r($a);
?>
特性11、自定義錯誤、異常處理
設(shè)置使用自定義錯誤或異常處理函數(shù)后,當(dāng)發(fā)生錯誤或異常時將調(diào)用自定義的處理函數(shù)代替系統(tǒng)錯誤處理函數(shù)。
?php
set_error_handler('error_handler'); // 設(shè)置錯誤時調(diào)用的自定義處理錯誤函數(shù),設(shè)置自定義異常處理函數(shù)為set_exception_handler()
print $a/0; // 這里邏輯錯了,調(diào)用error_handler()報錯
function error_handler($errno, $message, $filename, $line) {
if ($errno & (E_ALL ^ E_NOTICE)) {
$types = array(
1 => 'error',
2 => 'warning',
4 => 'parse error',
8 => 'notice',
16 => 'core error',
32 => 'core warning',
64 => 'compile error',
128 => 'compile warning',
256 => 'user error',
512 => 'user warning',
1024 => 'user notice',
2048 => 'strict warning'
);
print "\n~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~\n";
print $types[$errno] .': '. $message .' in '. $filename .' on line '. $line .'.';
print "\n~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~\n";
exit;
}
}
?>
特性12、高可配置性
可修改范圍為
PHP_INI_USER 的 配置選項可在用戶的PHP腳本或Windows注冊表中設(shè)置
PHP_INI_PERDIR 的 配置選項可在 php.ini, .htaccess 或 httpd.conf 中設(shè)置
PHP_INI_SYSTEM 的 配置選項可在 php.ini or httpd.conf 中設(shè)置
PHP_INI_ALL 的 配置選項可在各處設(shè)置
我們要熟悉在php.ini中修改php的配置,還會經(jīng)常在php腳本中修改腳本中可修改的配置項。
可以使用 ini_set()函數(shù)或?qū)iT修改運行時配置的函數(shù)來改變可修改范圍為PHP_INI_ALL
?php
// 修改php的配置參數(shù)
ini_set('session.save_path', 'D:/temp'); // 修改配置讓腳本把session文件保存到D:/temp目錄
?>
特性13、代碼中執(zhí)行php腳本
特性14、php的工作模型
(這個特性詳細(xì)的在http://blog.csdn.net/phphot/archive/2008/02/19/2105600.aspx)
作為一種純解釋型語言,PHP腳本在每次被解釋時進(jìn)行初始化,在解釋完畢后終止運行。這種運行是互相獨立的,每一次請求都會創(chuàng)建一個單獨的進(jìn)程或線程,來解釋相應(yīng)的頁面文件。頁面創(chuàng)建的變量和其他對象,都只在當(dāng)前的頁面內(nèi)部可見,無法跨越頁面訪問舊電腦回收。
在終止運行后,頁面中申請的、沒有被代碼顯式釋放的外部資源,包括內(nèi)存、數(shù)據(jù)庫連接、文件句柄、Socket連接等,都會被強行釋放。
也就是說,PHP無法在語言級別直接訪問跨越頁面的變量,也無法創(chuàng)建駐留內(nèi)存的對象。
PHP這種獨特的工作模型的優(yōu)勢在于,基本上解決了令人頭疼的資源泄漏問題。Web應(yīng)用的特點是大量的、短時間的并發(fā)處理,對各種資源的申請和釋放工作非常頻繁,很容易導(dǎo)致泄漏。
但是,這種機制的缺點也非常明顯。最直接的后果是,PHP在語言級別無法實現(xiàn)跨頁面的緩沖機制。
本文來自ChinaUnix博客,如果查看原文請點:http://blog.chinaunix.net/u/27731/showart_1328279.html |
|